|  | //===- InterfaceFile.cpp --------------------------------------------------===// | 
|  | // | 
|  | // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. | 
|  | // See https://llvm.org/LICENSE.txt for license information. | 
|  | // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception | 
|  | // | 
|  | //===----------------------------------------------------------------------===// | 
|  | // | 
|  | // Implements the Interface File. | 
|  | // | 
|  | //===----------------------------------------------------------------------===// | 
|  |  | 
|  | #include "llvm/TextAPI/InterfaceFile.h" | 
|  | #include <iomanip> | 
|  | #include <sstream> | 
|  |  | 
|  | using namespace llvm; | 
|  | using namespace llvm::MachO; | 
|  |  | 
|  | namespace { | 
|  | template <typename C> | 
|  | typename C::iterator addEntry(C &Container, StringRef InstallName) { | 
|  | auto I = partition_point(Container, [=](const InterfaceFileRef &O) { | 
|  | return O.getInstallName() < InstallName; | 
|  | }); | 
|  | if (I != Container.end() && I->getInstallName() == InstallName) | 
|  | return I; | 
|  |  | 
|  | return Container.emplace(I, InstallName); | 
|  | } | 
|  |  | 
|  | template <typename C> | 
|  | typename C::iterator addEntry(C &Container, const Target &Target_) { | 
|  | auto Iter = | 
|  | lower_bound(Container, Target_, [](const Target &LHS, const Target &RHS) { | 
|  | return LHS < RHS; | 
|  | }); | 
|  | if ((Iter != std::end(Container)) && !(Target_ < *Iter)) | 
|  | return Iter; | 
|  |  | 
|  | return Container.insert(Iter, Target_); | 
|  | } | 
|  | } // end namespace | 
|  |  | 
|  | void InterfaceFileRef::addTarget(const Target &Target) { | 
|  | addEntry(Targets, Target); | 
|  | } | 
|  |  | 
|  | void InterfaceFile::addAllowableClient(StringRef InstallName, | 
|  | const Target &Target) { | 
|  | auto Client = addEntry(AllowableClients, InstallName); | 
|  | Client->addTarget(Target); | 
|  | } | 
|  |  | 
|  | void InterfaceFile::addReexportedLibrary(StringRef InstallName, | 
|  | const Target &Target) { | 
|  | auto Lib = addEntry(ReexportedLibraries, InstallName); | 
|  | Lib->addTarget(Target); | 
|  | } | 
|  |  | 
|  | void InterfaceFile::addParentUmbrella(const Target &Target_, StringRef Parent) { | 
|  | auto Iter = lower_bound(ParentUmbrellas, Target_, | 
|  | [](const std::pair<Target, std::string> &LHS, | 
|  | Target RHS) { return LHS.first < RHS; }); | 
|  |  | 
|  | if ((Iter != ParentUmbrellas.end()) && !(Target_ < Iter->first)) { | 
|  | Iter->second = std::string(Parent); | 
|  | return; | 
|  | } | 
|  |  | 
|  | ParentUmbrellas.emplace(Iter, Target_, std::string(Parent)); | 
|  | } | 
|  |  | 
|  | void InterfaceFile::addUUID(const Target &Target_, StringRef UUID) { | 
|  | auto Iter = lower_bound(UUIDs, Target_, | 
|  | [](const std::pair<Target, std::string> &LHS, | 
|  | Target RHS) { return LHS.first < RHS; }); | 
|  |  | 
|  | if ((Iter != UUIDs.end()) && !(Target_ < Iter->first)) { | 
|  | Iter->second = std::string(UUID); | 
|  | return; | 
|  | } | 
|  |  | 
|  | UUIDs.emplace(Iter, Target_, std::string(UUID)); | 
|  | } | 
|  |  | 
|  | void InterfaceFile::addUUID(const Target &Target, uint8_t UUID[16]) { | 
|  | std::stringstream Stream; | 
|  | for (unsigned i = 0; i < 16; ++i) { | 
|  | if (i == 4 || i == 6 || i == 8 || i == 10) | 
|  | Stream << '-'; | 
|  | Stream << std::setfill('0') << std::setw(2) << std::uppercase << std::hex | 
|  | << static_cast<int>(UUID[i]); | 
|  | } | 
|  | addUUID(Target, Stream.str()); | 
|  | } | 
|  |  | 
|  | void InterfaceFile::addTarget(const Target &Target) { | 
|  | addEntry(Targets, Target); | 
|  | } | 
|  |  | 
|  | InterfaceFile::const_filtered_target_range | 
|  | InterfaceFile::targets(ArchitectureSet Archs) const { | 
|  | std::function<bool(const Target &)> fn = [Archs](const Target &Target_) { | 
|  | return Archs.has(Target_.Arch); | 
|  | }; | 
|  | return make_filter_range(Targets, fn); | 
|  | } | 
|  |  | 
|  | void InterfaceFile::addSymbol(SymbolKind Kind, StringRef Name, | 
|  | const TargetList &Targets, SymbolFlags Flags) { | 
|  | Name = copyString(Name); | 
|  | auto result = Symbols.try_emplace(SymbolsMapKey{Kind, Name}, nullptr); | 
|  | if (result.second) | 
|  | result.first->second = new (Allocator) Symbol{Kind, Name, Targets, Flags}; | 
|  | else | 
|  | for (const auto &Target : Targets) | 
|  | result.first->second->addTarget(Target); | 
|  | } | 
|  |  | 
|  | void InterfaceFile::addDocument(std::shared_ptr<InterfaceFile> &&Document) { | 
|  | auto Pos = llvm::lower_bound(Documents, Document, | 
|  | [](const std::shared_ptr<InterfaceFile> &LHS, | 
|  | const std::shared_ptr<InterfaceFile> &RHS) { | 
|  | return LHS->InstallName < RHS->InstallName; | 
|  | }); | 
|  | Document->Parent = this; | 
|  | Documents.insert(Pos, Document); | 
|  | } | 
|  |  | 
|  | bool InterfaceFile::operator==(const InterfaceFile &O) const { | 
|  | if (Targets != O.Targets) | 
|  | return false; | 
|  | if (InstallName != O.InstallName) | 
|  | return false; | 
|  | if ((CurrentVersion != O.CurrentVersion) || | 
|  | (CompatibilityVersion != O.CompatibilityVersion)) | 
|  | return false; | 
|  | if (SwiftABIVersion != O.SwiftABIVersion) | 
|  | return false; | 
|  | if (IsTwoLevelNamespace != O.IsTwoLevelNamespace) | 
|  | return false; | 
|  | if (IsAppExtensionSafe != O.IsAppExtensionSafe) | 
|  | return false; | 
|  | if (IsInstallAPI != O.IsInstallAPI) | 
|  | return false; | 
|  | if (ParentUmbrellas != O.ParentUmbrellas) | 
|  | return false; | 
|  | if (AllowableClients != O.AllowableClients) | 
|  | return false; | 
|  | if (ReexportedLibraries != O.ReexportedLibraries) | 
|  | return false; | 
|  | if (Symbols != O.Symbols) | 
|  | return false; | 
|  | if (!std::equal(Documents.begin(), Documents.end(), O.Documents.begin(), | 
|  | O.Documents.end(), | 
|  | [](const std::shared_ptr<InterfaceFile> LHS, | 
|  | const std::shared_ptr<InterfaceFile> RHS) { | 
|  | return *LHS == *RHS; | 
|  | })) | 
|  | return false; | 
|  | return true; | 
|  | } |