13a02247dSChandler Carruth //===--- Module.cpp - Describe a module -----------------------------------===// 2de3ef502SDouglas Gregor // 3de3ef502SDouglas Gregor // The LLVM Compiler Infrastructure 4de3ef502SDouglas Gregor // 5de3ef502SDouglas Gregor // This file is distributed under the University of Illinois Open Source 6de3ef502SDouglas Gregor // License. See LICENSE.TXT for details. 7de3ef502SDouglas Gregor // 8de3ef502SDouglas Gregor //===----------------------------------------------------------------------===// 9de3ef502SDouglas Gregor // 10de3ef502SDouglas Gregor // This file defines the Module class, which describes a module in the source 11de3ef502SDouglas Gregor // code. 12de3ef502SDouglas Gregor // 13de3ef502SDouglas Gregor //===----------------------------------------------------------------------===// 14a3feee2aSRichard Smith 15de3ef502SDouglas Gregor #include "clang/Basic/Module.h" 16de3ef502SDouglas Gregor #include "clang/Basic/FileManager.h" 171fb5c3a6SDouglas Gregor #include "clang/Basic/LangOptions.h" 180070c0bfSDouglas Gregor #include "clang/Basic/TargetInfo.h" 193c5305c1SArgyrios Kyrtzidis #include "llvm/ADT/ArrayRef.h" 201fb5c3a6SDouglas Gregor #include "llvm/ADT/SmallVector.h" 211fb5c3a6SDouglas Gregor #include "llvm/ADT/StringSwitch.h" 223a02247dSChandler Carruth #include "llvm/Support/ErrorHandling.h" 233a02247dSChandler Carruth #include "llvm/Support/raw_ostream.h" 24a3feee2aSRichard Smith 25de3ef502SDouglas Gregor using namespace clang; 26de3ef502SDouglas Gregor 27eb90e830SDouglas Gregor Module::Module(StringRef Name, SourceLocation DefinitionLoc, Module *Parent, 28eb90e830SDouglas Gregor bool IsFramework, bool IsExplicit) 29eb90e830SDouglas Gregor : Name(Name), DefinitionLoc(DefinitionLoc), Parent(Parent), 3043af5132SArgyrios Kyrtzidis Umbrella(), ASTFile(0), IsAvailable(true), IsFromModuleFile(false), 31a686e1b0SDouglas Gregor IsFramework(IsFramework), IsExplicit(IsExplicit), IsSystem(false), 32a686e1b0SDouglas Gregor InferSubmodules(false), InferExplicitSubmodules(false), 338d932427SDouglas Gregor InferExportWildcard(false), ConfigMacrosExhaustive(false), 348d932427SDouglas Gregor NameVisibility(Hidden) 35eb90e830SDouglas Gregor { 36eb90e830SDouglas Gregor if (Parent) { 37eb90e830SDouglas Gregor if (!Parent->isAvailable()) 38eb90e830SDouglas Gregor IsAvailable = false; 393ec6663bSDouglas Gregor if (Parent->IsSystem) 403ec6663bSDouglas Gregor IsSystem = true; 41eb90e830SDouglas Gregor 42eb90e830SDouglas Gregor Parent->SubModuleIndex[Name] = Parent->SubModules.size(); 43eb90e830SDouglas Gregor Parent->SubModules.push_back(this); 44eb90e830SDouglas Gregor } 45eb90e830SDouglas Gregor } 46eb90e830SDouglas Gregor 47de3ef502SDouglas Gregor Module::~Module() { 48eb90e830SDouglas Gregor for (submodule_iterator I = submodule_begin(), IEnd = submodule_end(); 49de3ef502SDouglas Gregor I != IEnd; ++I) { 50eb90e830SDouglas Gregor delete *I; 51de3ef502SDouglas Gregor } 52de3ef502SDouglas Gregor } 53de3ef502SDouglas Gregor 541fb5c3a6SDouglas Gregor /// \brief Determine whether a translation unit built using the current 551fb5c3a6SDouglas Gregor /// language options has the given feature. 5689929282SDouglas Gregor static bool hasFeature(StringRef Feature, const LangOptions &LangOpts, 5789929282SDouglas Gregor const TargetInfo &Target) { 581fb5c3a6SDouglas Gregor return llvm::StringSwitch<bool>(Feature) 590070c0bfSDouglas Gregor .Case("altivec", LangOpts.AltiVec) 601fb5c3a6SDouglas Gregor .Case("blocks", LangOpts.Blocks) 611fb5c3a6SDouglas Gregor .Case("cplusplus", LangOpts.CPlusPlus) 622bf7fdb7SRichard Smith .Case("cplusplus11", LangOpts.CPlusPlus11) 631fb5c3a6SDouglas Gregor .Case("objc", LangOpts.ObjC1) 641fb5c3a6SDouglas Gregor .Case("objc_arc", LangOpts.ObjCAutoRefCount) 650070c0bfSDouglas Gregor .Case("opencl", LangOpts.OpenCL) 660070c0bfSDouglas Gregor .Case("tls", Target.isTLSSupported()) 670070c0bfSDouglas Gregor .Default(Target.hasFeature(Feature)); 681fb5c3a6SDouglas Gregor } 691fb5c3a6SDouglas Gregor 701fb5c3a6SDouglas Gregor bool 7189929282SDouglas Gregor Module::isAvailable(const LangOptions &LangOpts, const TargetInfo &Target, 72a3feee2aSRichard Smith Requirement &Req) const { 731fb5c3a6SDouglas Gregor if (IsAvailable) 741fb5c3a6SDouglas Gregor return true; 751fb5c3a6SDouglas Gregor 761fb5c3a6SDouglas Gregor for (const Module *Current = this; Current; Current = Current->Parent) { 77a3feee2aSRichard Smith for (unsigned I = 0, N = Current->Requirements.size(); I != N; ++I) { 78a3feee2aSRichard Smith if (hasFeature(Current->Requirements[I].first, LangOpts, Target) != 79a3feee2aSRichard Smith Current->Requirements[I].second) { 80a3feee2aSRichard Smith Req = Current->Requirements[I]; 811fb5c3a6SDouglas Gregor return false; 821fb5c3a6SDouglas Gregor } 831fb5c3a6SDouglas Gregor } 841fb5c3a6SDouglas Gregor } 851fb5c3a6SDouglas Gregor 861fb5c3a6SDouglas Gregor llvm_unreachable("could not find a reason why module is unavailable"); 871fb5c3a6SDouglas Gregor } 881fb5c3a6SDouglas Gregor 89f5eedd05SDouglas Gregor bool Module::isSubModuleOf(Module *Other) const { 90f5eedd05SDouglas Gregor const Module *This = this; 91f5eedd05SDouglas Gregor do { 92f5eedd05SDouglas Gregor if (This == Other) 93f5eedd05SDouglas Gregor return true; 94f5eedd05SDouglas Gregor 95f5eedd05SDouglas Gregor This = This->Parent; 96f5eedd05SDouglas Gregor } while (This); 97f5eedd05SDouglas Gregor 98f5eedd05SDouglas Gregor return false; 99f5eedd05SDouglas Gregor } 100f5eedd05SDouglas Gregor 10173441091SDouglas Gregor const Module *Module::getTopLevelModule() const { 10273441091SDouglas Gregor const Module *Result = this; 10373441091SDouglas Gregor while (Result->Parent) 10473441091SDouglas Gregor Result = Result->Parent; 10573441091SDouglas Gregor 10673441091SDouglas Gregor return Result; 10773441091SDouglas Gregor } 10873441091SDouglas Gregor 109de3ef502SDouglas Gregor std::string Module::getFullModuleName() const { 110f857950dSDmitri Gribenko SmallVector<StringRef, 2> Names; 111de3ef502SDouglas Gregor 112de3ef502SDouglas Gregor // Build up the set of module names (from innermost to outermost). 113de3ef502SDouglas Gregor for (const Module *M = this; M; M = M->Parent) 114de3ef502SDouglas Gregor Names.push_back(M->Name); 115de3ef502SDouglas Gregor 116de3ef502SDouglas Gregor std::string Result; 11761ac906bSCraig Topper for (SmallVectorImpl<StringRef>::reverse_iterator I = Names.rbegin(), 118de3ef502SDouglas Gregor IEnd = Names.rend(); 119de3ef502SDouglas Gregor I != IEnd; ++I) { 120de3ef502SDouglas Gregor if (!Result.empty()) 121de3ef502SDouglas Gregor Result += '.'; 122de3ef502SDouglas Gregor 123de3ef502SDouglas Gregor Result += *I; 124de3ef502SDouglas Gregor } 125de3ef502SDouglas Gregor 126de3ef502SDouglas Gregor return Result; 127de3ef502SDouglas Gregor } 128de3ef502SDouglas Gregor 12973141fa9SDouglas Gregor const DirectoryEntry *Module::getUmbrellaDir() const { 13073141fa9SDouglas Gregor if (const FileEntry *Header = getUmbrellaHeader()) 13173141fa9SDouglas Gregor return Header->getDir(); 13273141fa9SDouglas Gregor 13373141fa9SDouglas Gregor return Umbrella.dyn_cast<const DirectoryEntry *>(); 13473141fa9SDouglas Gregor } 13573141fa9SDouglas Gregor 1363c5305c1SArgyrios Kyrtzidis ArrayRef<const FileEntry *> Module::getTopHeaders(FileManager &FileMgr) { 1373c5305c1SArgyrios Kyrtzidis if (!TopHeaderNames.empty()) { 1383c5305c1SArgyrios Kyrtzidis for (std::vector<std::string>::iterator 1393c5305c1SArgyrios Kyrtzidis I = TopHeaderNames.begin(), E = TopHeaderNames.end(); I != E; ++I) { 1403c5305c1SArgyrios Kyrtzidis if (const FileEntry *FE = FileMgr.getFile(*I)) 1413c5305c1SArgyrios Kyrtzidis TopHeaders.insert(FE); 1423c5305c1SArgyrios Kyrtzidis } 1433c5305c1SArgyrios Kyrtzidis TopHeaderNames.clear(); 1443c5305c1SArgyrios Kyrtzidis } 1453c5305c1SArgyrios Kyrtzidis 1463c5305c1SArgyrios Kyrtzidis return llvm::makeArrayRef(TopHeaders.begin(), TopHeaders.end()); 1473c5305c1SArgyrios Kyrtzidis } 1483c5305c1SArgyrios Kyrtzidis 149a3feee2aSRichard Smith void Module::addRequirement(StringRef Feature, bool RequiredState, 150a3feee2aSRichard Smith const LangOptions &LangOpts, 15189929282SDouglas Gregor const TargetInfo &Target) { 152a3feee2aSRichard Smith Requirements.push_back(Requirement(Feature, RequiredState)); 1531fb5c3a6SDouglas Gregor 1541fb5c3a6SDouglas Gregor // If this feature is currently available, we're done. 155a3feee2aSRichard Smith if (hasFeature(Feature, LangOpts, Target) == RequiredState) 1561fb5c3a6SDouglas Gregor return; 1571fb5c3a6SDouglas Gregor 1581fb5c3a6SDouglas Gregor if (!IsAvailable) 1591fb5c3a6SDouglas Gregor return; 1601fb5c3a6SDouglas Gregor 161f857950dSDmitri Gribenko SmallVector<Module *, 2> Stack; 1621fb5c3a6SDouglas Gregor Stack.push_back(this); 1631fb5c3a6SDouglas Gregor while (!Stack.empty()) { 1641fb5c3a6SDouglas Gregor Module *Current = Stack.back(); 1651fb5c3a6SDouglas Gregor Stack.pop_back(); 1661fb5c3a6SDouglas Gregor 1671fb5c3a6SDouglas Gregor if (!Current->IsAvailable) 1681fb5c3a6SDouglas Gregor continue; 1691fb5c3a6SDouglas Gregor 1701fb5c3a6SDouglas Gregor Current->IsAvailable = false; 171eb90e830SDouglas Gregor for (submodule_iterator Sub = Current->submodule_begin(), 172eb90e830SDouglas Gregor SubEnd = Current->submodule_end(); 1731fb5c3a6SDouglas Gregor Sub != SubEnd; ++Sub) { 174eb90e830SDouglas Gregor if ((*Sub)->IsAvailable) 175eb90e830SDouglas Gregor Stack.push_back(*Sub); 1761fb5c3a6SDouglas Gregor } 1771fb5c3a6SDouglas Gregor } 1781fb5c3a6SDouglas Gregor } 1791fb5c3a6SDouglas Gregor 180eb90e830SDouglas Gregor Module *Module::findSubmodule(StringRef Name) const { 181eb90e830SDouglas Gregor llvm::StringMap<unsigned>::const_iterator Pos = SubModuleIndex.find(Name); 182eb90e830SDouglas Gregor if (Pos == SubModuleIndex.end()) 183eb90e830SDouglas Gregor return 0; 184eb90e830SDouglas Gregor 185eb90e830SDouglas Gregor return SubModules[Pos->getValue()]; 186eb90e830SDouglas Gregor } 187eb90e830SDouglas Gregor 188f857950dSDmitri Gribenko static void printModuleId(raw_ostream &OS, const ModuleId &Id) { 18924bb923aSDouglas Gregor for (unsigned I = 0, N = Id.size(); I != N; ++I) { 19024bb923aSDouglas Gregor if (I) 19124bb923aSDouglas Gregor OS << "."; 19224bb923aSDouglas Gregor OS << Id[I].first; 19324bb923aSDouglas Gregor } 19424bb923aSDouglas Gregor } 19524bb923aSDouglas Gregor 1968739f7b7SArgyrios Kyrtzidis void Module::getExportedModules(SmallVectorImpl<Module *> &Exported) const { 197*e9bcf5b7SDmitri Gribenko // All non-explicit submodules are exported. 198*e9bcf5b7SDmitri Gribenko for (std::vector<Module *>::const_iterator I = SubModules.begin(), 199*e9bcf5b7SDmitri Gribenko E = SubModules.end(); 200*e9bcf5b7SDmitri Gribenko I != E; ++I) { 201*e9bcf5b7SDmitri Gribenko Module *Mod = *I; 202*e9bcf5b7SDmitri Gribenko if (!Mod->IsExplicit) 203*e9bcf5b7SDmitri Gribenko Exported.push_back(Mod); 204*e9bcf5b7SDmitri Gribenko } 205*e9bcf5b7SDmitri Gribenko 206*e9bcf5b7SDmitri Gribenko // Find re-exported modules by filtering the list of imported modules. 2078739f7b7SArgyrios Kyrtzidis bool AnyWildcard = false; 2088739f7b7SArgyrios Kyrtzidis bool UnrestrictedWildcard = false; 2098739f7b7SArgyrios Kyrtzidis SmallVector<Module *, 4> WildcardRestrictions; 2108739f7b7SArgyrios Kyrtzidis for (unsigned I = 0, N = Exports.size(); I != N; ++I) { 2118739f7b7SArgyrios Kyrtzidis Module *Mod = Exports[I].getPointer(); 2128739f7b7SArgyrios Kyrtzidis if (!Exports[I].getInt()) { 2138739f7b7SArgyrios Kyrtzidis // Export a named module directly; no wildcards involved. 2148739f7b7SArgyrios Kyrtzidis Exported.push_back(Mod); 2158739f7b7SArgyrios Kyrtzidis 2168739f7b7SArgyrios Kyrtzidis continue; 2178739f7b7SArgyrios Kyrtzidis } 2188739f7b7SArgyrios Kyrtzidis 2198739f7b7SArgyrios Kyrtzidis // Wildcard export: export all of the imported modules that match 2208739f7b7SArgyrios Kyrtzidis // the given pattern. 2218739f7b7SArgyrios Kyrtzidis AnyWildcard = true; 2228739f7b7SArgyrios Kyrtzidis if (UnrestrictedWildcard) 2238739f7b7SArgyrios Kyrtzidis continue; 2248739f7b7SArgyrios Kyrtzidis 2258739f7b7SArgyrios Kyrtzidis if (Module *Restriction = Exports[I].getPointer()) 2268739f7b7SArgyrios Kyrtzidis WildcardRestrictions.push_back(Restriction); 2278739f7b7SArgyrios Kyrtzidis else { 2288739f7b7SArgyrios Kyrtzidis WildcardRestrictions.clear(); 2298739f7b7SArgyrios Kyrtzidis UnrestrictedWildcard = true; 2308739f7b7SArgyrios Kyrtzidis } 2318739f7b7SArgyrios Kyrtzidis } 2328739f7b7SArgyrios Kyrtzidis 2338739f7b7SArgyrios Kyrtzidis // If there were any wildcards, push any imported modules that were 2348739f7b7SArgyrios Kyrtzidis // re-exported by the wildcard restriction. 2358739f7b7SArgyrios Kyrtzidis if (!AnyWildcard) 2368739f7b7SArgyrios Kyrtzidis return; 2378739f7b7SArgyrios Kyrtzidis 2388739f7b7SArgyrios Kyrtzidis for (unsigned I = 0, N = Imports.size(); I != N; ++I) { 2398739f7b7SArgyrios Kyrtzidis Module *Mod = Imports[I]; 2408739f7b7SArgyrios Kyrtzidis bool Acceptable = UnrestrictedWildcard; 2418739f7b7SArgyrios Kyrtzidis if (!Acceptable) { 2428739f7b7SArgyrios Kyrtzidis // Check whether this module meets one of the restrictions. 2438739f7b7SArgyrios Kyrtzidis for (unsigned R = 0, NR = WildcardRestrictions.size(); R != NR; ++R) { 2448739f7b7SArgyrios Kyrtzidis Module *Restriction = WildcardRestrictions[R]; 2458739f7b7SArgyrios Kyrtzidis if (Mod == Restriction || Mod->isSubModuleOf(Restriction)) { 2468739f7b7SArgyrios Kyrtzidis Acceptable = true; 2478739f7b7SArgyrios Kyrtzidis break; 2488739f7b7SArgyrios Kyrtzidis } 2498739f7b7SArgyrios Kyrtzidis } 2508739f7b7SArgyrios Kyrtzidis } 2518739f7b7SArgyrios Kyrtzidis 2528739f7b7SArgyrios Kyrtzidis if (!Acceptable) 2538739f7b7SArgyrios Kyrtzidis continue; 2548739f7b7SArgyrios Kyrtzidis 2558739f7b7SArgyrios Kyrtzidis Exported.push_back(Mod); 2568739f7b7SArgyrios Kyrtzidis } 2578739f7b7SArgyrios Kyrtzidis } 2588739f7b7SArgyrios Kyrtzidis 2590e5d7b8cSRichard Smith void Module::buildVisibleModulesCache() const { 2600e5d7b8cSRichard Smith assert(VisibleModulesCache.empty() && "cache does not need building"); 2610e5d7b8cSRichard Smith 2620e5d7b8cSRichard Smith // This module is visible to itself. 2630e5d7b8cSRichard Smith VisibleModulesCache.insert(this); 2640e5d7b8cSRichard Smith 2650e5d7b8cSRichard Smith // Every imported module is visible. 266dde17e74SRichard Smith SmallVector<Module *, 16> Stack(Imports.begin(), Imports.end()); 267dc360d57SDmitri Gribenko while (!Stack.empty()) { 268dc360d57SDmitri Gribenko Module *CurrModule = Stack.pop_back_val(); 269dc360d57SDmitri Gribenko 270dde17e74SRichard Smith // Every module transitively exported by an imported module is visible. 271dde17e74SRichard Smith if (VisibleModulesCache.insert(CurrModule).second) 272dde17e74SRichard Smith CurrModule->getExportedModules(Stack); 2730e5d7b8cSRichard Smith } 2740e5d7b8cSRichard Smith } 2750e5d7b8cSRichard Smith 276f857950dSDmitri Gribenko void Module::print(raw_ostream &OS, unsigned Indent) const { 277de3ef502SDouglas Gregor OS.indent(Indent); 278de3ef502SDouglas Gregor if (IsFramework) 279de3ef502SDouglas Gregor OS << "framework "; 280de3ef502SDouglas Gregor if (IsExplicit) 281de3ef502SDouglas Gregor OS << "explicit "; 282a686e1b0SDouglas Gregor OS << "module " << Name; 283a686e1b0SDouglas Gregor 284a686e1b0SDouglas Gregor if (IsSystem) { 285a686e1b0SDouglas Gregor OS.indent(Indent + 2); 286a686e1b0SDouglas Gregor OS << " [system]"; 287a686e1b0SDouglas Gregor } 288a686e1b0SDouglas Gregor 289a686e1b0SDouglas Gregor OS << " {\n"; 290de3ef502SDouglas Gregor 291a3feee2aSRichard Smith if (!Requirements.empty()) { 2921fb5c3a6SDouglas Gregor OS.indent(Indent + 2); 2931fb5c3a6SDouglas Gregor OS << "requires "; 294a3feee2aSRichard Smith for (unsigned I = 0, N = Requirements.size(); I != N; ++I) { 2951fb5c3a6SDouglas Gregor if (I) 2961fb5c3a6SDouglas Gregor OS << ", "; 297a3feee2aSRichard Smith if (!Requirements[I].second) 298a3feee2aSRichard Smith OS << "!"; 299a3feee2aSRichard Smith OS << Requirements[I].first; 3001fb5c3a6SDouglas Gregor } 3011fb5c3a6SDouglas Gregor OS << "\n"; 3021fb5c3a6SDouglas Gregor } 3031fb5c3a6SDouglas Gregor 30473141fa9SDouglas Gregor if (const FileEntry *UmbrellaHeader = getUmbrellaHeader()) { 305de3ef502SDouglas Gregor OS.indent(Indent + 2); 306322f633cSDouglas Gregor OS << "umbrella header \""; 307de3ef502SDouglas Gregor OS.write_escaped(UmbrellaHeader->getName()); 308de3ef502SDouglas Gregor OS << "\"\n"; 309322f633cSDouglas Gregor } else if (const DirectoryEntry *UmbrellaDir = getUmbrellaDir()) { 310322f633cSDouglas Gregor OS.indent(Indent + 2); 311322f633cSDouglas Gregor OS << "umbrella \""; 312322f633cSDouglas Gregor OS.write_escaped(UmbrellaDir->getName()); 313322f633cSDouglas Gregor OS << "\"\n"; 314de3ef502SDouglas Gregor } 315de3ef502SDouglas Gregor 31635b13eceSDouglas Gregor if (!ConfigMacros.empty() || ConfigMacrosExhaustive) { 31735b13eceSDouglas Gregor OS.indent(Indent + 2); 31835b13eceSDouglas Gregor OS << "config_macros "; 31935b13eceSDouglas Gregor if (ConfigMacrosExhaustive) 3208d932427SDouglas Gregor OS << "[exhaustive]"; 32135b13eceSDouglas Gregor for (unsigned I = 0, N = ConfigMacros.size(); I != N; ++I) { 32235b13eceSDouglas Gregor if (I) 32335b13eceSDouglas Gregor OS << ", "; 32435b13eceSDouglas Gregor OS << ConfigMacros[I]; 32535b13eceSDouglas Gregor } 3268d932427SDouglas Gregor OS << "\n"; 32735b13eceSDouglas Gregor } 32835b13eceSDouglas Gregor 329b53e5483SLawrence Crowl for (unsigned I = 0, N = NormalHeaders.size(); I != N; ++I) { 330de3ef502SDouglas Gregor OS.indent(Indent + 2); 331de3ef502SDouglas Gregor OS << "header \""; 332b53e5483SLawrence Crowl OS.write_escaped(NormalHeaders[I]->getName()); 333de3ef502SDouglas Gregor OS << "\"\n"; 334de3ef502SDouglas Gregor } 33559527666SDouglas Gregor 33659527666SDouglas Gregor for (unsigned I = 0, N = ExcludedHeaders.size(); I != N; ++I) { 33759527666SDouglas Gregor OS.indent(Indent + 2); 33859527666SDouglas Gregor OS << "exclude header \""; 33959527666SDouglas Gregor OS.write_escaped(ExcludedHeaders[I]->getName()); 34059527666SDouglas Gregor OS << "\"\n"; 34159527666SDouglas Gregor } 342de3ef502SDouglas Gregor 343b53e5483SLawrence Crowl for (unsigned I = 0, N = PrivateHeaders.size(); I != N; ++I) { 344b53e5483SLawrence Crowl OS.indent(Indent + 2); 345b53e5483SLawrence Crowl OS << "private header \""; 346b53e5483SLawrence Crowl OS.write_escaped(PrivateHeaders[I]->getName()); 347b53e5483SLawrence Crowl OS << "\"\n"; 348b53e5483SLawrence Crowl } 349b53e5483SLawrence Crowl 350eb90e830SDouglas Gregor for (submodule_const_iterator MI = submodule_begin(), MIEnd = submodule_end(); 351de3ef502SDouglas Gregor MI != MIEnd; ++MI) 352eb90e830SDouglas Gregor (*MI)->print(OS, Indent + 2); 353de3ef502SDouglas Gregor 35424bb923aSDouglas Gregor for (unsigned I = 0, N = Exports.size(); I != N; ++I) { 35524bb923aSDouglas Gregor OS.indent(Indent + 2); 3568c7c8352SDouglas Gregor OS << "export "; 3578c7c8352SDouglas Gregor if (Module *Restriction = Exports[I].getPointer()) { 3588c7c8352SDouglas Gregor OS << Restriction->getFullModuleName(); 35924bb923aSDouglas Gregor if (Exports[I].getInt()) 36024bb923aSDouglas Gregor OS << ".*"; 3618c7c8352SDouglas Gregor } else { 3628c7c8352SDouglas Gregor OS << "*"; 3638c7c8352SDouglas Gregor } 36424bb923aSDouglas Gregor OS << "\n"; 36524bb923aSDouglas Gregor } 36624bb923aSDouglas Gregor 36724bb923aSDouglas Gregor for (unsigned I = 0, N = UnresolvedExports.size(); I != N; ++I) { 36824bb923aSDouglas Gregor OS.indent(Indent + 2); 36924bb923aSDouglas Gregor OS << "export "; 37024bb923aSDouglas Gregor printModuleId(OS, UnresolvedExports[I].Id); 3718c7c8352SDouglas Gregor if (UnresolvedExports[I].Wildcard) { 3728c7c8352SDouglas Gregor if (UnresolvedExports[I].Id.empty()) 3738c7c8352SDouglas Gregor OS << "*"; 3748c7c8352SDouglas Gregor else 37524bb923aSDouglas Gregor OS << ".*"; 3768c7c8352SDouglas Gregor } 37724bb923aSDouglas Gregor OS << "\n"; 37824bb923aSDouglas Gregor } 37924bb923aSDouglas Gregor 380ba7f2f71SDaniel Jasper for (unsigned I = 0, N = DirectUses.size(); I != N; ++I) { 381ba7f2f71SDaniel Jasper OS.indent(Indent + 2); 382ba7f2f71SDaniel Jasper OS << "use "; 383ba7f2f71SDaniel Jasper OS << DirectUses[I]->getFullModuleName(); 384ba7f2f71SDaniel Jasper OS << "\n"; 385ba7f2f71SDaniel Jasper } 386ba7f2f71SDaniel Jasper 387ba7f2f71SDaniel Jasper for (unsigned I = 0, N = UnresolvedDirectUses.size(); I != N; ++I) { 388ba7f2f71SDaniel Jasper OS.indent(Indent + 2); 389ba7f2f71SDaniel Jasper OS << "use "; 390ba7f2f71SDaniel Jasper printModuleId(OS, UnresolvedDirectUses[I]); 391ba7f2f71SDaniel Jasper OS << "\n"; 392ba7f2f71SDaniel Jasper } 393ba7f2f71SDaniel Jasper 3946ddfca91SDouglas Gregor for (unsigned I = 0, N = LinkLibraries.size(); I != N; ++I) { 3956ddfca91SDouglas Gregor OS.indent(Indent + 2); 3966ddfca91SDouglas Gregor OS << "link "; 3976ddfca91SDouglas Gregor if (LinkLibraries[I].IsFramework) 3986ddfca91SDouglas Gregor OS << "framework "; 3996ddfca91SDouglas Gregor OS << "\""; 4006ddfca91SDouglas Gregor OS.write_escaped(LinkLibraries[I].Library); 4016ddfca91SDouglas Gregor OS << "\""; 4026ddfca91SDouglas Gregor } 4036ddfca91SDouglas Gregor 404fb912657SDouglas Gregor for (unsigned I = 0, N = UnresolvedConflicts.size(); I != N; ++I) { 405fb912657SDouglas Gregor OS.indent(Indent + 2); 406fb912657SDouglas Gregor OS << "conflict "; 407fb912657SDouglas Gregor printModuleId(OS, UnresolvedConflicts[I].Id); 408fb912657SDouglas Gregor OS << ", \""; 409fb912657SDouglas Gregor OS.write_escaped(UnresolvedConflicts[I].Message); 410fb912657SDouglas Gregor OS << "\"\n"; 411fb912657SDouglas Gregor } 412fb912657SDouglas Gregor 413fb912657SDouglas Gregor for (unsigned I = 0, N = Conflicts.size(); I != N; ++I) { 414fb912657SDouglas Gregor OS.indent(Indent + 2); 415fb912657SDouglas Gregor OS << "conflict "; 416fb912657SDouglas Gregor OS << Conflicts[I].Other->getFullModuleName(); 417fb912657SDouglas Gregor OS << ", \""; 418fb912657SDouglas Gregor OS.write_escaped(Conflicts[I].Message); 419fb912657SDouglas Gregor OS << "\"\n"; 420fb912657SDouglas Gregor } 421fb912657SDouglas Gregor 42273441091SDouglas Gregor if (InferSubmodules) { 42373441091SDouglas Gregor OS.indent(Indent + 2); 42473441091SDouglas Gregor if (InferExplicitSubmodules) 42573441091SDouglas Gregor OS << "explicit "; 42673441091SDouglas Gregor OS << "module * {\n"; 42773441091SDouglas Gregor if (InferExportWildcard) { 42873441091SDouglas Gregor OS.indent(Indent + 4); 42973441091SDouglas Gregor OS << "export *\n"; 43073441091SDouglas Gregor } 43173441091SDouglas Gregor OS.indent(Indent + 2); 43273441091SDouglas Gregor OS << "}\n"; 43373441091SDouglas Gregor } 43473441091SDouglas Gregor 435de3ef502SDouglas Gregor OS.indent(Indent); 436de3ef502SDouglas Gregor OS << "}\n"; 437de3ef502SDouglas Gregor } 438de3ef502SDouglas Gregor 439de3ef502SDouglas Gregor void Module::dump() const { 440de3ef502SDouglas Gregor print(llvm::errs()); 441de3ef502SDouglas Gregor } 442de3ef502SDouglas Gregor 443de3ef502SDouglas Gregor 444