Index: include/llvm/Object/COFF.h =================================================================== --- include/llvm/Object/COFF.h +++ include/llvm/Object/COFF.h @@ -971,6 +971,15 @@ return nullptr; return reinterpret_cast(base()); } + std::error_code getCOFFHeader(const coff_file_header *&Res) const { + Res = COFFHeader; + return std::error_code(); + } + std::error_code + getCOFFBigObjHeader(const coff_bigobj_file_header *&Res) const { + Res = COFFBigObjHeader; + return std::error_code(); + } std::error_code getPE32Header(const pe32_header *&Res) const; std::error_code getPE32PlusHeader(const pe32plus_header *&Res) const; std::error_code getDataDirectory(uint32_t index, Index: test/tools/llvm-objcopy/COFF/X86/lit.local.cfg =================================================================== --- /dev/null +++ test/tools/llvm-objcopy/COFF/X86/lit.local.cfg @@ -0,0 +1,4 @@ +# These tests require a registered x86 backend. + +if not 'X86' in config.root.targets: + config.unsupported = True Index: test/tools/llvm-objcopy/COFF/X86/remove-local-symbols.s =================================================================== --- /dev/null +++ test/tools/llvm-objcopy/COFF/X86/remove-local-symbols.s @@ -0,0 +1,37 @@ +# RUN: llvm-mc -triple=x86_64-windows-gnu %s -filetype=obj -o %t.o + +# RUN: llvm-readobj -relocations %t.o | FileCheck %s --check-prefix=RELOCS +# RUN: llvm-nm %t.o | FileCheck %s --check-prefix=SYMBOLS-PRE +# RUN: llvm-strip -x %t.o +# RUN: llvm-readobj -relocations %t.o | FileCheck %s --check-prefix=RELOCS +# RUN: llvm-nm %t.o | FileCheck %s --check-prefix=SYMBOLS-POST + +# RELOCS: Relocations [ +# RELOCS-NEXT: Section (1) .text { +# RELOCS-NEXT: 0x2 IMAGE_REL_AMD64_REL32 ptr +# RELOCS-NEXT: } +# RELOCS-NEXT: ] + +# SYMBOLS-PRE: 00000000 T mainfunc +# SYMBOLS-PRE: 0000000c t otherfunc +# SYMBOLS-PRE: 00000000 d ptr +# SYMBOLS-PRE-EMPTY: + +# SYMBOLS-POST: 00000000 T mainfunc +# SYMBOLS-POST: 00000000 d ptr +# SYMBOLS-POST-EMPTY: + + + .text + .globl mainfunc +mainfunc: + movl ptr(%rip), %eax + # This doesn't produce any relocation, and otherfunc can be dropped. + call otherfunc + ret +otherfunc: + ret + + .data +ptr: + .long 42 Index: test/tools/llvm-objcopy/COFF/add-gnu-debuglink.test =================================================================== --- /dev/null +++ test/tools/llvm-objcopy/COFF/add-gnu-debuglink.test @@ -0,0 +1,31 @@ +RUN: llvm-objcopy --strip-all %p/Inputs/x86_64-debug.exe %t.exe +RUN: llvm-objcopy --add-gnu-debuglink=%p/Inputs/x86_64-debug.exe %t.exe +RUN: llvm-readobj -sections %t.exe | FileCheck %s --check-prefix=SECTIONS +RUN: llvm-objdump -s %t.exe | FileCheck %s --check-prefix=CONTENTS + +SECTIONS: Section { +SECTIONS: Number: 5 +SECTIONS-NEXT: Name: .pdata +SECTIONS-NEXT: VirtualSize: 0x18 +SECTIONS-NEXT: VirtualAddress: 0x5000 +SECTIONS-NEXT: RawDataSize: 512 +SECTIONS: Section { +SECTIONS-NEXT: Number: 6 +SECTIONS-NEXT: Name: .gnu_debuglink +SECTIONS-NEXT: VirtualSize: 0x18 +SECTIONS-NEXT: VirtualAddress: 0x6000 +SECTIONS-NEXT: RawDataSize: 512 +SECTIONS-NEXT: PointerToRawData: +SECTIONS-NEXT: PointerToRelocations: +SECTIONS-NEXT: PointerToLineNumbers: +SECTIONS-NEXT: RelocationCount: +SECTIONS-NEXT: LineNumberCount: +SECTIONS-NEXT: Characteristics [ (0x42000040) +SECTIONS-NEXT: IMAGE_SCN_CNT_INITIALIZED_DATA (0x40) +SECTIONS-NEXT: IMAGE_SCN_MEM_DISCARDABLE (0x2000000) +SECTIONS-NEXT: IMAGE_SCN_MEM_READ (0x40000000) +SECTIONS-NEXT: ] + +CONTENTS: Contents of section .gnu_debuglink: +CONTENTS: 140006000 7838365f 36342d64 65627567 2e657865 x86_64-debug.exe +CONTENTS: 140006010 00000000 6b0793d9 ....k... Index: test/tools/llvm-objcopy/COFF/basic-copy.test =================================================================== --- /dev/null +++ test/tools/llvm-objcopy/COFF/basic-copy.test @@ -0,0 +1,30 @@ +RUN: llvm-objcopy %p/Inputs/i386.o %t.o +RUN: cmp %p/Inputs/i386.o %t.o + +RUN: llvm-objcopy %p/Inputs/x86_64.o %t.o +RUN: cmp %p/Inputs/x86_64.o %t.o + +RUN: llvm-objcopy %p/Inputs/i386-big.o %t.o +RUN: cmp %p/Inputs/i386-big.o %t.o + +RUN: llvm-objcopy %p/Inputs/x86_64-big.o %t.o +RUN: cmp %p/Inputs/x86_64-big.o %t.o + +RUN: llvm-objcopy %p/Inputs/i386.exe %t.exe +RUN: cmp %p/Inputs/i386.exe %t.exe + +RUN: llvm-objcopy %p/Inputs/x86_64.exe %t.exe +RUN: cmp %p/Inputs/x86_64.exe %t.exe + +Having exactly identical output, as this test requires, is pretty +brittle if considering any random input file. Details that can +vary are: +- The padding of executable sections (lld uses 0xcc, which is int3 on x86) +- The gap between headers and the contents of the first section + (lld currently can leave a whole empty sector inbetween) +- The actual layout of of the string table (it can be filled linearly, + strings can be dedupliated, the table can be optimized by sharing tails + of longer strings; different parts in llvm do each of these three options) +- The size indication for an empty/missing string table can either be 4 + or left out altogether +- Checksums Index: test/tools/llvm-objcopy/COFF/discard-all.test =================================================================== --- /dev/null +++ test/tools/llvm-objcopy/COFF/discard-all.test @@ -0,0 +1,87 @@ +RUN: llvm-readobj -symbols %p/Inputs/x86_64.o | FileCheck %s --check-prefix=SYMS-ORIG + +RUN: cp %p/Inputs/x86_64.o %t.o +RUN: llvm-strip -x %t.o +RUN: llvm-readobj -symbols %t.o | FileCheck %s --check-prefix=SYMS-NEW + +RUN: llvm-readobj -sections %t.o | FileCheck %s --check-prefix=SECTIONS + +For this specific input, the debug sections and symbols relating +to the debug sections are removed. + +SYMS-ORIG: Name: .text +SYMS-ORIG: Name: .data +SYMS-ORIG: Name: .bss +SYMS-ORIG: Name: .xdata +SYMS-ORIG: Name: .debug_str +SYMS-ORIG: Name: .debug_abbrev +SYMS-ORIG: Name: .debug_info +SYMS-ORIG: Name: .debug_macinfo +SYMS-ORIG: Name: .pdata +SYMS-ORIG: Name: .debug_line +SYMS-ORIG: Name: .llvm_addrsig +SYMS-ORIG: Name: @feat.00 +SYMS-ORIG: Name: f +SYMS-ORIG: Name: x +SYMS-ORIG: Name: __main +SYMS-ORIG: Name: main + +SYMS-NEW-NOT: Name: .text +SYMS-NEW-NOT: Name: .data +SYMS-NEW-NOT: Name: .bss +SYMS-NEW: Name: .xdata +SYMS-NEW-NOT: Name: .debug_str +SYMS-NEW-NOT: Name: .debug_abbrev +SYMS-NEW-NOT: Name: .debug_info +SYMS-NEW-NOT: Name: .debug_macinfo +SYMS-NEW-NOT: Name: .pdata +SYMS-NEW-NOT: Name: .debug_line +SYMS-NEW-NOT: Name: .llvm_addrsig +SYMS-NEW-NOT: Name: @feat.00 +SYMS-NEW: Name: f +SYMS-NEW: Name: x +SYMS-NEW: Name: __main +SYMS-NEW: Name: main + +SECTIONS: Sections [ +SECTIONS-NEXT: Section { +SECTIONS-NEXT: Number: 1 +SECTIONS-NEXT: Name: .text +SECTIONS-NEXT: VirtualSize: +SECTIONS-NEXT: VirtualAddress: +SECTIONS-NEXT: RawDataSize: 87 +SECTIONS: Section { +SECTIONS-NEXT: Number: 2 +SECTIONS-NEXT: Name: .data +SECTIONS-NEXT: VirtualSize: +SECTIONS-NEXT: VirtualAddress: +SECTIONS-NEXT: RawDataSize: 4 +SECTIONS: Section { +SECTIONS-NEXT: Number: 3 +SECTIONS-NEXT: Name: .bss +SECTIONS-NEXT: VirtualSize: +SECTIONS-NEXT: VirtualAddress: +SECTIONS-NEXT: RawDataSize: 0 +SECTIONS: Section { +SECTIONS-NEXT: Number: 4 +SECTIONS-NEXT: Name: .xdata +SECTIONS-NEXT: VirtualSize: +SECTIONS-NEXT: VirtualAddress: +SECTIONS-NEXT: RawDataSize: 20 +SECTIONS: Section { +SECTIONS-NEXT: Number: 5 +SECTIONS-NEXT: Name: .pdata +SECTIONS-NEXT: VirtualSize: +SECTIONS-NEXT: VirtualAddress: +SECTIONS-NEXT: RawDataSize: 24 +SECTIONS: Section { +SECTIONS-NEXT: Number: 6 +SECTIONS-NEXT: Name: .llvm_addrsig +SECTIONS-NEXT: VirtualSize: +SECTIONS-NEXT: VirtualAddress: +SECTIONS-NEXT: RawDataSize: 2 +SECTIONS-NOT: Name: .debug_abbrev +SECTIONS-NOT: Name: .debug_info +SECTIONS-NOT: Name: .debug_line +SECTIONS-NOT: Name: .debug_macinfo +SECTIONS-NOT: Name: .debug_str Index: test/tools/llvm-objcopy/COFF/only-keep-debug.test =================================================================== --- /dev/null +++ test/tools/llvm-objcopy/COFF/only-keep-debug.test @@ -0,0 +1,71 @@ +RUN: llvm-objcopy --only-keep-debug %p/Inputs/x86_64-debug.exe %t.exe +RUN: llvm-nm %p/Inputs/x86_64-debug.exe > %t.syms.orig +RUN: llvm-nm %t.exe > %t.syms.new +RUN: cmp %t.syms.orig %t.syms.new + +RUN: llvm-readobj -sections %t.exe | FileCheck %s + +The non-debug sections except for .buildid all get RawDataSize: set to +zero. + +CHECK: Sections [ +CHECK-NEXT: Section { +CHECK-NEXT: Number: 1 +CHECK-NEXT: Name: .text +CHECK-NEXT: VirtualSize: 0x57 +CHECK-NEXT: VirtualAddress: +CHECK-NEXT: RawDataSize: 0 +CHECK: Section { +CHECK-NEXT: Number: 2 +CHECK-NEXT: Name: .rdata +CHECK-NEXT: VirtualSize: 0x34 +CHECK-NEXT: VirtualAddress: +CHECK-NEXT: RawDataSize: 0 +CHECK: Section { +CHECK-NEXT: Number: 3 +CHECK-NEXT: Name: .buildid +CHECK-NEXT: VirtualSize: 0x35 +CHECK-NEXT: VirtualAddress: +CHECK-NEXT: RawDataSize: 512 +CHECK: Section { +CHECK-NEXT: Number: 4 +CHECK-NEXT: Name: .data +CHECK-NEXT: VirtualSize: 0x4 +CHECK-NEXT: VirtualAddress: +CHECK-NEXT: RawDataSize: 0 +CHECK: Section { +CHECK-NEXT: Number: 5 +CHECK-NEXT: Name: .pdata +CHECK-NEXT: VirtualSize: 0x18 +CHECK-NEXT: VirtualAddress: +CHECK-NEXT: RawDataSize: 0 +CHECK: Section { +CHECK-NEXT: Number: 6 +CHECK-NEXT: Name: .debug_abbrev +CHECK-NEXT: VirtualSize: 0x80 +CHECK-NEXT: VirtualAddress: +CHECK-NEXT: RawDataSize: 512 +CHECK: Section { +CHECK-NEXT: Number: 7 +CHECK-NEXT: Name: .debug_info +CHECK-NEXT: VirtualSize: 0x9D +CHECK-NEXT: VirtualAddress: +CHECK-NEXT: RawDataSize: 512 +CHECK: Section { +CHECK-NEXT: Number: 8 +CHECK-NEXT: Name: .debug_line +CHECK-NEXT: VirtualSize: 0x5B +CHECK-NEXT: VirtualAddress: +CHECK-NEXT: RawDataSize: 512 +CHECK: Section { +CHECK-NEXT: Number: 9 +CHECK-NEXT: Name: .debug_macinfo +CHECK-NEXT: VirtualSize: 0x1 +CHECK-NEXT: VirtualAddress: +CHECK-NEXT: RawDataSize: 512 +CHECK: Section { +CHECK-NEXT: Number: 10 +CHECK-NEXT: Name: .debug_str +CHECK-NEXT: VirtualSize: 0x81 +CHECK-NEXT: VirtualAddress: +CHECK-NEXT: RawDataSize: 512 Index: test/tools/llvm-objcopy/COFF/strip-executable.test =================================================================== --- /dev/null +++ test/tools/llvm-objcopy/COFF/strip-executable.test @@ -0,0 +1,70 @@ +RUN: llvm-nm %p/Inputs/x86_64-debug.exe | FileCheck %s --check-prefix=SYMS-ORIG + +RUN: llvm-objcopy --strip-all %p/Inputs/x86_64-debug.exe %t.exe +RUN: llvm-nm %t.exe 2>&1 | FileCheck %s --check-prefix=SYMS-NEW +RUN: llvm-readobj -sections %t.exe | FileCheck %s --check-prefix=SECTIONS + +RUN: llvm-objcopy --strip-unneeded %p/Inputs/x86_64-debug.exe %t.exe +RUN: llvm-nm %t.exe 2>&1 | FileCheck %s --check-prefix=SYMS-NEW +RUN: llvm-readobj -sections %t.exe | FileCheck %s --check-prefix=SECTIONS + +RUN: cp %p/Inputs/x86_64-debug.exe %t.exe +RUN: llvm-strip %t.exe +RUN: llvm-nm %t.exe 2>&1 | FileCheck %s --check-prefix=SYMS-NEW +RUN: llvm-readobj -sections %t.exe | FileCheck %s --check-prefix=SECTIONS + +The debug sections and all symbols are removed. + +SYMS-ORIG: 140004004 d .bss +SYMS-ORIG: 140004000 d .data +SYMS-ORIG: 140006000 N .debug_abbrev +SYMS-ORIG: 140007000 N .debug_info +SYMS-ORIG: 140008000 N .debug_line +SYMS-ORIG: 140009000 N .debug_macinfo +SYMS-ORIG: 14000a000 N .debug_str +SYMS-ORIG: 140005000 r .pdata +SYMS-ORIG: 140001000 t .text +SYMS-ORIG: 140002020 r .xdata +SYMS-ORIG: 140001020 T __main +SYMS-ORIG: 140001000 T f +SYMS-ORIG: 140001030 T main +SYMS-ORIG: 140004000 D x + +SYMS-NEW: no symbols + +SECTIONS: Sections [ +SECTIONS-NEXT: Section { +SECTIONS-NEXT: Number: 1 +SECTIONS-NEXT: Name: .text +SECTIONS-NEXT: VirtualSize: 0x57 +SECTIONS-NEXT: VirtualAddress: +SECTIONS-NEXT: RawDataSize: 512 +SECTIONS: Section { +SECTIONS-NEXT: Number: 2 +SECTIONS-NEXT: Name: .rdata +SECTIONS-NEXT: VirtualSize: 0x34 +SECTIONS-NEXT: VirtualAddress: +SECTIONS-NEXT: RawDataSize: 512 +SECTIONS: Section { +SECTIONS-NEXT: Number: 3 +SECTIONS-NEXT: Name: .buildid +SECTIONS-NEXT: VirtualSize: 0x35 +SECTIONS-NEXT: VirtualAddress: +SECTIONS-NEXT: RawDataSize: 512 +SECTIONS: Section { +SECTIONS-NEXT: Number: 4 +SECTIONS-NEXT: Name: .data +SECTIONS-NEXT: VirtualSize: 0x4 +SECTIONS-NEXT: VirtualAddress: +SECTIONS-NEXT: RawDataSize: 512 +SECTIONS: Section { +SECTIONS-NEXT: Number: 5 +SECTIONS-NEXT: Name: .pdata +SECTIONS-NEXT: VirtualSize: 0x18 +SECTIONS-NEXT: VirtualAddress: +SECTIONS-NEXT: RawDataSize: 512 +SECTIONS-NOT: Name: .debug_abbrev +SECTIONS-NOT: Name: .debug_info +SECTIONS-NOT: Name: .debug_line +SECTIONS-NOT: Name: .debug_macinfo +SECTIONS-NOT: Name: .debug_str Index: test/tools/llvm-objcopy/COFF/strip-object.test =================================================================== --- /dev/null +++ test/tools/llvm-objcopy/COFF/strip-object.test @@ -0,0 +1,86 @@ +RUN: llvm-readobj -symbols %p/Inputs/x86_64.o | FileCheck %s --check-prefix=SYMS-ORIG + +RUN: llvm-objcopy --strip-debug %p/Inputs/x86_64.o %t.o +RUN: llvm-readobj -symbols %t.o | FileCheck %s --check-prefix=SYMS-NEW + +RUN: llvm-readobj -sections %t.o | FileCheck %s --check-prefix=SECTIONS + +The debug sections are removed, symbols relating to the debug sections +are removed. + +SYMS-ORIG: Name: .text +SYMS-ORIG: Name: .data +SYMS-ORIG: Name: .bss +SYMS-ORIG: Name: .xdata +SYMS-ORIG: Name: .debug_str +SYMS-ORIG: Name: .debug_abbrev +SYMS-ORIG: Name: .debug_info +SYMS-ORIG: Name: .debug_macinfo +SYMS-ORIG: Name: .pdata +SYMS-ORIG: Name: .debug_line +SYMS-ORIG: Name: .llvm_addrsig +SYMS-ORIG: Name: @feat.00 +SYMS-ORIG: Name: f +SYMS-ORIG: Name: x +SYMS-ORIG: Name: __main +SYMS-ORIG: Name: main + +SYMS-NEW: Name: .text +SYMS-NEW: Name: .data +SYMS-NEW: Name: .bss +SYMS-NEW: Name: .xdata +SYMS-NEW-NOT: Name: .debug_str +SYMS-NEW-NOT: Name: .debug_abbrev +SYMS-NEW-NOT: Name: .debug_info +SYMS-NEW-NOT: Name: .debug_macinfo +SYMS-NEW: Name: .pdata +SYMS-NEW-NOT: Name: .debug_line +SYMS-NEW: Name: .llvm_addrsig +SYMS-NEW: Name: @feat.00 +SYMS-NEW: Name: f +SYMS-NEW: Name: x +SYMS-NEW: Name: __main +SYMS-NEW: Name: main + +SECTIONS: Sections [ +SECTIONS-NEXT: Section { +SECTIONS-NEXT: Number: 1 +SECTIONS-NEXT: Name: .text +SECTIONS-NEXT: VirtualSize: +SECTIONS-NEXT: VirtualAddress: +SECTIONS-NEXT: RawDataSize: 87 +SECTIONS: Section { +SECTIONS-NEXT: Number: 2 +SECTIONS-NEXT: Name: .data +SECTIONS-NEXT: VirtualSize: +SECTIONS-NEXT: VirtualAddress: +SECTIONS-NEXT: RawDataSize: 4 +SECTIONS: Section { +SECTIONS-NEXT: Number: 3 +SECTIONS-NEXT: Name: .bss +SECTIONS-NEXT: VirtualSize: +SECTIONS-NEXT: VirtualAddress: +SECTIONS-NEXT: RawDataSize: 0 +SECTIONS: Section { +SECTIONS-NEXT: Number: 4 +SECTIONS-NEXT: Name: .xdata +SECTIONS-NEXT: VirtualSize: +SECTIONS-NEXT: VirtualAddress: +SECTIONS-NEXT: RawDataSize: 20 +SECTIONS: Section { +SECTIONS-NEXT: Number: 5 +SECTIONS-NEXT: Name: .pdata +SECTIONS-NEXT: VirtualSize: +SECTIONS-NEXT: VirtualAddress: +SECTIONS-NEXT: RawDataSize: 24 +SECTIONS: Section { +SECTIONS-NEXT: Number: 6 +SECTIONS-NEXT: Name: .llvm_addrsig +SECTIONS-NEXT: VirtualSize: +SECTIONS-NEXT: VirtualAddress: +SECTIONS-NEXT: RawDataSize: 2 +SECTIONS-NOT: Name: .debug_abbrev +SECTIONS-NOT: Name: .debug_info +SECTIONS-NOT: Name: .debug_line +SECTIONS-NOT: Name: .debug_macinfo +SECTIONS-NOT: Name: .debug_str Index: tools/llvm-objcopy/CMakeLists.txt =================================================================== --- tools/llvm-objcopy/CMakeLists.txt +++ tools/llvm-objcopy/CMakeLists.txt @@ -17,6 +17,8 @@ Buffer.cpp CopyConfig.cpp llvm-objcopy.cpp + COFF/COFFObjcopy.cpp + COFF/Object.cpp ELF/ELFObjcopy.cpp ELF/Object.cpp DEPENDS Index: tools/llvm-objcopy/COFF/COFFObjcopy.h =================================================================== --- /dev/null +++ tools/llvm-objcopy/COFF/COFFObjcopy.h @@ -0,0 +1,31 @@ +//===- COFFObjcopy.h --------------------------------------------*- C++ -*-===// +// +// The LLVM Compiler Infrastructure +// +// This file is distributed under the University of Illinois Open Source +// License. See LICENSE.TXT for details. +// +//===----------------------------------------------------------------------===// + +#ifndef LLVM_TOOLS_OBJCOPY_COFFOBJCOPY_H +#define LLVM_TOOLS_OBJCOPY_COFFOBJCOPY_H + +namespace llvm { + +namespace object { +class COFFObjectFile; +} // end namespace object + +namespace objcopy { +struct CopyConfig; +class Buffer; + +namespace coff { +void executeObjcopyOnBinary(const CopyConfig &Config, + object::COFFObjectFile &In, Buffer &Out); + +} // end namespace coff +} // end namespace objcopy +} // end namespace llvm + +#endif // LLVM_TOOLS_OBJCOPY_COFFOBJCOPY_H Index: tools/llvm-objcopy/COFF/COFFObjcopy.cpp =================================================================== --- /dev/null +++ tools/llvm-objcopy/COFF/COFFObjcopy.cpp @@ -0,0 +1,301 @@ +//===- COFFObjcopy.cpp ----------------------------------------------------===// +// +// The LLVM Compiler Infrastructure +// +// This file is distributed under the University of Illinois Open Source +// License. See LICENSE.TXT for details. +// +//===----------------------------------------------------------------------===// + +#include "COFFObjcopy.h" +#include "Buffer.h" +#include "CopyConfig.h" +#include "Object.h" +#include "llvm-objcopy.h" + +#include "llvm/ADT/DenseMap.h" +#include "llvm/ADT/DenseSet.h" +#include "llvm/ADT/StringMap.h" +#include "llvm/Object/Binary.h" +#include "llvm/Object/COFF.h" +#include "llvm/Support/Endian.h" +#include "llvm/Support/JamCRC.h" +#include "llvm/Support/Path.h" +#include + +namespace llvm { +namespace objcopy { +namespace coff { + +using namespace object; +using namespace COFF; + +static bool isDebugSection(const Section &Sec) { + return Sec.Name.startswith(".debug"); +} + +static bool isNondebugKeepableSection(const Section &Sec) { + return Sec.Name == ".buildid"; +} + +template +static uint64_t getNextRVA(const Object &Obj, const PeHeaderTy &PeHeader) { + if (Obj.Sections.empty()) + return 0; + const Section &Last = Obj.Sections.back(); + uint32_t End = Last.Header.VirtualAddress + Last.Header.VirtualSize; + uint32_t Alignment = PeHeader.SectionAlignment; + End = alignTo(End, Alignment); + return End; +} + +static uint32_t getCRC32(StringRef Data) { + JamCRC CRC; + CRC.update(ArrayRef(Data.data(), Data.size())); + // The CRC32 value needs to be complemented because the JamCRC dosn't + // finalize the CRC32 value. It also dosn't negate the initial CRC32 value + // but it starts by default at 0xFFFFFFFF which is the complement of zero. + return ~CRC.getCRC(); +} + +static void createGnuDebugLinkSectionContents(std::vector &Data, + StringRef File) { + auto LinkTargetOrErr = MemoryBuffer::getFile(File); + if (!LinkTargetOrErr) + error("'" + File + "': " + LinkTargetOrErr.getError().message()); + auto LinkTarget = std::move(*LinkTargetOrErr); + uint32_t CRC32 = getCRC32(LinkTarget->getBuffer()); + + StringRef FileName = sys::path::filename(File); + size_t CRCPos = alignTo(FileName.size() + 1, 4); + Data.resize(CRCPos + 4, 0); + memcpy(Data.data(), FileName.data(), FileName.size()); + support::endian::write32le(Data.data() + CRCPos, CRC32); +} + +template +static void addGnuDebugLink(Object &Obj, PeHeaderTy &PeHeader, + StringRef DebugLinkFile) { + uint32_t StartRVA = getNextRVA(Obj, PeHeader); + + Obj.Sections.push_back(Section()); + Section &Sec = Obj.Sections.back(); + createGnuDebugLinkSectionContents(Sec.LocalContents, DebugLinkFile); + Sec.Contents = Sec.LocalContents; + Sec.Name = ".gnu_debuglink"; + Sec.Header.VirtualSize = Sec.Contents.size(); + Sec.Header.VirtualAddress = StartRVA; + Sec.Header.SizeOfRawData = + alignTo(Sec.Header.VirtualSize, PeHeader.FileAlignment); + // Sec.Header.PointerToRawData is filled in by the writer. + Sec.Header.PointerToRelocations = 0; + Sec.Header.PointerToLinenumbers = 0; + Sec.Header.NumberOfRelocations = 0; + Sec.Header.NumberOfLinenumbers = 0; + Sec.Header.Characteristics = IMAGE_SCN_CNT_INITIALIZED_DATA | + IMAGE_SCN_MEM_READ | IMAGE_SCN_MEM_DISCARDABLE; +} + +static void handleArgs(const CopyConfig &Config, Object &Obj) { + size_t OrigNumSymbols = Obj.Symbols.size(); + std::vector RawSymbolTable; + // Initialize mappings for the relocations, to allow remapping them + // after changing the symbol table. + for (Symbol &S : Obj.Symbols) { + RawSymbolTable.push_back(&S); + S.Referenced = false; + for (size_t I = 0; I < S.Sym.NumberOfAuxSymbols; I++) + RawSymbolTable.push_back(nullptr); + } + for (Section &Sec : Obj.Sections) { + for (const coff_relocation &Reloc : Sec.Relocs) { + if (Reloc.SymbolTableIndex >= RawSymbolTable.size()) + reportError(Config.InputFilename, + make_error("SymbolTableIndex out of range", + object_error::parse_failed)); + else if (RawSymbolTable[Reloc.SymbolTableIndex] == nullptr) + reportError(Config.InputFilename, + make_error("Invalid SymbolTableIndex", + object_error::parse_failed)); + Sec.RelocTargets.push_back(RawSymbolTable[Reloc.SymbolTableIndex]->Name); + } + } + + if (Config.OnlyKeepDebug) { + // For keeping only debug info, we keep the other symbols as such + // but remove their content data. + Obj.truncateSections([](const Section &Sec) { + return !isDebugSection(Sec) && !isNondebugKeepableSection(Sec); + }); + } + + std::function RemovePred = [](const Section &) { + return false; + }; + + // Removes: + if (!Config.ToRemove.empty()) { + RemovePred = [&Config](const Section &Sec) { + return is_contained(Config.ToRemove, Sec.Name); + }; + } + + if (Config.StripAllGNU || Config.StripDebug || Config.StripAll || + Config.DiscardAll || Config.StripUnneeded) { + RemovePred = [RemovePred](const Section &Sec) { + return RemovePred(Sec) || isDebugSection(Sec); + }; + } + + // Explicit copies: + if (!Config.OnlyKeep.empty()) { + RemovePred = [&Config, RemovePred, &Obj](const Section &Sec) { + // Explicitly keep these sections regardless of previous removes. + if (is_contained(Config.OnlyKeep, Sec.Name)) + return false; + + // Allow all implicit removes. + if (RemovePred(Sec)) + return true; + + // Remove everything else. + return true; + }; + } + + if (!Config.KeepSection.empty()) { + RemovePred = [&Config, RemovePred](const Section &Sec) { + // Explicitly keep these sections regardless of previous removes. + if (is_contained(Config.KeepSection, Sec.Name)) + return false; + // Otherwise defer to RemovePred. + return RemovePred(Sec); + }; + } + + size_t Idx = 1; + for (Section &Sec : Obj.Sections) + Sec.Idx = Idx++; + DenseSet RemovedSections; + RemovePred = [&RemovedSections, RemovePred](const Section &Sec) { + bool Remove = RemovePred(Sec); + if (Remove) + RemovedSections.insert(Sec.Idx); + return Remove; + }; + + // NOTE: If we remove sections, we don't update section number references + // in coff_aux_section_definition for associative comdats pointing behind + // removed sections. Config.StripUnneeded or Config.DiscardAll will likely + // also break comdats. + Obj.removeSections(RemovePred); + + // If we need to do per-symbol removals, initialize the Referenced field, + // based on the remaining sections. This uses raw symbol indexes from + // Reloc.SymbolTableIndex instead of looking things up using the symbol + // names, before we prune the symbol table. + if (Config.StripUnneeded || Config.DiscardAll || + !Config.SymbolsToRemove.empty()) { + for (const Section &Sec : Obj.Sections) { + for (const coff_relocation &Reloc : Sec.Relocs) { + if (Reloc.SymbolTableIndex >= RawSymbolTable.size()) + reportError(Config.InputFilename, + make_error("SymbolTableIndex out of range", + object_error::parse_failed)); + else if (RawSymbolTable[Reloc.SymbolTableIndex] == nullptr) + reportError(Config.InputFilename, + make_error("Invalid SymbolTableIndex", + object_error::parse_failed)); + RawSymbolTable[Reloc.SymbolTableIndex]->Referenced = true; + } + } + } + + // If we did remove sections, remove all symbols pointing to those sections + // and update symbols to point to new section numbers. + if (!RemovedSections.empty()) { + DenseMap NewSectionIndex; + Idx = 1; + for (Section &Sec : Obj.Sections) { + NewSectionIndex[Sec.Idx] = Idx; + Sec.Idx = Idx++; + } + Obj.removeSymbols([&](const Symbol &Sym) { + return RemovedSections.count(Sym.Sym.SectionNumber) == 1; + }); + for (Symbol &Sym : Obj.Symbols) + Sym.Sym.SectionNumber = NewSectionIndex[Sym.Sym.SectionNumber]; + } + + // Actually do removals of symbols. + Obj.removeSymbols([&](const Symbol &Sym) { + if (!Config.SymbolsToKeep.empty() && + is_contained(Config.SymbolsToKeep, Sym.Name)) + return false; + + if (Config.StripAll || Config.StripAllGNU) + return true; + + if (!Config.SymbolsToRemove.empty() && + is_contained(Config.SymbolsToRemove, Sym.Name)) { + if (Sym.Referenced) + reportError(Config.OutputFilename, + make_error("Can't remove referenced symbol " + + Sym.Name, + object_error::parse_failed)); + return true; + } + + // GNU binutils objcopy keeps referenced local symbols if Config.DiscaredAll + // is set. + if ((Config.StripUnneeded || Config.DiscardAll) && !Sym.Referenced && + (Sym.Sym.StorageClass == IMAGE_SYM_CLASS_STATIC || Obj.IsPE)) + return true; + + return false; + }); + + if (Obj.Symbols.size() != OrigNumSymbols) { + // Symbol list has changed, update relocations + StringMap SymMap; + uint32_t Idx = 0; + for (const Symbol &Sym : Obj.Symbols) { + SymMap[Sym.Name] = Idx; + Idx += 1 + Sym.Sym.NumberOfAuxSymbols; + } + for (Section &Sec : Obj.Sections) { + assert(Sec.Relocs.size() == Sec.RelocTargets.size()); + for (size_t I = 0, E = Sec.Relocs.size(); I < E; I++) { + auto It = SymMap.find(Sec.RelocTargets[I]); + if (It != SymMap.end()) + Sec.Relocs[I].SymbolTableIndex = It->second; + else + reportError(Config.OutputFilename, + make_error("Relocation to removed symbol " + + Sec.RelocTargets[I], + object_error::parse_failed)); + } + } + } + + if (!Config.AddGnuDebugLink.empty() && Obj.IsPE) { + if (Obj.Is64) + addGnuDebugLink(Obj, Obj.PePlusHeader, Config.AddGnuDebugLink); + else + addGnuDebugLink(Obj, Obj.PeHeader, Config.AddGnuDebugLink); + } +} + +void executeObjcopyOnBinary(const CopyConfig &Config, + object::COFFObjectFile &In, Buffer &Out) { + COFFReader Reader(In); + std::unique_ptr Obj = Reader.create(); + assert(Obj && "Unable to deserialize COFF object"); + handleArgs(Config, *Obj); + COFFWriter Writer(*Obj, Out); + Writer.write(); +} + +} // end namespace coff +} // end namespace objcopy +} // end namespace llvm Index: tools/llvm-objcopy/COFF/Object.h =================================================================== --- /dev/null +++ tools/llvm-objcopy/COFF/Object.h @@ -0,0 +1,135 @@ +//===- Object.h -------------------------------------------------*- C++ -*-===// +// +// The LLVM Compiler Infrastructure +// +// This file is distributed under the University of Illinois Open Source +// License. See LICENSE.TXT for details. +// +//===----------------------------------------------------------------------===// + +#ifndef LLVM_TOOLS_OBJCOPY_COFF_OBJECT_H +#define LLVM_TOOLS_OBJCOPY_COFF_OBJECT_H + +#include "Buffer.h" +#include "CopyConfig.h" +#include "llvm/ADT/ArrayRef.h" +#include "llvm/ADT/StringRef.h" +#include "llvm/BinaryFormat/COFF.h" +#include "llvm/MC/StringTableBuilder.h" +#include "llvm/Object/COFF.h" +#include "llvm/Support/FileOutputBuffer.h" +#include +#include +#include +#include + +namespace llvm { +namespace objcopy { +namespace coff { + +class Object; + +using object::Binary; +using object::COFFObjectFile; + +class Writer { +protected: + Object &Obj; + Buffer &Buf; + +public: + virtual ~Writer(); + virtual void write() = 0; + + Writer(Object &O, Buffer &B) : Obj(O), Buf(B) {} +}; + +class COFFWriter : public Writer { +private: + size_t FileSize; + size_t FileAlignment; + StringTableBuilder StrTabBuilder; + + template + void finalize(CoffHeaderTy &CoffFileHeader, PeHeaderTy &PeHeader); + template + void write(CoffHeaderTy &CoffFileHeader, PeHeaderTy &PeHeader); + + void patchDebugDirectory(); + +public: + virtual ~COFFWriter() {} + + void write() override; + COFFWriter(Object &Obj, Buffer &Buf) + : Writer(Obj, Buf), StrTabBuilder(StringTableBuilder::WinCOFF) {} +}; + +class Reader { +public: + virtual ~Reader(); + virtual std::unique_ptr create() const = 0; +}; + +class COFFReader : public Reader { + const COFFObjectFile &COFFObj; + +public: + std::unique_ptr create() const override; + explicit COFFReader(const COFFObjectFile &O) : COFFObj(O) {} +}; + +struct Section { + object::coff_section Header; + ArrayRef Contents; + std::vector Relocs; + StringRef Name; + + // Fields used by some transformations in objcopy, but not necessary + // for actually writing back to a file. + std::vector RelocTargets; + size_t Idx; + + // A container for data, for filling in Contents with synthesized data + // not originating from the input file. + std::vector LocalContents; +}; + +struct Symbol { + object::coff_symbol32 Sym; + StringRef Name; + ArrayRef AuxData; + + // Fields used by some transformations in objcopy, but not necessary + // for actually writing back to a file. + bool Referenced; +}; + +struct Object { + bool IsPE = false; + + object::dos_header DosHeader; + ArrayRef DosStub; + + bool IsBigObj = false; + object::coff_file_header CoffFileHeader; + object::coff_bigobj_file_header CoffBigObjFileHeader; + + bool Is64 = false; + object::pe32_header PeHeader; + object::pe32plus_header PePlusHeader; + + std::vector DataDirectories; + std::vector
Sections; + std::vector Symbols; + + void removeSymbols(function_ref ToRemove); + void removeSections(function_ref ToRemove); + void truncateSections(function_ref ToTruncate); +}; + +} // end namespace coff +} // end namespace objcopy +} // end namespace llvm + +#endif // LLVM_TOOLS_OBJCOPY_COFF_OBJECT_H Index: tools/llvm-objcopy/COFF/Object.cpp =================================================================== --- /dev/null +++ tools/llvm-objcopy/COFF/Object.cpp @@ -0,0 +1,376 @@ +//===- Object.cpp ---------------------------------------------------------===// +// +// The LLVM Compiler Infrastructure +// +// This file is distributed under the University of Illinois Open Source +// License. See LICENSE.TXT for details. +// +//===----------------------------------------------------------------------===// + +#include "Object.h" +#include "llvm-objcopy.h" +#include "llvm/ADT/ArrayRef.h" +#include "llvm/ADT/STLExtras.h" +#include "llvm/ADT/StringRef.h" +#include "llvm/BinaryFormat/COFF.h" +#include "llvm/Object/COFF.h" +#include "llvm/Support/Endian.h" +#include "llvm/Support/ErrorHandling.h" +#include "llvm/Support/FileOutputBuffer.h" +#include +#include +#include +#include + +namespace llvm { +namespace objcopy { +namespace coff { + +using namespace object; +using namespace COFF; + +Writer::~Writer() {} + +Reader::~Reader() {} + +template static void copySymbol(A &Dest, const B &Src) { + static_assert(sizeof(Dest.Name.ShortName) == sizeof(Src.Name.ShortName), + "Mismatched name sizes"); + memcpy(Dest.Name.ShortName, Src.Name.ShortName, sizeof(Dest.Name.ShortName)); + Dest.Value = Src.Value; + Dest.SectionNumber = Src.SectionNumber; + Dest.Type = Src.Type; + Dest.StorageClass = Src.StorageClass; + Dest.NumberOfAuxSymbols = Src.NumberOfAuxSymbols; +} + +std::unique_ptr COFFReader::create() const { + auto Obj = llvm::make_unique(); + + const coff_file_header *CFH = nullptr; + const coff_bigobj_file_header *CBFH = nullptr; + COFFObj.getCOFFHeader(CFH); + COFFObj.getCOFFBigObjHeader(CBFH); + if (CFH) { + Obj->CoffFileHeader = *CFH; + Obj->IsBigObj = false; + } else { + if (!CBFH) + reportError(COFFObj.getFileName(), + make_error("No COFF file header returned", + object_error::parse_failed)); + Obj->CoffBigObjFileHeader = *CBFH; + Obj->IsBigObj = true; + } + + const dos_header *DH = COFFObj.getDOSHeader(); + Obj->Is64 = COFFObj.is64(); + if (DH) { + Obj->IsPE = true; + Obj->DosHeader = *DH; + if (DH->AddressOfNewExeHeader > sizeof(*DH)) + Obj->DosStub = + ArrayRef(reinterpret_cast(&DH[1]), + DH->AddressOfNewExeHeader - sizeof(*DH)); + + const pe32_header *PE32 = nullptr; + const pe32plus_header *PE32Plus = nullptr; + if (COFFObj.is64()) { + if (auto EC = COFFObj.getPE32PlusHeader(PE32Plus)) + reportError(COFFObj.getFileName(), std::move(EC)); + Obj->PePlusHeader = *PE32Plus; + } else { + if (auto EC = COFFObj.getPE32Header(PE32)) + reportError(COFFObj.getFileName(), std::move(EC)); + Obj->PeHeader = *PE32; + } + + size_t NumberOfDataDirectory = + PE32Plus ? PE32Plus->NumberOfRvaAndSize : PE32->NumberOfRvaAndSize; + for (size_t I = 0; I < NumberOfDataDirectory; I++) { + const data_directory *Dir; + if (auto EC = COFFObj.getDataDirectory(I, Dir)) + reportError(COFFObj.getFileName(), std::move(EC)); + Obj->DataDirectories.emplace_back(*Dir); + } + } + + for (size_t I = 1, E = COFFObj.getNumberOfSections(); I <= E; I++) { + const coff_section *Sec; + if (auto EC = COFFObj.getSection(I, Sec)) + reportError(COFFObj.getFileName(), std::move(EC)); + Obj->Sections.push_back(Section()); + Section &S = Obj->Sections.back(); + S.Header = *Sec; + if (auto EC = COFFObj.getSectionContents(Sec, S.Contents)) + reportError(COFFObj.getFileName(), std::move(EC)); + ArrayRef Relocs = COFFObj.getRelocations(Sec); + S.Relocs.insert(S.Relocs.end(), Relocs.begin(), Relocs.end()); + if (auto EC = COFFObj.getSectionName(Sec, S.Name)) + reportError(COFFObj.getFileName(), std::move(EC)); + if (Sec->hasExtendedRelocations()) + reportError( + COFFObj.getFileName(), + make_error("Extended relocations not supported yet", + object_error::parse_failed)); + } + + for (uint32_t I = 0, E = COFFObj.getRawNumberOfSymbols(); I < E;) { + Expected SymOrErr = COFFObj.getSymbol(I); + if (!SymOrErr) + reportError(COFFObj.getFileName(), SymOrErr.takeError()); + COFFSymbolRef SymRef = *SymOrErr; + + Obj->Symbols.push_back(Symbol()); + Symbol &Sym = Obj->Symbols.back(); + if (Obj->IsBigObj) + copySymbol(Sym.Sym, + *reinterpret_cast(SymRef.getRawPtr())); + else + copySymbol(Sym.Sym, + *reinterpret_cast(SymRef.getRawPtr())); + if (auto EC = COFFObj.getSymbolName(SymRef, Sym.Name)) + reportError(COFFObj.getFileName(), std::move(EC)); + Sym.AuxData = COFFObj.getSymbolAuxData(SymRef); + assert((Sym.AuxData.size() % (Obj->IsBigObj ? sizeof(coff_symbol32) + : sizeof(coff_symbol16))) == 0); + I += 1 + SymRef.getNumberOfAuxSymbols(); + } + + return Obj; +} + +template +void COFFWriter::finalize(CoffHeaderTy &CoffFileHeader, PeHeaderTy &PeHeader) { + size_t SizeOfHeaders = 0; + size_t FileAlignment = 1; + if (Obj.IsPE) { + Obj.DosHeader.AddressOfNewExeHeader = + sizeof(Obj.DosHeader) + Obj.DosStub.size(); + SizeOfHeaders += Obj.DosHeader.AddressOfNewExeHeader + sizeof(PEMagic); + + FileAlignment = PeHeader.FileAlignment; + PeHeader.NumberOfRvaAndSize = Obj.DataDirectories.size(); + + SizeOfHeaders += + sizeof(PeHeader) + sizeof(data_directory) * Obj.DataDirectories.size(); + } + CoffFileHeader.NumberOfSections = Obj.Sections.size(); + SizeOfHeaders += sizeof(CoffFileHeader); + SizeOfHeaders += sizeof(coff_section) * Obj.Sections.size(); + SizeOfHeaders = alignTo(SizeOfHeaders, FileAlignment); + + // Directly accessing Obj.CoffFileHeader, as CoffBigObjFileHeader doesn't + // have this field. + if (!Obj.IsBigObj && Obj.IsPE) + Obj.CoffFileHeader.SizeOfOptionalHeader = + sizeof(PeHeader) + sizeof(data_directory) * Obj.DataDirectories.size(); + + FileSize = SizeOfHeaders; + size_t SizeOfInitializedData = 0; + for (auto &S : Obj.Sections) { + if (S.Header.SizeOfRawData > 0) + S.Header.PointerToRawData = FileSize; + FileSize += S.Header.SizeOfRawData; // For executables, this is aligned to + // FileAlignment. + if (S.Header.NumberOfRelocations > 0) + S.Header.PointerToRelocations = FileSize; + FileSize += S.Relocs.size() * sizeof(coff_relocation); + FileSize = alignTo(FileSize, FileAlignment); + if (S.Name.size() > COFF::NameSize) + StrTabBuilder.add(S.Name); + if (S.Header.Characteristics & IMAGE_SCN_CNT_INITIALIZED_DATA) + SizeOfInitializedData += S.Header.SizeOfRawData; + } + if (Obj.IsPE) { + PeHeader.SizeOfHeaders = SizeOfHeaders; + PeHeader.SizeOfInitializedData = SizeOfInitializedData; + PeHeader.CheckSum = 0; + if (!Obj.Sections.empty()) { + const Section &S = Obj.Sections.back(); + PeHeader.SizeOfImage = + alignTo(S.Header.VirtualAddress + S.Header.VirtualSize, + PeHeader.SectionAlignment); + } + } + + size_t SymTabSize = Obj.Symbols.size() * sizeof(SymbolTy); + for (const auto &S : Obj.Symbols) { + SymTabSize += S.AuxData.size(); + if (S.Name.size() > COFF::NameSize) + StrTabBuilder.add(S.Name); + } + + StrTabBuilder.finalize(); + + for (auto &S : Obj.Sections) { + if (S.Name.size() > COFF::NameSize) { + snprintf(S.Header.Name, sizeof(S.Header.Name), "/%d", + (int)StrTabBuilder.getOffset(S.Name)); + } else { + strncpy(S.Header.Name, S.Name.data(), COFF::NameSize); + } + } + for (auto &S : Obj.Symbols) { + if (S.Name.size() > COFF::NameSize) { + S.Sym.Name.Offset.Zeroes = 0; + S.Sym.Name.Offset.Offset = StrTabBuilder.getOffset(S.Name); + } else { + strncpy(S.Sym.Name.ShortName, S.Name.data(), COFF::NameSize); + } + } + + size_t StrTabSize = StrTabBuilder.getSize(); + size_t PointerToSymbolTable = FileSize; + if (SymTabSize == 0 && StrTabSize <= 4) { + PointerToSymbolTable = 0; + // For executables, skip the length field of an empty string table. + if (Obj.IsPE) + StrTabSize = 0; + } + + size_t NumRawSymbols = SymTabSize / sizeof(SymbolTy); + CoffFileHeader.PointerToSymbolTable = PointerToSymbolTable; + CoffFileHeader.NumberOfSymbols = NumRawSymbols; + FileSize += SymTabSize + StrTabSize; + FileSize = alignTo(FileSize, FileAlignment); +} + +template +void COFFWriter::write(CoffHeaderTy &CoffFileHeader, PeHeaderTy &PeHeader) { + finalize(CoffFileHeader, PeHeader); + + Buf.allocate(FileSize); + + uint8_t *Ptr = Buf.getBufferStart(); + if (Obj.IsPE) { + memcpy(Ptr, &Obj.DosHeader, sizeof(Obj.DosHeader)); + Ptr += sizeof(Obj.DosHeader); + memcpy(Ptr, Obj.DosStub.data(), Obj.DosStub.size()); + Ptr += Obj.DosStub.size(); + memcpy(Ptr, PEMagic, sizeof(PEMagic)); + Ptr += sizeof(PEMagic); + } + memcpy(Ptr, &CoffFileHeader, sizeof(CoffFileHeader)); + Ptr += sizeof(CoffFileHeader); + if (Obj.IsPE) { + memcpy(Ptr, &PeHeader, sizeof(PeHeader)); + Ptr += sizeof(PeHeader); + for (const auto &DD : Obj.DataDirectories) { + memcpy(Ptr, &DD, sizeof(DD)); + Ptr += sizeof(DD); + } + } + for (const auto &S : Obj.Sections) { + memcpy(Ptr, &S.Header, sizeof(S.Header)); + Ptr += sizeof(S.Header); + } + + for (const auto &S : Obj.Sections) { + Ptr = Buf.getBufferStart() + S.Header.PointerToRawData; + memcpy(Ptr, S.Contents.data(), S.Contents.size()); + if ((S.Header.Characteristics & IMAGE_SCN_CNT_CODE) && + S.Header.SizeOfRawData > S.Contents.size()) + memset(Ptr + S.Contents.size(), 0xcc, + S.Header.SizeOfRawData - S.Contents.size()); + Ptr += S.Header.SizeOfRawData; + memcpy(Ptr, S.Relocs.data(), S.Relocs.size() * sizeof(coff_relocation)); + } + + Ptr = Buf.getBufferStart() + CoffFileHeader.PointerToSymbolTable; + for (const auto &S : Obj.Symbols) { + copySymbol(*reinterpret_cast(Ptr), + S.Sym); + Ptr += sizeof(SymbolTy); + memcpy(Ptr, S.AuxData.data(), S.AuxData.size()); + Ptr += S.AuxData.size(); + } + if (StrTabBuilder.getSize() > 4 || !Obj.IsPE) { + // Always write a string table in object files, even an empty one. + StrTabBuilder.write(Ptr); + Ptr += StrTabBuilder.getSize(); + } + + if (Obj.IsPE && Obj.DataDirectories.size() >= DEBUG_DIRECTORY) + patchDebugDirectory(); + + if (auto E = Buf.commit()) + reportError(Buf.getName(), errorToErrorCode(std::move(E))); +} + +void COFFWriter::patchDebugDirectory() { + const data_directory *Dir = &Obj.DataDirectories[DEBUG_DIRECTORY]; + if (Dir->Size <= 0) + return; + bool Found = false; + for (const auto &S : Obj.Sections) { + if (Dir->RelativeVirtualAddress >= S.Header.VirtualAddress && + Dir->RelativeVirtualAddress < + S.Header.VirtualAddress + S.Header.SizeOfRawData) { + if (Dir->RelativeVirtualAddress + Dir->Size > + S.Header.VirtualAddress + S.Header.SizeOfRawData) + reportError(Buf.getName(), + make_error( + "Debug directory extends past end of section", + object_error::parse_failed)); + size_t Offset = Dir->RelativeVirtualAddress - S.Header.VirtualAddress; + uint8_t *Ptr = Buf.getBufferStart() + S.Header.PointerToRawData + Offset; + uint8_t *End = Ptr + Dir->Size; + while (Ptr < End) { + debug_directory *Debug = reinterpret_cast(Ptr); + Debug->PointerToRawData = + S.Header.PointerToRawData + Offset + sizeof(debug_directory); + Ptr += sizeof(debug_directory) + Debug->SizeOfData; + Offset += sizeof(debug_directory) + Debug->SizeOfData; + } + Found = true; + break; + } + } + if (!Found) + reportError(Buf.getName(), + make_error("Debug directory missing", + object_error::parse_failed)); +} + +void COFFWriter::write() { + assert(!(Obj.IsBigObj && Obj.IsPE) && + "Can't have BigObj headers in a PE executable"); + if (Obj.IsBigObj) + write( + Obj.CoffBigObjFileHeader, Obj.PeHeader); + else if (Obj.Is64) + write(Obj.CoffFileHeader, + Obj.PePlusHeader); + else + write(Obj.CoffFileHeader, + Obj.PeHeader); +} + +void Object::removeSymbols(function_ref ToRemove) { + Symbols.erase( + std::remove_if(std::begin(Symbols), std::end(Symbols), + [ToRemove](const Symbol &Sym) { return ToRemove(Sym); }), + std::end(Symbols)); +} + +void Object::removeSections(function_ref ToRemove) { + Sections.erase( + std::remove_if(std::begin(Sections), std::end(Sections), + [ToRemove](const Section &Sec) { return ToRemove(Sec); }), + std::end(Sections)); +} + +void Object::truncateSections(function_ref ToTruncate) { + for (Section &Sec : Sections) { + if (ToTruncate(Sec)) { + Sec.Contents = ArrayRef(); + Sec.Relocs = ArrayRef(); + Sec.Header.SizeOfRawData = 0; + } + } +} + +} // end namespace coff +} // end namespace objcopy +} // end namespace llvm Index: tools/llvm-objcopy/llvm-objcopy.cpp =================================================================== --- tools/llvm-objcopy/llvm-objcopy.cpp +++ tools/llvm-objcopy/llvm-objcopy.cpp @@ -9,6 +9,7 @@ #include "llvm-objcopy.h" #include "Buffer.h" +#include "COFF/COFFObjcopy.h" #include "CopyConfig.h" #include "ELF/ELFObjcopy.h" @@ -19,6 +20,7 @@ #include "llvm/Object/Archive.h" #include "llvm/Object/ArchiveWriter.h" #include "llvm/Object/Binary.h" +#include "llvm/Object/COFF.h" #include "llvm/Object/ELFObjectFile.h" #include "llvm/Object/ELFTypes.h" #include "llvm/Object/Error.h" @@ -125,6 +127,8 @@ Buffer &Out) { if (auto *ELFBinary = dyn_cast(&In)) return elf::executeObjcopyOnBinary(Config, *ELFBinary, Out); + else if (auto *COFFBinary = dyn_cast(&In)) + return coff::executeObjcopyOnBinary(Config, *COFFBinary, Out); else error("Unsupported object file format"); }