diff --git a/include/llvm/Object/Archive.h b/include/llvm/Object/Archive.h index c0513a7c20f..1cba51954ec 100644 --- a/include/llvm/Object/Archive.h +++ b/include/llvm/Object/Archive.h @@ -188,6 +188,8 @@ public: // check if a symbol is in the archive child_iterator findSym(StringRef name) const; + bool hasSymbolTable() const; + private: child_iterator SymbolTable; child_iterator StringTable; diff --git a/lib/Object/Archive.cpp b/lib/Object/Archive.cpp index c6cf219c09b..71efca2b186 100644 --- a/lib/Object/Archive.cpp +++ b/lib/Object/Archive.cpp @@ -385,7 +385,7 @@ Archive::Symbol Archive::Symbol::getNext() const { } Archive::symbol_iterator Archive::begin_symbols() const { - if (SymbolTable == end_children()) + if (!hasSymbolTable()) return symbol_iterator(Symbol(this, 0, 0)); const char *buf = SymbolTable->getBuffer().begin(); @@ -408,7 +408,7 @@ Archive::symbol_iterator Archive::begin_symbols() const { } Archive::symbol_iterator Archive::end_symbols() const { - if (SymbolTable == end_children()) + if (!hasSymbolTable()) return symbol_iterator(Symbol(this, 0, 0)); const char *buf = SymbolTable->getBuffer().begin(); @@ -444,3 +444,7 @@ Archive::child_iterator Archive::findSym(StringRef name) const { } return end_children(); } + +bool Archive::hasSymbolTable() const { + return SymbolTable != end_children(); +} diff --git a/test/Object/Inputs/archive-test.a-corrupt-symbol-table b/test/Object/Inputs/archive-test.a-corrupt-symbol-table new file mode 100644 index 00000000000..34e5ed799d3 Binary files /dev/null and b/test/Object/Inputs/archive-test.a-corrupt-symbol-table differ diff --git a/test/Object/archive-symtab.test b/test/Object/archive-symtab.test index 1cf0ce449d8..0d2504d2561 100644 --- a/test/Object/archive-symtab.test +++ b/test/Object/archive-symtab.test @@ -16,3 +16,35 @@ CHECK-NEXT: trivial-object-test2.elf-x86-64: CHECK-NEXT: 00000000 t bar CHECK-NEXT: 00000006 T foo CHECK-NEXT: 00000016 T main + +RUN: rm -f %t.a +RUN: llvm-ar rcS %t.a %p/Inputs/trivial-object-test.elf-x86-64 %p/Inputs/trivial-object-test2.elf-x86-64 +RUN: llvm-nm -s %t.a | FileCheck %s --check-prefix=NOMAP + +NOMAP-NOT: Archive map + +RUN: llvm-ar s %t.a +RUN: llvm-nm -s %t.a | FileCheck %s + +check that the archive does have a corrupt symbol table. +RUN: rm -f %t.a +RUN: cp %p/Inputs/archive-test.a-corrupt-symbol-table %t.a +RUN: llvm-nm -s %t.a | FileCheck %s --check-prefix=CORRUPT + +CORRUPT: Archive map +CORRUPT-NEXT: mbin in trivial-object-test.elf-x86-64 +CORRUPT-NEXT: foo in trivial-object-test2.elf-x86-64 +CORRUPT-NEXT: main in trivial-object-test2.elf-x86-64 + +CORRUPT: trivial-object-test.elf-x86-64: +CORRUPT-NEXT: U SomeOtherFunction +CORRUPT-NEXT: 00000000 T main +CORRUPT-NEXT: U puts +CORRUPT-NEXT: trivial-object-test2.elf-x86-64: +CORRUPT-NEXT: 00000000 t bar +CORRUPT-NEXT: 00000006 T foo +CORRUPT-NEXT: 00000016 T main + +check that the we *don't* update the symbol table. +RUN: llvm-ar s %t.a +RUN: llvm-nm -s %t.a | FileCheck %s --check-prefix=CORRUPT diff --git a/tools/llvm-ar/llvm-ar.cpp b/tools/llvm-ar/llvm-ar.cpp index 261446cc782..6026fa7cfea 100644 --- a/tools/llvm-ar/llvm-ar.cpp +++ b/tools/llvm-ar/llvm-ar.cpp @@ -112,7 +112,8 @@ enum ArchiveOperation { QuickAppend, ///< Quickly append to end of archive ReplaceOrInsert, ///< Replace or Insert members DisplayTable, ///< Display the table of contents - Extract ///< Extract files back to file system + Extract, ///< Extract files back to file system + CreateSymTab ///< Create a symbol table in an existing archive }; // Modifiers to follow operation to vary behavior @@ -186,6 +187,8 @@ static ArchiveOperation parseCommandLine() { // Keep track of which operation was requested ArchiveOperation Operation; + bool MaybeJustCreateSymTab = false; + for(unsigned i=0; ihasSymbolTable()) + return; + + performWriteOperation(CreateSymTab, OldArchive); +} + static void performOperation(ArchiveOperation Operation, object::Archive *OldArchive) { switch (Operation) { @@ -825,6 +850,9 @@ static void performOperation(ArchiveOperation Operation, case ReplaceOrInsert: performWriteOperation(Operation, OldArchive); return; + case CreateSymTab: + createSymbolTable(OldArchive); + return; } llvm_unreachable("Unknown operation."); }