1
0
mirror of https://github.com/RPCS3/llvm-mirror.git synced 2024-11-22 10:42:39 +01:00

[llvm-libtool-darwin] Allow flattening archives

Add support for flattening archives while creating static libraries.
Hence, can now pass archives as input in addition to Mach-O binaries.
Furthermore, archives themselves must only conatain Mach-O binaries. As
per cctools' libtool's behavior, llvm-libtool-darwin does not flatten
archives recursively.

Reviewed by alexshap, smeenai, jhenderson

Differential Revision: https://reviews.llvm.org/D83520
This commit is contained in:
Sameer Arora 2020-07-02 14:32:55 -07:00
parent 3086c80b8b
commit 10d39a6183
4 changed files with 139 additions and 5 deletions

View File

@ -0,0 +1,95 @@
## This test checks that an archive is flattened correctly.
# RUN: yaml2obj %S/Inputs/input1.yaml -o %t-input1.o
# RUN: yaml2obj %S/Inputs/input2.yaml -o %t-input2.o
## Input a correct archive:
# RUN: rm -f %t.correct.ar
# RUN: llvm-ar cr %t.correct.ar %t-input1.o %t-input2.o
# RUN: llvm-libtool-darwin -static -o %t.lib %t.correct.ar
## Check that binaries are present:
# RUN: llvm-ar t %t.lib | \
# RUN: FileCheck %s --check-prefix=CHECK-NAMES --implicit-check-not={{.}} -DPREFIX=%basename_t.tmp
# CHECK-NAMES: [[PREFIX]]-input1.o
# CHECK-NAMES-NEXT: [[PREFIX]]-input2.o
## Check that symbols are present:
# RUN: llvm-nm --print-armap %t.lib | \
# RUN: FileCheck %s --check-prefix=CHECK-SYMBOLS -DPREFIX=%basename_t.tmp --match-full-lines
# CHECK-SYMBOLS: Archive map
# CHECK-SYMBOLS-NEXT: _symbol1 in [[PREFIX]]-input1.o
# CHECK-SYMBOLS-NEXT: _symbol2 in [[PREFIX]]-input2.o
# CHECK-SYMBOLS-EMPTY:
## Check that output archive is in Darwin format:
# RUN: llvm-objdump --macho --archive-headers %t.lib | \
# RUN: FileCheck %s --check-prefix=FORMAT -DPREFIX=%basename_t.tmp -DARCHIVE=%t.lib
# FORMAT: Archive : [[ARCHIVE]]
# FORMAT-NEXT: __.SYMDEF
# FORMAT-NEXT: [[PREFIX]]-input1.o
# FORMAT-NEXT: [[PREFIX]]-input2.o
# FORMAT-NOT: {{.}}
## Passing both archive and object file:
# RUN: llvm-libtool-darwin -static -o %t.lib %t-input2.o %t.correct.ar %t-input1.o
# RUN: llvm-ar t %t.lib | \
# RUN: FileCheck %s --check-prefix=BOTH-NAMES --implicit-check-not={{.}} -DPREFIX=%basename_t.tmp
# RUN: llvm-nm --print-armap %t.lib | \
# RUN: FileCheck %s --check-prefix=BOTH-SYMBOLS -DPREFIX=%basename_t.tmp --match-full-lines
# BOTH-NAMES: [[PREFIX]]-input2.o
# BOTH-NAMES-NEXT: [[PREFIX]]-input1.o
# BOTH-NAMES-NEXT: [[PREFIX]]-input2.o
# BOTH-NAMES-NEXT: [[PREFIX]]-input1.o
# BOTH-SYMBOLS: Archive map
# BOTH-SYMBOLS-NEXT: _symbol2 in [[PREFIX]]-input2.o
# BOTH-SYMBOLS-NEXT: _symbol1 in [[PREFIX]]-input1.o
# BOTH-SYMBOLS-NEXT: _symbol2 in [[PREFIX]]-input2.o
# BOTH-SYMBOLS-NEXT: _symbol1 in [[PREFIX]]-input1.o
# BOTH-SYMBOLS-EMPTY:
## Cannot read archive:
# RUN: echo '!<arch>' > %t-invalid-archive.lib
# RUN: echo 'invalid' >> %t-invalid-archive.lib
# RUN: not llvm-libtool-darwin -static -o %t.lib %t-invalid-archive.lib 2>&1 | \
# RUN: FileCheck %s --check-prefix=INVALID-ARCHIVE -DARCHIVE=%t-invalid-archive.lib
# INVALID-ARCHIVE: error: '[[ARCHIVE]]': truncated or malformed archive
## Archive member not an object file:
# RUN: rm -f %t.not-object.ar
# RUN: touch %t.txt
# RUN: llvm-ar cr %t.not-object.ar %t.txt
# RUN: not llvm-libtool-darwin -static -o %t.lib %t.not-object.ar 2>&1 | \
# RUN: FileCheck %s --check-prefix=NOT-OBJECT -DARCHIVE=%t.not-object.ar -DFILE=%basename_t.tmp.txt
## Do not recursively flatten archives:
# RUN: rm -f %t.inner
# RUN: rm -f %t.outer
# RUN: llvm-ar cr %t.inner %t-input1.o
# RUN: llvm-ar cr %t.outer %t.inner
# RUN: not llvm-libtool-darwin -static -o %t.lib %t.outer 2>&1 | \
# RUN: FileCheck %s --check-prefix=NOT-OBJECT -DARCHIVE=%t.outer -DFILE=%basename_t.tmp.inner
# NOT-OBJECT: error: '[[ARCHIVE]]': '[[FILE]]': The file was not recognized as a valid object file
## Archive member not a Mach-O object file:
# RUN: rm -f %t.not-macho.ar
# RUN: yaml2obj %s -o %t.elf
# RUN: llvm-ar cr %t.not-macho.ar %t.elf
# RUN: not llvm-libtool-darwin -static -o %t.lib %t.not-macho.ar 2>&1 | \
# RUN: FileCheck %s --check-prefix=NOT-MACHO -DARCHIVE=%t.not-macho.ar -DFILE=%basename_t.tmp.elf
# NOT-MACHO: error: '[[ARCHIVE]]': '[[FILE]]': format not supported
--- !ELF
FileHeader:
Class: ELFCLASS64
Data: ELFDATA2LSB
Type: ET_REL
Machine: EM_X86_64

