diff --git a/llvm/tools/llvm-libtool-darwin/llvm-libtool-darwin.cpp b/llvm/tools/llvm-libtool-darwin/llvm-libtool-darwin.cpp --- a/llvm/tools/llvm-libtool-darwin/llvm-libtool-darwin.cpp +++ b/llvm/tools/llvm-libtool-darwin/llvm-libtool-darwin.cpp @@ -25,15 +25,13 @@ #include "llvm/Support/WithColor.h" #include "llvm/TextAPI/Architecture.h" #include +#include using namespace llvm; using namespace llvm::object; static LLVMContext LLVMCtx; -typedef std::map> - MembersPerArchitectureMap; - cl::OptionCategory LibtoolCategory("llvm-libtool-darwin Options"); static cl::opt OutputFile("o", cl::desc("Specify output filename"), @@ -205,244 +203,277 @@ } } -// Check that a file's architecture [FileCPUType, FileCPUSubtype] -// matches the architecture specified under -arch_only flag. -static bool acceptFileArch(uint32_t FileCPUType, uint32_t FileCPUSubtype, - const Config &C) { - if (C.ArchCPUType != FileCPUType) - return false; - - switch (C.ArchCPUType) { - case MachO::CPU_TYPE_ARM: - case MachO::CPU_TYPE_ARM64_32: - case MachO::CPU_TYPE_X86_64: - return C.ArchCPUSubtype == FileCPUSubtype; - - case MachO::CPU_TYPE_ARM64: - if (C.ArchCPUSubtype == MachO::CPU_SUBTYPE_ARM64_ALL) - return FileCPUSubtype == MachO::CPU_SUBTYPE_ARM64_ALL || - FileCPUSubtype == MachO::CPU_SUBTYPE_ARM64_V8; - else +struct MembersPerArchitectureMap + : public std::map> { + typedef std::map> Parent; + + using Parent::Parent; + MembersPerArchitectureMap(const MembersPerArchitectureMap &other) = delete; + MembersPerArchitectureMap & + operator=(const MembersPerArchitectureMap &other) = delete; + MembersPerArchitectureMap(MembersPerArchitectureMap &&other) = default; +}; + +struct MembersData { + MembersPerArchitectureMap NewMembers; + std::vector> FileBuffers; + + static_assert( + !std::is_copy_assignable::value, + "NewMembers has a dependency on FileBuffers so it should not be able to " + "be copied on its own without FileBuffers."); + static_assert( + !std::is_copy_constructible::value, + "NewMembers has a dependency on FileBuffers so it should not be able to " + "be copied on its own without FileBuffers."); +}; + +class MembersBuilder { +public: + MembersBuilder(const Config &C) : C(C) {} + + Expected build() { + for (StringRef FileName : InputFiles) + if (Error E = addMember(FileName)) + return E; + + if (!ArchType.empty()) { + uint64_t ArchCPUID = getCPUID(C.ArchCPUType, C.ArchCPUSubtype); + if (Data.NewMembers.find(ArchCPUID) == Data.NewMembers.end()) + return createStringError(std::errc::invalid_argument, + "no library created (no object files in input " + "files matching -arch_only %s)", + ArchType.c_str()); + } + return std::move(Data); + } + +private: + // Check that a file's architecture [FileCPUType, FileCPUSubtype] + // matches the architecture specified under -arch_only flag. + bool acceptFileArch(uint32_t FileCPUType, uint32_t FileCPUSubtype) { + if (C.ArchCPUType != FileCPUType) + return false; + + switch (C.ArchCPUType) { + case MachO::CPU_TYPE_ARM: + case MachO::CPU_TYPE_ARM64_32: + case MachO::CPU_TYPE_X86_64: return C.ArchCPUSubtype == FileCPUSubtype; - default: - return true; + case MachO::CPU_TYPE_ARM64: + if (C.ArchCPUSubtype == MachO::CPU_SUBTYPE_ARM64_ALL) + return FileCPUSubtype == MachO::CPU_SUBTYPE_ARM64_ALL || + FileCPUSubtype == MachO::CPU_SUBTYPE_ARM64_V8; + else + return C.ArchCPUSubtype == FileCPUSubtype; + + default: + return true; + } } -} - -static Error verifyAndAddMachOObject(MembersPerArchitectureMap &Members, - NewArchiveMember Member, const Config &C) { - auto MBRef = Member.Buf->getMemBufferRef(); - Expected> ObjOrErr = - object::ObjectFile::createObjectFile(MBRef); - - // Throw error if not a valid object file. - if (!ObjOrErr) - return createFileError(Member.MemberName, ObjOrErr.takeError()); - - // Throw error if not in Mach-O format. - if (!isa(**ObjOrErr)) - return createStringError(std::errc::invalid_argument, - "'%s': format not supported", - Member.MemberName.data()); - - auto *O = dyn_cast(ObjOrErr->get()); - uint32_t FileCPUType, FileCPUSubtype; - std::tie(FileCPUType, FileCPUSubtype) = MachO::getCPUTypeFromArchitecture( - MachO::getArchitectureFromName(O->getArchTriple().getArchName())); - - // If -arch_only is specified then skip this file if it doesn't match - // the architecture specified. - if (!ArchType.empty() && !acceptFileArch(FileCPUType, FileCPUSubtype, C)) { + + Error verifyAndAddMachOObject(NewArchiveMember Member) { + auto MBRef = Member.Buf->getMemBufferRef(); + Expected> ObjOrErr = + object::ObjectFile::createObjectFile(MBRef); + + // Throw error if not a valid object file. + if (!ObjOrErr) + return createFileError(Member.MemberName, ObjOrErr.takeError()); + + // Throw error if not in Mach-O format. + if (!isa(**ObjOrErr)) + return createStringError(std::errc::invalid_argument, + "'%s': format not supported", + Member.MemberName.data()); + + auto *O = dyn_cast(ObjOrErr->get()); + uint32_t FileCPUType, FileCPUSubtype; + std::tie(FileCPUType, FileCPUSubtype) = MachO::getCPUTypeFromArchitecture( + MachO::getArchitectureFromName(O->getArchTriple().getArchName())); + + // If -arch_only is specified then skip this file if it doesn't match + // the architecture specified. + if (!ArchType.empty() && !acceptFileArch(FileCPUType, FileCPUSubtype)) { + return Error::success(); + } + + if (!NoWarningForNoSymbols && O->symbols().empty()) + WithColor::warning() << Member.MemberName + " has no symbols\n"; + + uint64_t FileCPUID = getCPUID(FileCPUType, FileCPUSubtype); + Data.NewMembers[FileCPUID].push_back(std::move(Member)); return Error::success(); } - if (!NoWarningForNoSymbols && O->symbols().empty()) - WithColor::warning() << Member.MemberName + " has no symbols\n"; + Error verifyAndAddIRObject(NewArchiveMember Member) { + auto MBRef = Member.Buf->getMemBufferRef(); + Expected> IROrErr = + object::IRObjectFile::create(MBRef, LLVMCtx); + + // Throw error if not a valid IR object file. + if (!IROrErr) + return createFileError(Member.MemberName, IROrErr.takeError()); + + Triple TT = Triple(IROrErr->get()->getTargetTriple()); - uint64_t FileCPUID = getCPUID(FileCPUType, FileCPUSubtype); - Members[FileCPUID].push_back(std::move(Member)); - return Error::success(); -} + Expected FileCPUTypeOrErr = MachO::getCPUType(TT); + if (!FileCPUTypeOrErr) + return FileCPUTypeOrErr.takeError(); -static Error verifyAndAddIRObject(MembersPerArchitectureMap &Members, - NewArchiveMember Member, const Config &C) { - auto MBRef = Member.Buf->getMemBufferRef(); - Expected> IROrErr = - object::IRObjectFile::create(MBRef, LLVMCtx); + Expected FileCPUSubTypeOrErr = MachO::getCPUSubType(TT); + if (!FileCPUSubTypeOrErr) + return FileCPUSubTypeOrErr.takeError(); - // Throw error if not a valid IR object file. - if (!IROrErr) - return createFileError(Member.MemberName, IROrErr.takeError()); + // If -arch_only is specified then skip this file if it doesn't match + // the architecture specified. + if (!ArchType.empty() && + !acceptFileArch(*FileCPUTypeOrErr, *FileCPUSubTypeOrErr)) { + return Error::success(); + } + + uint64_t FileCPUID = getCPUID(*FileCPUTypeOrErr, *FileCPUSubTypeOrErr); + Data.NewMembers[FileCPUID].push_back(std::move(Member)); + return Error::success(); + } - Triple TT = Triple(IROrErr->get()->getTargetTriple()); + Error addChildMember(const object::Archive::Child &M) { + Expected NewMemberOrErr = + NewArchiveMember::getOldMember(M, C.Deterministic); + if (!NewMemberOrErr) + return NewMemberOrErr.takeError(); + auto &NewMember = *NewMemberOrErr; - Expected FileCPUTypeOrErr = MachO::getCPUType(TT); - if (!FileCPUTypeOrErr) - return FileCPUTypeOrErr.takeError(); + file_magic Magic = identify_magic(NewMember.Buf->getBuffer()); + + if (Magic == file_magic::bitcode) + return verifyAndAddIRObject(std::move(NewMember)); + + return verifyAndAddMachOObject(std::move(NewMember)); + } - Expected FileCPUSubTypeOrErr = MachO::getCPUSubType(TT); - if (!FileCPUSubTypeOrErr) - return FileCPUSubTypeOrErr.takeError(); + Error processArchive(object::Archive &Lib, StringRef FileName) { + Error Err = Error::success(); + for (const object::Archive::Child &Child : Lib.children(Err)) + if (Error E = addChildMember(Child)) + return createFileError(FileName, std::move(E)); + if (Err) + return createFileError(FileName, std::move(Err)); - // If -arch_only is specified then skip this file if it doesn't match - // the architecture specified. - if (!ArchType.empty() && - !acceptFileArch(*FileCPUTypeOrErr, *FileCPUSubTypeOrErr, C)) { return Error::success(); } - uint64_t FileCPUID = getCPUID(*FileCPUTypeOrErr, *FileCPUSubTypeOrErr); - Members[FileCPUID].push_back(std::move(Member)); - return Error::success(); -} - -static Error addChildMember(MembersPerArchitectureMap &Members, - const object::Archive::Child &M, const Config &C) { - Expected NewMemberOrErr = - NewArchiveMember::getOldMember(M, C.Deterministic); - if (!NewMemberOrErr) - return NewMemberOrErr.takeError(); - auto &NewMember = *NewMemberOrErr; - - file_magic Magic = identify_magic(NewMember.Buf->getBuffer()); - - if (Magic == file_magic::bitcode) - return verifyAndAddIRObject(Members, std::move(NewMember), C); - - return verifyAndAddMachOObject(Members, std::move(NewMember), C); -} - -static Error processArchive(MembersPerArchitectureMap &Members, - object::Archive &Lib, StringRef FileName, - const Config &C) { - Error Err = Error::success(); - for (const object::Archive::Child &Child : Lib.children(Err)) - if (Error E = addChildMember(Members, Child, C)) - return createFileError(FileName, std::move(E)); - if (Err) - return createFileError(FileName, std::move(Err)); - - return Error::success(); -} - -static Error -addArchiveMembers(MembersPerArchitectureMap &Members, - std::vector> &ArchiveBuffers, - NewArchiveMember NewMember, StringRef FileName, - const Config &C) { - Expected> LibOrErr = - object::Archive::create(NewMember.Buf->getMemBufferRef()); - if (!LibOrErr) - return createFileError(FileName, LibOrErr.takeError()); - - if (Error E = processArchive(Members, **LibOrErr, FileName, C)) - return E; - - // Update vector ArchiveBuffers with the MemoryBuffers to transfer - // ownership. - ArchiveBuffers.push_back(std::move(NewMember.Buf)); - return Error::success(); -} - -static Error addUniversalMembers( - MembersPerArchitectureMap &Members, - std::vector> &UniversalBuffers, - NewArchiveMember NewMember, StringRef FileName, const Config &C) { - Expected> BinaryOrErr = - MachOUniversalBinary::create(NewMember.Buf->getMemBufferRef()); - if (!BinaryOrErr) - return createFileError(FileName, BinaryOrErr.takeError()); - - auto *UO = BinaryOrErr->get(); - for (const MachOUniversalBinary::ObjectForArch &O : UO->objects()) { - - Expected> MachOObjOrErr = - O.getAsObjectFile(); - if (MachOObjOrErr) { - NewArchiveMember NewMember = - NewArchiveMember(MachOObjOrErr->get()->getMemoryBufferRef()); - NewMember.MemberName = sys::path::filename(NewMember.MemberName); - - if (Error E = verifyAndAddMachOObject(Members, std::move(NewMember), C)) - return E; - continue; - } + Error addArchiveMembers(NewArchiveMember NewMember, StringRef FileName) { + Expected> LibOrErr = + object::Archive::create(NewMember.Buf->getMemBufferRef()); + if (!LibOrErr) + return createFileError(FileName, LibOrErr.takeError()); - Expected> IRObjectOrError = - O.getAsIRObject(LLVMCtx); - if (IRObjectOrError) { - // A universal file member can be a MachOObjectFile, an IRObject or an - // Archive. In case we can successfully cast the member as an IRObject, it - // is safe to throw away the error generated due to casting the object as - // a MachOObjectFile. - consumeError(MachOObjOrErr.takeError()); - - NewArchiveMember NewMember = - NewArchiveMember(IRObjectOrError->get()->getMemoryBufferRef()); - NewMember.MemberName = sys::path::filename(NewMember.MemberName); - - if (Error E = verifyAndAddIRObject(Members, std::move(NewMember), C)) - return E; - continue; - } + if (Error E = processArchive(**LibOrErr, FileName)) + return E; - Expected> ArchiveOrError = O.getAsArchive(); - if (ArchiveOrError) { - // A universal file member can be a MachOObjectFile, an IRObject or an - // Archive. In case we can successfully cast the member as an Archive, it - // is safe to throw away the error generated due to casting the object as - // a MachOObjectFile. - consumeError(MachOObjOrErr.takeError()); - consumeError(IRObjectOrError.takeError()); - - if (Error E = processArchive(Members, **ArchiveOrError, FileName, C)) - return E; - continue; + // Update vector FileBuffers with the MemoryBuffers to transfer + // ownership. + Data.FileBuffers.push_back(std::move(NewMember.Buf)); + return Error::success(); + } + + Error addUniversalMembers(NewArchiveMember NewMember, StringRef FileName) { + Expected> BinaryOrErr = + MachOUniversalBinary::create(NewMember.Buf->getMemBufferRef()); + if (!BinaryOrErr) + return createFileError(FileName, BinaryOrErr.takeError()); + + auto *UO = BinaryOrErr->get(); + for (const MachOUniversalBinary::ObjectForArch &O : UO->objects()) { + + Expected> MachOObjOrErr = + O.getAsObjectFile(); + if (MachOObjOrErr) { + NewArchiveMember NewMember = + NewArchiveMember(MachOObjOrErr->get()->getMemoryBufferRef()); + NewMember.MemberName = sys::path::filename(NewMember.MemberName); + + if (Error E = verifyAndAddMachOObject(std::move(NewMember))) + return E; + continue; + } + + Expected> IRObjectOrError = + O.getAsIRObject(LLVMCtx); + if (IRObjectOrError) { + // A universal file member can be a MachOObjectFile, an IRObject or an + // Archive. In case we can successfully cast the member as an IRObject, + // it is safe to throw away the error generated due to casting the + // object as a MachOObjectFile. + consumeError(MachOObjOrErr.takeError()); + + NewArchiveMember NewMember = + NewArchiveMember(IRObjectOrError->get()->getMemoryBufferRef()); + NewMember.MemberName = sys::path::filename(NewMember.MemberName); + + if (Error E = verifyAndAddIRObject(std::move(NewMember))) + return E; + continue; + } + + Expected> ArchiveOrError = O.getAsArchive(); + if (ArchiveOrError) { + // A universal file member can be a MachOObjectFile, an IRObject or an + // Archive. In case we can successfully cast the member as an Archive, + // it is safe to throw away the error generated due to casting the + // object as a MachOObjectFile. + consumeError(MachOObjOrErr.takeError()); + consumeError(IRObjectOrError.takeError()); + + if (Error E = processArchive(**ArchiveOrError, FileName)) + return E; + continue; + } + + Error CombinedError = joinErrors( + ArchiveOrError.takeError(), + joinErrors(IRObjectOrError.takeError(), MachOObjOrErr.takeError())); + return createFileError(FileName, std::move(CombinedError)); } - Error CombinedError = joinErrors( - ArchiveOrError.takeError(), - joinErrors(IRObjectOrError.takeError(), MachOObjOrErr.takeError())); - return createFileError(FileName, std::move(CombinedError)); + // Update vector FileBuffers with the MemoryBuffers to transfer + // ownership. + Data.FileBuffers.push_back(std::move(NewMember.Buf)); + return Error::success(); + } + + Error addMember(StringRef FileName) { + Expected NewMemberOrErr = + NewArchiveMember::getFile(FileName, C.Deterministic); + if (!NewMemberOrErr) + return createFileError(FileName, NewMemberOrErr.takeError()); + auto &NewMember = *NewMemberOrErr; + + // For regular archives, use the basename of the object path for the member + // name. + NewMember.MemberName = sys::path::filename(NewMember.MemberName); + file_magic Magic = identify_magic(NewMember.Buf->getBuffer()); + + // Flatten archives. + if (Magic == file_magic::archive) + return addArchiveMembers(std::move(NewMember), FileName); + + // Flatten universal files. + if (Magic == file_magic::macho_universal_binary) + return addUniversalMembers(std::move(NewMember), FileName); + + // Bitcode files. + if (Magic == file_magic::bitcode) + return verifyAndAddIRObject(std::move(NewMember)); + + return verifyAndAddMachOObject(std::move(NewMember)); } - // Update vector UniversalBuffers with the MemoryBuffers to transfer - // ownership. - UniversalBuffers.push_back(std::move(NewMember.Buf)); - return Error::success(); -} - -static Error addMember(MembersPerArchitectureMap &Members, - std::vector> &FileBuffers, - StringRef FileName, const Config &C) { - Expected NewMemberOrErr = - NewArchiveMember::getFile(FileName, C.Deterministic); - if (!NewMemberOrErr) - return createFileError(FileName, NewMemberOrErr.takeError()); - auto &NewMember = *NewMemberOrErr; - - // For regular archives, use the basename of the object path for the member - // name. - NewMember.MemberName = sys::path::filename(NewMember.MemberName); - file_magic Magic = identify_magic(NewMember.Buf->getBuffer()); - - // Flatten archives. - if (Magic == file_magic::archive) - return addArchiveMembers(Members, FileBuffers, std::move(NewMember), - FileName, C); - - // Flatten universal files. - if (Magic == file_magic::macho_universal_binary) - return addUniversalMembers(Members, FileBuffers, std::move(NewMember), - FileName, C); - - // Bitcode files. - if (Magic == file_magic::bitcode) - return verifyAndAddIRObject(Members, std::move(NewMember), C); - - return verifyAndAddMachOObject(Members, std::move(NewMember), C); -} + MembersData Data; + const Config &C; +}; static Expected> buildSlices(ArrayRef> OutputBinaries) { @@ -459,21 +490,14 @@ } static Error createStaticLibrary(const Config &C) { - MembersPerArchitectureMap NewMembers; - std::vector> FileBuffers; - for (StringRef FileName : InputFiles) - if (Error E = addMember(NewMembers, FileBuffers, FileName, C)) - return E; - - if (!ArchType.empty()) { - uint64_t ArchCPUID = getCPUID(C.ArchCPUType, C.ArchCPUSubtype); - if (NewMembers.find(ArchCPUID) == NewMembers.end()) - return createStringError(std::errc::invalid_argument, - "no library created (no object files in input " - "files matching -arch_only %s)", - ArchType.c_str()); + MembersBuilder Builder(C); + auto DataOrError = Builder.build(); + if (auto Error = DataOrError.takeError()) { + return Error; } + const auto &NewMembers = DataOrError->NewMembers; + if (NewMembers.size() == 1) { return writeArchive(OutputFile, NewMembers.begin()->second, /*WriteSymtab=*/true,