diff options
-rw-r--r-- | llvm/test/tools/llvm-objcopy/basic-keep.test | 19 | ||||
-rw-r--r-- | llvm/test/tools/llvm-objcopy/basic-only-keep.test | 23 | ||||
-rw-r--r-- | llvm/test/tools/llvm-objcopy/dump-section.test | 28 | ||||
-rw-r--r-- | llvm/test/tools/llvm-objcopy/explicit-keep-remove.test | 21 | ||||
-rw-r--r-- | llvm/test/tools/llvm-objcopy/explicit-only-keep-remove.test | 21 | ||||
-rw-r--r-- | llvm/test/tools/llvm-objcopy/keep-many.test | 27 | ||||
-rw-r--r-- | llvm/test/tools/llvm-objcopy/keep-only-keep.test | 27 | ||||
-rw-r--r-- | llvm/test/tools/llvm-objcopy/only-keep-many.test | 28 | ||||
-rw-r--r-- | llvm/test/tools/llvm-objcopy/only-keep-remove-strtab.test | 21 | ||||
-rw-r--r-- | llvm/test/tools/llvm-objcopy/only-keep-strip-non-alloc.test | 19 | ||||
-rw-r--r-- | llvm/test/tools/llvm-objcopy/strip-sections-keep.test | 13 | ||||
-rw-r--r-- | llvm/test/tools/llvm-objcopy/strip-sections-only-keep.test | 20 | ||||
-rw-r--r-- | llvm/tools/llvm-objcopy/Object.h | 2 | ||||
-rw-r--r-- | llvm/tools/llvm-objcopy/llvm-objcopy.cpp | 53 |
14 files changed, 322 insertions, 0 deletions
diff --git a/llvm/test/tools/llvm-objcopy/basic-keep.test b/llvm/test/tools/llvm-objcopy/basic-keep.test new file mode 100644 index 00000000000..2ea4ea35577 --- /dev/null +++ b/llvm/test/tools/llvm-objcopy/basic-keep.test @@ -0,0 +1,19 @@ +# RUN: yaml2obj %s > %t +# RUN: llvm-objcopy -strip-non-alloc -keep=.test %t %t2 +# RUN: llvm-readobj -file-headers -sections %t2 | FileCheck %s + +!ELF +FileHeader: + Class: ELFCLASS64 + Data: ELFDATA2LSB + Type: ET_REL + Machine: EM_X86_64 +Sections: + - Name: .test + Type: SHT_PROGBITS + Flags: [ ] + +# CHECK: SectionHeaderCount: 3 + +# CHECK: Name: .test +# CHECK: Name: .shstrtab diff --git a/llvm/test/tools/llvm-objcopy/basic-only-keep.test b/llvm/test/tools/llvm-objcopy/basic-only-keep.test new file mode 100644 index 00000000000..4d97012a6cc --- /dev/null +++ b/llvm/test/tools/llvm-objcopy/basic-only-keep.test @@ -0,0 +1,23 @@ +# RUN: yaml2obj %s > %t +# RUN: llvm-objcopy -only-keep=.test %t %t2 +# RUN: llvm-objcopy -j=.test %t %t3 +# RUN: llvm-readobj -file-headers -sections %t2 | FileCheck %s +# RUN: diff %t2 %t3 + +!ELF +FileHeader: + Class: ELFCLASS64 + Data: ELFDATA2LSB + Type: ET_REL + Machine: EM_X86_64 +Sections: + - Name: .test + Type: SHT_PROGBITS + Flags: [ ] + +# CHECK: SectionHeaderCount: 5 + +# CHECK: Name: .test +# CHECK: Name: .symtab +# CHECK: Name: .strtab +# CHECK: Name: .shstrtab diff --git a/llvm/test/tools/llvm-objcopy/dump-section.test b/llvm/test/tools/llvm-objcopy/dump-section.test new file mode 100644 index 00000000000..123f4736ca0 --- /dev/null +++ b/llvm/test/tools/llvm-objcopy/dump-section.test @@ -0,0 +1,28 @@ +# RUN: yaml2obj %s > %t +# RUN: llvm-objcopy -O binary -j .text %t %t2 +# RUN: llvm-objcopy -O binary -only-keep .text %t %t3 +# RUN: od -t x1 %t2 | FileCheck %s +# RUN: wc -c %t2 | FileCheck %s --check-prefix=SIZE +# RUN: diff %t2 %t3 + +!ELF +FileHeader: + Class: ELFCLASS64 + Data: ELFDATA2LSB + Type: ET_EXEC + Machine: EM_X86_64 +Sections: + - Name: .text + Type: SHT_PROGBITS + Flags: [ SHF_ALLOC, SHF_EXECINSTR ] + AddressAlign: 0x0000000000001000 + Content: "DEADBEEF" +ProgramHeaders: +- Type: PT_LOAD + Flags: [ PF_X, PF_R ] + Sections: + - Section: .text + +#CHECK: 0000000 de ad be ef + +#SIZE: 4 diff --git a/llvm/test/tools/llvm-objcopy/explicit-keep-remove.test b/llvm/test/tools/llvm-objcopy/explicit-keep-remove.test new file mode 100644 index 00000000000..5ebd2a5081c --- /dev/null +++ b/llvm/test/tools/llvm-objcopy/explicit-keep-remove.test @@ -0,0 +1,21 @@ +# RUN: yaml2obj %s > %t +# RUN: llvm-objcopy -R=.test -keep=.test %t %t2 +# RUN: llvm-readobj -file-headers -sections %t2 | FileCheck %s + +!ELF +FileHeader: + Class: ELFCLASS64 + Data: ELFDATA2LSB + Type: ET_REL + Machine: EM_X86_64 +Sections: + - Name: .test + Type: SHT_PROGBITS + Flags: [ ] + +# CHECK: SectionHeaderCount: 5 + +# CHECK: Name: .test +# CHECK: Name: .symtab +# CHECK: Name: .strtab +# CHECK: Name: .shstrtab diff --git a/llvm/test/tools/llvm-objcopy/explicit-only-keep-remove.test b/llvm/test/tools/llvm-objcopy/explicit-only-keep-remove.test new file mode 100644 index 00000000000..10d49e10c09 --- /dev/null +++ b/llvm/test/tools/llvm-objcopy/explicit-only-keep-remove.test @@ -0,0 +1,21 @@ +# RUN: yaml2obj %s > %t +# RUN: llvm-objcopy -R=.test -only-keep=.test %t %t2 +# RUN: llvm-readobj -file-headers -sections %t2 | FileCheck %s + +!ELF +FileHeader: + Class: ELFCLASS64 + Data: ELFDATA2LSB + Type: ET_REL + Machine: EM_X86_64 +Sections: + - Name: .test + Type: SHT_PROGBITS + Flags: [ ] + +# CHECK: SectionHeaderCount: 5 + +# CHECK: Name: .test +# CHECK: Name: .symtab +# CHECK: Name: .strtab +# CHECK: Name: .shstrtab diff --git a/llvm/test/tools/llvm-objcopy/keep-many.test b/llvm/test/tools/llvm-objcopy/keep-many.test new file mode 100644 index 00000000000..66273778993 --- /dev/null +++ b/llvm/test/tools/llvm-objcopy/keep-many.test @@ -0,0 +1,27 @@ +# RUN: yaml2obj %s > %t +# RUN: llvm-objcopy -strip-non-alloc -keep=.test -keep=.test3 %t %t2 +# RUN: llvm-readobj -file-headers -sections %t2 | FileCheck %s + +!ELF +FileHeader: + Class: ELFCLASS64 + Data: ELFDATA2LSB + Type: ET_REL + Machine: EM_X86_64 +Sections: + - Name: .test + Type: SHT_PROGBITS + Flags: [ ] + - Name: .test2 + Type: SHT_PROGBITS + Flags: [ ] + - Name: .test3 + Type: SHT_PROGBITS + Flags: [ ] + + +# CHECK: SectionHeaderCount: 4 + +# CHECK: Name: .test +# CHECK: Name: .test3 +# CHECK: Name: .shstrtab diff --git a/llvm/test/tools/llvm-objcopy/keep-only-keep.test b/llvm/test/tools/llvm-objcopy/keep-only-keep.test new file mode 100644 index 00000000000..b1f08fe5879 --- /dev/null +++ b/llvm/test/tools/llvm-objcopy/keep-only-keep.test @@ -0,0 +1,27 @@ +# RUN: yaml2obj %s > %t +# RUN: llvm-objcopy -keep=.test2 -only-keep=.test %t %t2 +# RUN: llvm-objcopy -j=.test -keep=.test2 %t %t3 +# RUN: llvm-readobj -file-headers -sections %t2 | FileCheck %s +# RUN: diff %t2 %t3 + +!ELF +FileHeader: + Class: ELFCLASS64 + Data: ELFDATA2LSB + Type: ET_REL + Machine: EM_X86_64 +Sections: + - Name: .test + Type: SHT_PROGBITS + - Name: .test2 + Type: SHT_PROGBITS + - Name: .test3 + Type: SHT_PROGBITS + +# CHECK: SectionHeaderCount: 6 + +# CHECK: Name: .test +# CHECK: Name: .test2 +# CHECK: Name: .symtab +# CHECK: Name: .strtab +# CHECK: Name: .shstrtab diff --git a/llvm/test/tools/llvm-objcopy/only-keep-many.test b/llvm/test/tools/llvm-objcopy/only-keep-many.test new file mode 100644 index 00000000000..2f956238dc9 --- /dev/null +++ b/llvm/test/tools/llvm-objcopy/only-keep-many.test @@ -0,0 +1,28 @@ +# RUN: yaml2obj %s > %t +# RUN: llvm-objcopy -j .test1 -j .test2 %t %t2 +# RUN: llvm-readobj -file-headers -sections %t2 | FileCheck %s + +!ELF +FileHeader: + Class: ELFCLASS64 + Data: ELFDATA2LSB + Type: ET_REL + Machine: EM_X86_64 +Sections: + - Name: .test1 + Type: SHT_PROGBITS + Flags: [ ] + - Name: .test2 + Type: SHT_PROGBITS + Flags: [ ] + - Name: .test3 + Type: SHT_PROGBITS + Flags: [ ] + +# CHECK: SectionHeaderCount: 6 + +# CHECK: Name: .test1 +# CHECK: Name: .test2 +# CHECK: Name: .symtab +# CHECK: Name: .strtab +# CHECK: Name: .shstrtab diff --git a/llvm/test/tools/llvm-objcopy/only-keep-remove-strtab.test b/llvm/test/tools/llvm-objcopy/only-keep-remove-strtab.test new file mode 100644 index 00000000000..c8946d3294d --- /dev/null +++ b/llvm/test/tools/llvm-objcopy/only-keep-remove-strtab.test @@ -0,0 +1,21 @@ +# RUN: yaml2obj %s > %t +# RUN: llvm-objcopy -R .symtab -R .strtab -only-keep=.test %t %t2 +# RUN: llvm-objcopy -j=.test -R .strtab -R .symtab %t %t3 +# RUN: llvm-readobj -file-headers -sections %t2 | FileCheck %s +# RUN: diff %t2 %t3 + +!ELF +FileHeader: + Class: ELFCLASS64 + Data: ELFDATA2LSB + Type: ET_REL + Machine: EM_X86_64 +Sections: + - Name: .test + Type: SHT_PROGBITS + Flags: [ ] + +# CHECK: SectionHeaderCount: 3 + +# CHECK: Name: .test +# CHECK: Name: .shstrtab diff --git a/llvm/test/tools/llvm-objcopy/only-keep-strip-non-alloc.test b/llvm/test/tools/llvm-objcopy/only-keep-strip-non-alloc.test new file mode 100644 index 00000000000..bad00228492 --- /dev/null +++ b/llvm/test/tools/llvm-objcopy/only-keep-strip-non-alloc.test @@ -0,0 +1,19 @@ +# RUN: yaml2obj %s > %t +# RUN: llvm-objcopy -strip-non-alloc -only-keep=.test %t %t2 +# RUN: llvm-readobj -file-headers -sections %t2 | FileCheck %s + +!ELF +FileHeader: + Class: ELFCLASS64 + Data: ELFDATA2LSB + Type: ET_REL + Machine: EM_X86_64 +Sections: + - Name: .test + Type: SHT_PROGBITS + Flags: [ ] + +# CHECK: SectionHeaderCount: 3 + +# CHECK: Name: .test +# CHECK: Name: .shstrtab diff --git a/llvm/test/tools/llvm-objcopy/strip-sections-keep.test b/llvm/test/tools/llvm-objcopy/strip-sections-keep.test new file mode 100644 index 00000000000..dcf69685382 --- /dev/null +++ b/llvm/test/tools/llvm-objcopy/strip-sections-keep.test @@ -0,0 +1,13 @@ +# RUN: yaml2obj %s > %t +# RUN: llvm-objcopy -strip-sections -keep=.shstrtab %t %t2 +# RUN: od -Ax -t c %t2 | FileCheck %s + +!ELF +FileHeader: + Class: ELFCLASS64 + Data: ELFDATA2LSB + Type: ET_REL + Machine: EM_X86_64 +Sections: + +# CHECK: \0 . s h s t r t a b \0 diff --git a/llvm/test/tools/llvm-objcopy/strip-sections-only-keep.test b/llvm/test/tools/llvm-objcopy/strip-sections-only-keep.test new file mode 100644 index 00000000000..2c9400cf34c --- /dev/null +++ b/llvm/test/tools/llvm-objcopy/strip-sections-only-keep.test @@ -0,0 +1,20 @@ +# RUN: yaml2obj %s > %t +# RUN: llvm-objcopy -strip-sections -only-keep=.test %t %t2 +# RUN: od -Ax -t x1 %t2 | FileCheck %s +# RUN: od -Ax -t c %t2 | FileCheck %s -check-prefix=TEXT + +!ELF +FileHeader: + Class: ELFCLASS64 + Data: ELFDATA2LSB + Type: ET_REL + Machine: EM_X86_64 +Sections: + - Name: .test + Type: SHT_PROGBITS + Flags: [ ] + Content: "DEADBEEF" + +# CHECK: de ad be ef + +# TEXT-NOT: t e s t diff --git a/llvm/tools/llvm-objcopy/Object.h b/llvm/tools/llvm-objcopy/Object.h index f12e6da7d21..9f98c04ad9b 100644 --- a/llvm/tools/llvm-objcopy/Object.h +++ b/llvm/tools/llvm-objcopy/Object.h @@ -196,6 +196,7 @@ public: SectionBase *DefinedIn, uint64_t Value, uint16_t Shndx, uint64_t Sz); void addSymbolNames(); + const SectionBase *getStrTab() const { return SymbolNames; } const Symbol *getSymbolByIndex(uint32_t Index) const; void removeSectionReferences(const SectionBase *Sec) override; void initialize(SectionTableRef SecTable) override; @@ -368,6 +369,7 @@ public: Object(const object::ELFObjectFile<ELFT> &Obj); virtual ~Object() = default; + const SymbolTableSection *getSymTab() const { return SymbolTable; } const SectionBase *getSectionHeaderStrTab() const { return SectionNames; } void removeSections(std::function<bool(const SectionBase &)> ToRemove); virtual size_t totalSize() const = 0; diff --git a/llvm/tools/llvm-objcopy/llvm-objcopy.cpp b/llvm/tools/llvm-objcopy/llvm-objcopy.cpp index 09553e85202..615667bf99a 100644 --- a/llvm/tools/llvm-objcopy/llvm-objcopy.cpp +++ b/llvm/tools/llvm-objcopy/llvm-objcopy.cpp @@ -89,6 +89,13 @@ static cl::opt<bool> StripAll( static cl::opt<bool> StripAllGNU("strip-all-gnu", cl::desc("Removes symbol, relocation, and debug information")); +static cl::list<std::string> Keep("keep", cl::desc("Keep <section>"), + cl::value_desc("section")); +static cl::list<std::string> OnlyKeep("only-keep", + cl::desc("Remove all but <section>"), + cl::value_desc("section")); +static cl::alias OnlyKeepA("j", cl::desc("Alias for only-keep"), + cl::aliasopt(OnlyKeep)); static cl::opt<bool> StripDebug("strip-debug", cl::desc("Removes all debug information")); static cl::opt<bool> StripSections("strip-sections", @@ -150,6 +157,13 @@ void SplitDWOToFile(const ELFObjectFile<ELFT> &ObjFile, StringRef File) { WriteObjectFile(DWOFile, File); } +// This function handles the high level operations of GNU objcopy including +// handling command line options. It's important to outline certain properties +// we expect to hold of the command line operations. Any operation that "keeps" +// should keep regardless of a remove. Additionally any removal should respect +// any previous removals. Lastly whether or not something is removed shouldn't +// depend a) on the order the options occur in or b) on some opaque priority +// system. The only priority is that keeps/copies overrule removes. template <class ELFT> void CopyBinary(const ELFObjectFile<ELFT> &ObjFile) { std::unique_ptr<Object<ELFT>> Obj; @@ -166,6 +180,8 @@ void CopyBinary(const ELFObjectFile<ELFT> &ObjFile) { SectionPred RemovePred = [](const SectionBase &) { return false; }; + // Removes: + if (!ToRemove.empty()) { RemovePred = [&](const SectionBase &Sec) { return std::find(std::begin(ToRemove), std::end(ToRemove), Sec.Name) != @@ -234,6 +250,43 @@ void CopyBinary(const ELFObjectFile<ELFT> &ObjFile) { return (Sec.Flags & SHF_ALLOC) == 0; }; + // Explicit copies: + + if (!OnlyKeep.empty()) { + RemovePred = [RemovePred, &Obj](const SectionBase &Sec) { + // Explicitly keep these sections regardless of previous removes. + if (std::find(std::begin(OnlyKeep), std::end(OnlyKeep), Sec.Name) != + std::end(OnlyKeep)) + return false; + + // Allow all implicit removes. + if (RemovePred(Sec)) { + return true; + } + + // Keep special sections. + if (Obj->getSectionHeaderStrTab() == &Sec) { + return false; + } + if (Obj->getSymTab() == &Sec || Obj->getSymTab()->getStrTab() == &Sec) { + return false; + } + // Remove everything else. + return true; + }; + } + + if (!Keep.empty()) { + RemovePred = [RemovePred](const SectionBase &Sec) { + // Explicitly keep these sections regardless of previous removes. + if (std::find(std::begin(Keep), std::end(Keep), Sec.Name) != + std::end(Keep)) + return false; + // Otherwise defer to RemovePred. + return RemovePred(Sec); + }; + } + Obj->removeSections(RemovePred); Obj->finalize(); WriteObjectFile(*Obj, OutputFilename.getValue()); |