View File

@ -1,4 +1,5 @@
set(LLVM_LINK_COMPONENTS
BinaryFormat
Object
Support
)

View File

@ -17,4 +17,4 @@
type = Tool
name = llvm-libtool-darwin
parent = Tools
required_libraries = Object Support
required_libraries = BinaryFormat Object Support

View File

@ -10,6 +10,7 @@
//
//===----------------------------------------------------------------------===//
#include "llvm/BinaryFormat/Magic.h"
#include "llvm/Object/ArchiveWriter.h"
#include "llvm/Object/MachO.h"
#include "llvm/Object/ObjectFile.h"
@ -58,11 +59,26 @@ static Error verifyMachOObject(const NewArchiveMember &Member) {
return Error::success();
}
static Error addMember(std::vector<NewArchiveMember> &Members,
StringRef FileName) {
static Error addChildMember(std::vector<NewArchiveMember> &Members,
const object::Archive::Child &M) {
Expected<NewArchiveMember> NMOrErr =
NewArchiveMember::getOldMember(M, /*Deterministic=*/true);
if (!NMOrErr)
return NMOrErr.takeError();
// Verify that Member is a Mach-O object file.
if (Error E = verifyMachOObject(*NMOrErr))
return E;
Members.push_back(std::move(*NMOrErr));
return Error::success();
}
static Error
addMember(std::vector<NewArchiveMember> &Members, StringRef FileName,
std::vector<std::unique_ptr<MemoryBuffer>> &ArchiveBuffers) {
Expected<NewArchiveMember> NMOrErr =
NewArchiveMember::getFile(FileName, /*Deterministic=*/true);
if (!NMOrErr)
return createFileError(FileName, NMOrErr.takeError());
@ -70,6 +86,27 @@ static Error addMember(std::vector<NewArchiveMember> &Members,
// name.
NMOrErr->MemberName = sys::path::filename(NMOrErr->MemberName);
// Flatten archives.
if (identify_magic(NMOrErr->Buf->getBuffer()) == file_magic::archive) {
Expected<std::unique_ptr<Archive>> LibOrErr =
object::Archive::create(NMOrErr->Buf->getMemBufferRef());
if (!LibOrErr)
return createFileError(FileName, LibOrErr.takeError());
object::Archive &Lib = **LibOrErr;
Error Err = Error::success();
for (const object::Archive::Child &Child : Lib.children(Err))
if (Error E = addChildMember(Members, Child))
return createFileError(FileName, std::move(E));
if (Err)
return createFileError(FileName, std::move(Err));
// Update vector ArchiveBuffers with the MemoryBuffers to transfer
// ownership.
ArchiveBuffers.push_back(std::move(NMOrErr->Buf));
return Error::success();
}
// Verify that Member is a Mach-O object file.
if (Error E = verifyMachOObject(*NMOrErr))
return E;
@ -80,8 +117,9 @@ static Error addMember(std::vector<NewArchiveMember> &Members,
static Error createStaticLibrary() {
std::vector<NewArchiveMember> NewMembers;
std::vector<std::unique_ptr<MemoryBuffer>> ArchiveBuffers;
for (StringRef Member : InputFiles)
if (Error E = addMember(NewMembers, Member))
if (Error E = addMember(NewMembers, Member, ArchiveBuffers))
return E;
if (Error E = writeArchive(OutputFile, NewMembers,