forked from OSchip/llvm-project
Recommit "[llvm-objcopy][MachO] Implement a layout algorithm for executables"
Summary: The layout algorithm for relocatable objects and for executable are somewhat different. This patch implements the latter one based on the algorithm in LLD (MachOFileLayout). Reviewers: alexshap, rupprecht, jhenderson Reviewed By: alexshap Subscribers: jakehehrlich, abrachet, llvm-commits Tags: #llvm Differential Revision: https://reviews.llvm.org/D65539 llvm-svn: 369301
This commit is contained in:
parent
55ccd16354
commit
12bd490427
|
@ -0,0 +1,291 @@
|
||||||
|
## This test verifies that llvm-objcopy copies an executable properly. It
|
||||||
|
## uses llvm-readobj instead of cmp because some parts of the object
|
||||||
|
## (e.g., the string table) are not identical; the output file is correct but
|
||||||
|
## some offsets differ from the input file.
|
||||||
|
# RUN: yaml2obj %s > %t
|
||||||
|
# RUN: llvm-objcopy %t %t2
|
||||||
|
# RUN: llvm-readobj --file-headers --sections %t2 | FileCheck %s
|
||||||
|
|
||||||
|
--- !mach-o
|
||||||
|
FileHeader:
|
||||||
|
magic: 0xFEEDFACF
|
||||||
|
cputype: 0x01000007
|
||||||
|
cpusubtype: 0x80000003
|
||||||
|
filetype: 0x00000002
|
||||||
|
ncmds: 15
|
||||||
|
sizeofcmds: 976
|
||||||
|
flags: 0x00200085
|
||||||
|
reserved: 0x00000000
|
||||||
|
LoadCommands:
|
||||||
|
- cmd: LC_SEGMENT_64
|
||||||
|
cmdsize: 72
|
||||||
|
segname: __PAGEZERO
|
||||||
|
vmaddr: 0
|
||||||
|
vmsize: 4294967296
|
||||||
|
fileoff: 0
|
||||||
|
filesize: 0
|
||||||
|
maxprot: 0
|
||||||
|
initprot: 0
|
||||||
|
nsects: 0
|
||||||
|
flags: 0
|
||||||
|
- cmd: LC_SEGMENT_64
|
||||||
|
cmdsize: 232
|
||||||
|
segname: __TEXT
|
||||||
|
vmaddr: 4294967296
|
||||||
|
vmsize: 4096
|
||||||
|
fileoff: 0
|
||||||
|
filesize: 4096
|
||||||
|
maxprot: 7
|
||||||
|
initprot: 5
|
||||||
|
nsects: 2
|
||||||
|
flags: 0
|
||||||
|
Sections:
|
||||||
|
- sectname: __text
|
||||||
|
segname: __TEXT
|
||||||
|
addr: 0x0000000100000F70
|
||||||
|
size: 58
|
||||||
|
offset: 0x00000F70
|
||||||
|
align: 4
|
||||||
|
reloff: 0x00000000
|
||||||
|
nreloc: 0
|
||||||
|
flags: 0x80000400
|
||||||
|
reserved1: 0x00000000
|
||||||
|
reserved2: 0x00000000
|
||||||
|
reserved3: 0x00000000
|
||||||
|
- sectname: __unwind_info
|
||||||
|
segname: __TEXT
|
||||||
|
addr: 0x0000000100000FAC
|
||||||
|
size: 72
|
||||||
|
offset: 0x00000FAC
|
||||||
|
align: 2
|
||||||
|
reloff: 0x00000000
|
||||||
|
nreloc: 0
|
||||||
|
flags: 0x00000000
|
||||||
|
reserved1: 0x00000000
|
||||||
|
reserved2: 0x00000000
|
||||||
|
reserved3: 0x00000000
|
||||||
|
- cmd: LC_SEGMENT_64
|
||||||
|
cmdsize: 232
|
||||||
|
segname: __DATA
|
||||||
|
vmaddr: 4294971392
|
||||||
|
vmsize: 4096
|
||||||
|
fileoff: 4096
|
||||||
|
filesize: 4096
|
||||||
|
maxprot: 7
|
||||||
|
initprot: 3
|
||||||
|
nsects: 2
|
||||||
|
flags: 0
|
||||||
|
Sections:
|
||||||
|
- sectname: __data
|
||||||
|
segname: __DATA
|
||||||
|
addr: 0x0000000100001000
|
||||||
|
size: 4
|
||||||
|
offset: 0x00001000
|
||||||
|
align: 2
|
||||||
|
reloff: 0x00000000
|
||||||
|
nreloc: 0
|
||||||
|
flags: 0x00000000
|
||||||
|
reserved1: 0x00000000
|
||||||
|
reserved2: 0x00000000
|
||||||
|
reserved3: 0x00000000
|
||||||
|
- sectname: __common
|
||||||
|
segname: __DATA
|
||||||
|
addr: 0x0000000100001004
|
||||||
|
size: 4
|
||||||
|
offset: 0x00000000
|
||||||
|
align: 2
|
||||||
|
reloff: 0x00000000
|
||||||
|
nreloc: 0
|
||||||
|
flags: 0x00000001
|
||||||
|
reserved1: 0x00000000
|
||||||
|
reserved2: 0x00000000
|
||||||
|
reserved3: 0x00000000
|
||||||
|
- cmd: LC_SEGMENT_64
|
||||||
|
cmdsize: 72
|
||||||
|
segname: __LINKEDIT
|
||||||
|
vmaddr: 4294975488
|
||||||
|
vmsize: 4096
|
||||||
|
fileoff: 8192
|
||||||
|
filesize: 232
|
||||||
|
maxprot: 7
|
||||||
|
initprot: 1
|
||||||
|
nsects: 0
|
||||||
|
flags: 0
|
||||||
|
- cmd: LC_DYLD_INFO_ONLY
|
||||||
|
cmdsize: 48
|
||||||
|
rebase_off: 0
|
||||||
|
rebase_size: 0
|
||||||
|
bind_off: 0
|
||||||
|
bind_size: 0
|
||||||
|
weak_bind_off: 0
|
||||||
|
weak_bind_size: 0
|
||||||
|
lazy_bind_off: 0
|
||||||
|
lazy_bind_size: 0
|
||||||
|
export_off: 8192
|
||||||
|
export_size: 72
|
||||||
|
- cmd: LC_SYMTAB
|
||||||
|
cmdsize: 24
|
||||||
|
symoff: 8272
|
||||||
|
nsyms: 6
|
||||||
|
stroff: 8368
|
||||||
|
strsize: 56
|
||||||
|
- cmd: LC_DYSYMTAB
|
||||||
|
cmdsize: 80
|
||||||
|
ilocalsym: 0
|
||||||
|
nlocalsym: 0
|
||||||
|
iextdefsym: 0
|
||||||
|
nextdefsym: 5
|
||||||
|
iundefsym: 5
|
||||||
|
nundefsym: 1
|
||||||
|
tocoff: 0
|
||||||
|
ntoc: 0
|
||||||
|
modtaboff: 0
|
||||||
|
nmodtab: 0
|
||||||
|
extrefsymoff: 0
|
||||||
|
nextrefsyms: 0
|
||||||
|
indirectsymoff: 0
|
||||||
|
nindirectsyms: 0
|
||||||
|
extreloff: 0
|
||||||
|
nextrel: 0
|
||||||
|
locreloff: 0
|
||||||
|
nlocrel: 0
|
||||||
|
- cmd: LC_LOAD_DYLINKER
|
||||||
|
cmdsize: 32
|
||||||
|
name: 12
|
||||||
|
PayloadString: '/usr/lib/dyld'
|
||||||
|
ZeroPadBytes: 7
|
||||||
|
- cmd: LC_UUID
|
||||||
|
cmdsize: 24
|
||||||
|
uuid: B6EE4FB7-4E1E-3C7A-80D3-CFBD89DBC0FE
|
||||||
|
- cmd: LC_BUILD_VERSION
|
||||||
|
cmdsize: 32
|
||||||
|
platform: 1
|
||||||
|
minos: 658944
|
||||||
|
sdk: 658944
|
||||||
|
ntools: 1
|
||||||
|
Tools:
|
||||||
|
- tool: 3
|
||||||
|
version: 29491968
|
||||||
|
- cmd: LC_SOURCE_VERSION
|
||||||
|
cmdsize: 16
|
||||||
|
version: 0
|
||||||
|
- cmd: LC_MAIN
|
||||||
|
cmdsize: 24
|
||||||
|
entryoff: 3984
|
||||||
|
stacksize: 0
|
||||||
|
- cmd: LC_LOAD_DYLIB
|
||||||
|
cmdsize: 56
|
||||||
|
dylib:
|
||||||
|
name: 24
|
||||||
|
timestamp: 2
|
||||||
|
current_version: 82115073
|
||||||
|
compatibility_version: 65536
|
||||||
|
PayloadString: '/usr/lib/libSystem.B.dylib'
|
||||||
|
ZeroPadBytes: 6
|
||||||
|
- cmd: LC_FUNCTION_STARTS
|
||||||
|
cmdsize: 16
|
||||||
|
dataoff: 8264
|
||||||
|
datasize: 8
|
||||||
|
- cmd: LC_DATA_IN_CODE
|
||||||
|
cmdsize: 16
|
||||||
|
dataoff: 8272
|
||||||
|
datasize: 0
|
||||||
|
LinkEditData:
|
||||||
|
ExportTrie:
|
||||||
|
TerminalSize: 0
|
||||||
|
NodeOffset: 0
|
||||||
|
Name: ''
|
||||||
|
Flags: 0x0000000000000000
|
||||||
|
Address: 0x0000000000000000
|
||||||
|
Other: 0x0000000000000000
|
||||||
|
ImportName: ''
|
||||||
|
Children:
|
||||||
|
- TerminalSize: 0
|
||||||
|
NodeOffset: 5
|
||||||
|
Name: _
|
||||||
|
Flags: 0x0000000000000000
|
||||||
|
Address: 0x0000000000000000
|
||||||
|
Other: 0x0000000000000000
|
||||||
|
ImportName: ''
|
||||||
|
Children:
|
||||||
|
- TerminalSize: 2
|
||||||
|
NodeOffset: 44
|
||||||
|
Name: _mh_execute_header
|
||||||
|
Flags: 0x0000000000000000
|
||||||
|
Address: 0x0000000000000000
|
||||||
|
Other: 0x0000000000000000
|
||||||
|
ImportName: ''
|
||||||
|
- TerminalSize: 3
|
||||||
|
NodeOffset: 48
|
||||||
|
Name: foo
|
||||||
|
Flags: 0x0000000000000000
|
||||||
|
Address: 0x0000000000000F70
|
||||||
|
Other: 0x0000000000000000
|
||||||
|
ImportName: ''
|
||||||
|
- TerminalSize: 3
|
||||||
|
NodeOffset: 53
|
||||||
|
Name: main
|
||||||
|
Flags: 0x0000000000000000
|
||||||
|
Address: 0x0000000000000F90
|
||||||
|
Other: 0x0000000000000000
|
||||||
|
ImportName: ''
|
||||||
|
- TerminalSize: 3
|
||||||
|
NodeOffset: 58
|
||||||
|
Name: b
|
||||||
|
Flags: 0x0000000000000000
|
||||||
|
Address: 0x0000000000001000
|
||||||
|
Other: 0x0000000000000000
|
||||||
|
ImportName: ''
|
||||||
|
- TerminalSize: 3
|
||||||
|
NodeOffset: 63
|
||||||
|
Name: a
|
||||||
|
Flags: 0x0000000000000000
|
||||||
|
Address: 0x0000000000001004
|
||||||
|
Other: 0x0000000000000000
|
||||||
|
ImportName: ''
|
||||||
|
NameList:
|
||||||
|
- n_strx: 2
|
||||||
|
n_type: 0x0F
|
||||||
|
n_sect: 1
|
||||||
|
n_desc: 16
|
||||||
|
n_value: 4294967296
|
||||||
|
- n_strx: 22
|
||||||
|
n_type: 0x0F
|
||||||
|
n_sect: 4
|
||||||
|
n_desc: 0
|
||||||
|
n_value: 4294971396
|
||||||
|
- n_strx: 25
|
||||||
|
n_type: 0x0F
|
||||||
|
n_sect: 3
|
||||||
|
n_desc: 0
|
||||||
|
n_value: 4294971392
|
||||||
|
- n_strx: 28
|
||||||
|
n_type: 0x0F
|
||||||
|
n_sect: 1
|
||||||
|
n_desc: 0
|
||||||
|
n_value: 4294971248
|
||||||
|
- n_strx: 33
|
||||||
|
n_type: 0x0F
|
||||||
|
n_sect: 1
|
||||||
|
n_desc: 0
|
||||||
|
n_value: 4294971280
|
||||||
|
- n_strx: 39
|
||||||
|
n_type: 0x01
|
||||||
|
n_sect: 0
|
||||||
|
n_desc: 256
|
||||||
|
n_value: 0
|
||||||
|
StringTable:
|
||||||
|
- ' '
|
||||||
|
- __mh_execute_header
|
||||||
|
- _a
|
||||||
|
- _b
|
||||||
|
- _foo
|
||||||
|
- _main
|
||||||
|
- dyld_stub_binder
|
||||||
|
...
|
||||||
|
|
||||||
|
# CHECK: FileType: Executable (0x2)
|
||||||
|
# CHECK: Name: __text
|
||||||
|
# CHECK: Name: __unwind_info
|
||||||
|
# CHECK: Name: __data
|
||||||
|
# CHECK: Name: __common
|
|
@ -17,7 +17,7 @@ namespace macho {
|
||||||
uint32_t MachOLayoutBuilder::computeSizeOfCmds() const {
|
uint32_t MachOLayoutBuilder::computeSizeOfCmds() const {
|
||||||
uint32_t Size = 0;
|
uint32_t Size = 0;
|
||||||
for (const auto &LC : O.LoadCommands) {
|
for (const auto &LC : O.LoadCommands) {
|
||||||
auto &MLC = LC.MachOLoadCommand;
|
const MachO::macho_load_command &MLC = LC.MachOLoadCommand;
|
||||||
auto cmd = MLC.load_command_data.cmd;
|
auto cmd = MLC.load_command_data.cmd;
|
||||||
switch (cmd) {
|
switch (cmd) {
|
||||||
case MachO::LC_SEGMENT:
|
case MachO::LC_SEGMENT:
|
||||||
|
@ -101,20 +101,25 @@ void MachOLayoutBuilder::updateDySymTab(MachO::macho_load_command &MLC) {
|
||||||
uint64_t MachOLayoutBuilder::layoutSegments() {
|
uint64_t MachOLayoutBuilder::layoutSegments() {
|
||||||
auto HeaderSize =
|
auto HeaderSize =
|
||||||
Is64Bit ? sizeof(MachO::mach_header_64) : sizeof(MachO::mach_header);
|
Is64Bit ? sizeof(MachO::mach_header_64) : sizeof(MachO::mach_header);
|
||||||
auto Offset = HeaderSize + O.Header.SizeOfCmds;
|
const bool IsObjectFile =
|
||||||
|
O.Header.FileType == MachO::HeaderFileType::MH_OBJECT;
|
||||||
// Lay out sections.
|
uint64_t Offset = IsObjectFile ? (HeaderSize + O.Header.SizeOfCmds) : 0;
|
||||||
for (auto &LC : O.LoadCommands) {
|
for (auto &LC : O.LoadCommands) {
|
||||||
uint64_t FileOff = Offset;
|
|
||||||
auto &MLC = LC.MachOLoadCommand;
|
auto &MLC = LC.MachOLoadCommand;
|
||||||
StringRef Segname;
|
StringRef Segname;
|
||||||
|
uint64_t SegmentVmAddr;
|
||||||
|
uint64_t SegmentVmSize;
|
||||||
switch (MLC.load_command_data.cmd) {
|
switch (MLC.load_command_data.cmd) {
|
||||||
case MachO::LC_SEGMENT:
|
case MachO::LC_SEGMENT:
|
||||||
|
SegmentVmAddr = MLC.segment_command_data.vmaddr;
|
||||||
|
SegmentVmSize = MLC.segment_command_data.vmsize;
|
||||||
Segname = StringRef(MLC.segment_command_data.segname,
|
Segname = StringRef(MLC.segment_command_data.segname,
|
||||||
strnlen(MLC.segment_command_data.segname,
|
strnlen(MLC.segment_command_data.segname,
|
||||||
sizeof(MLC.segment_command_data.segname)));
|
sizeof(MLC.segment_command_data.segname)));
|
||||||
break;
|
break;
|
||||||
case MachO::LC_SEGMENT_64:
|
case MachO::LC_SEGMENT_64:
|
||||||
|
SegmentVmAddr = MLC.segment_command_64_data.vmaddr;
|
||||||
|
SegmentVmSize = MLC.segment_command_64_data.vmsize;
|
||||||
Segname = StringRef(MLC.segment_command_64_data.segname,
|
Segname = StringRef(MLC.segment_command_64_data.segname,
|
||||||
strnlen(MLC.segment_command_64_data.segname,
|
strnlen(MLC.segment_command_64_data.segname,
|
||||||
sizeof(MLC.segment_command_64_data.segname)));
|
sizeof(MLC.segment_command_64_data.segname)));
|
||||||
|
@ -131,43 +136,64 @@ uint64_t MachOLayoutBuilder::layoutSegments() {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Update file offsets and sizes of sections.
|
// Update file offsets and sizes of sections.
|
||||||
|
uint64_t SegOffset = Offset;
|
||||||
|
uint64_t SegFileSize = 0;
|
||||||
uint64_t VMSize = 0;
|
uint64_t VMSize = 0;
|
||||||
uint64_t FileOffsetInSegment = 0;
|
|
||||||
for (auto &Sec : LC.Sections) {
|
for (auto &Sec : LC.Sections) {
|
||||||
if (!Sec.isVirtualSection()) {
|
if (IsObjectFile) {
|
||||||
auto FilePaddingSize =
|
if (Sec.isVirtualSection()) {
|
||||||
OffsetToAlignment(FileOffsetInSegment, 1ull << Sec.Align);
|
Sec.Offset = 0;
|
||||||
Sec.Offset = Offset + FileOffsetInSegment + FilePaddingSize;
|
} else {
|
||||||
Sec.Size = Sec.Content.size();
|
uint64_t PaddingSize = OffsetToAlignment(SegFileSize, 1 << Sec.Align);
|
||||||
FileOffsetInSegment += FilePaddingSize + Sec.Size;
|
Sec.Offset = SegOffset + SegFileSize + PaddingSize;
|
||||||
|
Sec.Size = Sec.Content.size();
|
||||||
|
SegFileSize += PaddingSize + Sec.Size;
|
||||||
|
}
|
||||||
|
VMSize = std::max(VMSize, Sec.Addr + Sec.Size);
|
||||||
|
} else {
|
||||||
|
if (Sec.isVirtualSection()) {
|
||||||
|
Sec.Offset = 0;
|
||||||
|
VMSize += Sec.Size;
|
||||||
|
} else {
|
||||||
|
uint32_t SectOffset = Sec.Addr - SegmentVmAddr;
|
||||||
|
Sec.Offset = SegOffset + SectOffset;
|
||||||
|
Sec.Size = Sec.Content.size();
|
||||||
|
SegFileSize = std::max(SegFileSize, SectOffset + Sec.Size);
|
||||||
|
VMSize = std::max(VMSize, SegFileSize);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
VMSize = std::max(VMSize, Sec.Addr + Sec.Size);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO: Handle the __PAGEZERO segment.
|
if (IsObjectFile) {
|
||||||
|
Offset += SegFileSize;
|
||||||
|
} else {
|
||||||
|
Offset = alignTo(Offset + SegFileSize, PageSize);
|
||||||
|
SegFileSize = alignTo(SegFileSize, PageSize);
|
||||||
|
// Use the original vmsize if the segment is __PAGEZERO.
|
||||||
|
VMSize =
|
||||||
|
Segname == "__PAGEZERO" ? SegmentVmSize : alignTo(VMSize, PageSize);
|
||||||
|
}
|
||||||
|
|
||||||
switch (MLC.load_command_data.cmd) {
|
switch (MLC.load_command_data.cmd) {
|
||||||
case MachO::LC_SEGMENT:
|
case MachO::LC_SEGMENT:
|
||||||
MLC.segment_command_data.cmdsize =
|
MLC.segment_command_data.cmdsize =
|
||||||
sizeof(MachO::segment_command) +
|
sizeof(MachO::segment_command) +
|
||||||
sizeof(MachO::section) * LC.Sections.size();
|
sizeof(MachO::section) * LC.Sections.size();
|
||||||
MLC.segment_command_data.nsects = LC.Sections.size();
|
MLC.segment_command_data.nsects = LC.Sections.size();
|
||||||
MLC.segment_command_data.fileoff = FileOff;
|
MLC.segment_command_data.fileoff = SegOffset;
|
||||||
MLC.segment_command_data.vmsize = VMSize;
|
MLC.segment_command_data.vmsize = VMSize;
|
||||||
MLC.segment_command_data.filesize = FileOffsetInSegment;
|
MLC.segment_command_data.filesize = SegFileSize;
|
||||||
break;
|
break;
|
||||||
case MachO::LC_SEGMENT_64:
|
case MachO::LC_SEGMENT_64:
|
||||||
MLC.segment_command_64_data.cmdsize =
|
MLC.segment_command_64_data.cmdsize =
|
||||||
sizeof(MachO::segment_command_64) +
|
sizeof(MachO::segment_command_64) +
|
||||||
sizeof(MachO::section_64) * LC.Sections.size();
|
sizeof(MachO::section_64) * LC.Sections.size();
|
||||||
MLC.segment_command_64_data.nsects = LC.Sections.size();
|
MLC.segment_command_64_data.nsects = LC.Sections.size();
|
||||||
MLC.segment_command_64_data.fileoff = FileOff;
|
MLC.segment_command_64_data.fileoff = SegOffset;
|
||||||
MLC.segment_command_64_data.vmsize = VMSize;
|
MLC.segment_command_64_data.vmsize = VMSize;
|
||||||
MLC.segment_command_64_data.filesize = FileOffsetInSegment;
|
MLC.segment_command_64_data.filesize = SegFileSize;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
|
|
||||||
Offset += FileOffsetInSegment;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return Offset;
|
return Offset;
|
||||||
|
|
Loading…
Reference in New Issue