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, 72*0761a8a0SDaniel Jasper Requirement &Req, HeaderDirective &MissingHeader) const { 731fb5c3a6SDouglas Gregor if (IsAvailable) 741fb5c3a6SDouglas Gregor return true; 751fb5c3a6SDouglas Gregor 761fb5c3a6SDouglas Gregor for (const Module *Current = this; Current; Current = Current->Parent) { 77*0761a8a0SDaniel Jasper if (!Current->MissingHeaders.empty()) { 78*0761a8a0SDaniel Jasper MissingHeader = Current->MissingHeaders.front(); 79*0761a8a0SDaniel Jasper return false; 80*0761a8a0SDaniel Jasper } 81a3feee2aSRichard Smith for (unsigned I = 0, N = Current->Requirements.size(); I != N; ++I) { 82a3feee2aSRichard Smith if (hasFeature(Current->Requirements[I].first, LangOpts, Target) != 83a3feee2aSRichard Smith Current->Requirements[I].second) { 84a3feee2aSRichard Smith Req = Current->Requirements[I]; 851fb5c3a6SDouglas Gregor return false; 861fb5c3a6SDouglas Gregor } 871fb5c3a6SDouglas Gregor } 881fb5c3a6SDouglas Gregor } 891fb5c3a6SDouglas Gregor 901fb5c3a6SDouglas Gregor llvm_unreachable("could not find a reason why module is unavailable"); 911fb5c3a6SDouglas Gregor } 921fb5c3a6SDouglas Gregor 93f5eedd05SDouglas Gregor bool Module::isSubModuleOf(Module *Other) const { 94f5eedd05SDouglas Gregor const Module *This = this; 95f5eedd05SDouglas Gregor do { 96f5eedd05SDouglas Gregor if (This == Other) 97f5eedd05SDouglas Gregor return true; 98f5eedd05SDouglas Gregor 99f5eedd05SDouglas Gregor This = This->Parent; 100f5eedd05SDouglas Gregor } while (This); 101f5eedd05SDouglas Gregor 102f5eedd05SDouglas Gregor return false; 103f5eedd05SDouglas Gregor } 104f5eedd05SDouglas Gregor 10573441091SDouglas Gregor const Module *Module::getTopLevelModule() const { 10673441091SDouglas Gregor const Module *Result = this; 10773441091SDouglas Gregor while (Result->Parent) 10873441091SDouglas Gregor Result = Result->Parent; 10973441091SDouglas Gregor 11073441091SDouglas Gregor return Result; 11173441091SDouglas Gregor } 11273441091SDouglas Gregor 113de3ef502SDouglas Gregor std::string Module::getFullModuleName() const { 114f857950dSDmitri Gribenko SmallVector<StringRef, 2> Names; 115de3ef502SDouglas Gregor 116de3ef502SDouglas Gregor // Build up the set of module names (from innermost to outermost). 117de3ef502SDouglas Gregor for (const Module *M = this; M; M = M->Parent) 118de3ef502SDouglas Gregor Names.push_back(M->Name); 119de3ef502SDouglas Gregor 120de3ef502SDouglas Gregor std::string Result; 12161ac906bSCraig Topper for (SmallVectorImpl<StringRef>::reverse_iterator I = Names.rbegin(), 122de3ef502SDouglas Gregor IEnd = Names.rend(); 123de3ef502SDouglas Gregor I != IEnd; ++I) { 124de3ef502SDouglas Gregor if (!Result.empty()) 125de3ef502SDouglas Gregor Result += '.'; 126de3ef502SDouglas Gregor 127de3ef502SDouglas Gregor Result += *I; 128de3ef502SDouglas Gregor } 129de3ef502SDouglas Gregor 130de3ef502SDouglas Gregor return Result; 131de3ef502SDouglas Gregor } 132de3ef502SDouglas Gregor 13373141fa9SDouglas Gregor const DirectoryEntry *Module::getUmbrellaDir() const { 13473141fa9SDouglas Gregor if (const FileEntry *Header = getUmbrellaHeader()) 13573141fa9SDouglas Gregor return Header->getDir(); 13673141fa9SDouglas Gregor 13773141fa9SDouglas Gregor return Umbrella.dyn_cast<const DirectoryEntry *>(); 13873141fa9SDouglas Gregor } 13973141fa9SDouglas Gregor 1403c5305c1SArgyrios Kyrtzidis ArrayRef<const FileEntry *> Module::getTopHeaders(FileManager &FileMgr) { 1413c5305c1SArgyrios Kyrtzidis if (!TopHeaderNames.empty()) { 1423c5305c1SArgyrios Kyrtzidis for (std::vector<std::string>::iterator 1433c5305c1SArgyrios Kyrtzidis I = TopHeaderNames.begin(), E = TopHeaderNames.end(); I != E; ++I) { 1443c5305c1SArgyrios Kyrtzidis if (const FileEntry *FE = FileMgr.getFile(*I)) 1453c5305c1SArgyrios Kyrtzidis TopHeaders.insert(FE); 1463c5305c1SArgyrios Kyrtzidis } 1473c5305c1SArgyrios Kyrtzidis TopHeaderNames.clear(); 1483c5305c1SArgyrios Kyrtzidis } 1493c5305c1SArgyrios Kyrtzidis 1503c5305c1SArgyrios Kyrtzidis return llvm::makeArrayRef(TopHeaders.begin(), TopHeaders.end()); 1513c5305c1SArgyrios Kyrtzidis } 1523c5305c1SArgyrios Kyrtzidis 153a3feee2aSRichard Smith void Module::addRequirement(StringRef Feature, bool RequiredState, 154a3feee2aSRichard Smith const LangOptions &LangOpts, 15589929282SDouglas Gregor const TargetInfo &Target) { 156a3feee2aSRichard Smith Requirements.push_back(Requirement(Feature, RequiredState)); 1571fb5c3a6SDouglas Gregor 1581fb5c3a6SDouglas Gregor // If this feature is currently available, we're done. 159a3feee2aSRichard Smith if (hasFeature(Feature, LangOpts, Target) == RequiredState) 1601fb5c3a6SDouglas Gregor return; 1611fb5c3a6SDouglas Gregor 1621fb5c3a6SDouglas Gregor if (!IsAvailable) 1631fb5c3a6SDouglas Gregor return; 1641fb5c3a6SDouglas Gregor 165f857950dSDmitri Gribenko SmallVector<Module *, 2> Stack; 1661fb5c3a6SDouglas Gregor Stack.push_back(this); 1671fb5c3a6SDouglas Gregor while (!Stack.empty()) { 1681fb5c3a6SDouglas Gregor Module *Current = Stack.back(); 1691fb5c3a6SDouglas Gregor Stack.pop_back(); 1701fb5c3a6SDouglas Gregor 1711fb5c3a6SDouglas Gregor if (!Current->IsAvailable) 1721fb5c3a6SDouglas Gregor continue; 1731fb5c3a6SDouglas Gregor 1741fb5c3a6SDouglas Gregor Current->IsAvailable = false; 175eb90e830SDouglas Gregor for (submodule_iterator Sub = Current->submodule_begin(), 176eb90e830SDouglas Gregor SubEnd = Current->submodule_end(); 1771fb5c3a6SDouglas Gregor Sub != SubEnd; ++Sub) { 178eb90e830SDouglas Gregor if ((*Sub)->IsAvailable) 179eb90e830SDouglas Gregor Stack.push_back(*Sub); 1801fb5c3a6SDouglas Gregor } 1811fb5c3a6SDouglas Gregor } 1821fb5c3a6SDouglas Gregor } 1831fb5c3a6SDouglas Gregor 184eb90e830SDouglas Gregor Module *Module::findSubmodule(StringRef Name) const { 185eb90e830SDouglas Gregor llvm::StringMap<unsigned>::const_iterator Pos = SubModuleIndex.find(Name); 186eb90e830SDouglas Gregor if (Pos == SubModuleIndex.end()) 187eb90e830SDouglas Gregor return 0; 188eb90e830SDouglas Gregor 189eb90e830SDouglas Gregor return SubModules[Pos->getValue()]; 190eb90e830SDouglas Gregor } 191eb90e830SDouglas Gregor 192f857950dSDmitri Gribenko static void printModuleId(raw_ostream &OS, const ModuleId &Id) { 19324bb923aSDouglas Gregor for (unsigned I = 0, N = Id.size(); I != N; ++I) { 19424bb923aSDouglas Gregor if (I) 19524bb923aSDouglas Gregor OS << "."; 19624bb923aSDouglas Gregor OS << Id[I].first; 19724bb923aSDouglas Gregor } 19824bb923aSDouglas Gregor } 19924bb923aSDouglas Gregor 2008739f7b7SArgyrios Kyrtzidis void Module::getExportedModules(SmallVectorImpl<Module *> &Exported) const { 201e9bcf5b7SDmitri Gribenko // All non-explicit submodules are exported. 202e9bcf5b7SDmitri Gribenko for (std::vector<Module *>::const_iterator I = SubModules.begin(), 203e9bcf5b7SDmitri Gribenko E = SubModules.end(); 204e9bcf5b7SDmitri Gribenko I != E; ++I) { 205e9bcf5b7SDmitri Gribenko Module *Mod = *I; 206e9bcf5b7SDmitri Gribenko if (!Mod->IsExplicit) 207e9bcf5b7SDmitri Gribenko Exported.push_back(Mod); 208e9bcf5b7SDmitri Gribenko } 209e9bcf5b7SDmitri Gribenko 210e9bcf5b7SDmitri Gribenko // Find re-exported modules by filtering the list of imported modules. 2118739f7b7SArgyrios Kyrtzidis bool AnyWildcard = false; 2128739f7b7SArgyrios Kyrtzidis bool UnrestrictedWildcard = false; 2138739f7b7SArgyrios Kyrtzidis SmallVector<Module *, 4> WildcardRestrictions; 2148739f7b7SArgyrios Kyrtzidis for (unsigned I = 0, N = Exports.size(); I != N; ++I) { 2158739f7b7SArgyrios Kyrtzidis Module *Mod = Exports[I].getPointer(); 2168739f7b7SArgyrios Kyrtzidis if (!Exports[I].getInt()) { 2178739f7b7SArgyrios Kyrtzidis // Export a named module directly; no wildcards involved. 2188739f7b7SArgyrios Kyrtzidis Exported.push_back(Mod); 2198739f7b7SArgyrios Kyrtzidis 2208739f7b7SArgyrios Kyrtzidis continue; 2218739f7b7SArgyrios Kyrtzidis } 2228739f7b7SArgyrios Kyrtzidis 2238739f7b7SArgyrios Kyrtzidis // Wildcard export: export all of the imported modules that match 2248739f7b7SArgyrios Kyrtzidis // the given pattern. 2258739f7b7SArgyrios Kyrtzidis AnyWildcard = true; 2268739f7b7SArgyrios Kyrtzidis if (UnrestrictedWildcard) 2278739f7b7SArgyrios Kyrtzidis continue; 2288739f7b7SArgyrios Kyrtzidis 2298739f7b7SArgyrios Kyrtzidis if (Module *Restriction = Exports[I].getPointer()) 2308739f7b7SArgyrios Kyrtzidis WildcardRestrictions.push_back(Restriction); 2318739f7b7SArgyrios Kyrtzidis else { 2328739f7b7SArgyrios Kyrtzidis WildcardRestrictions.clear(); 2338739f7b7SArgyrios Kyrtzidis UnrestrictedWildcard = true; 2348739f7b7SArgyrios Kyrtzidis } 2358739f7b7SArgyrios Kyrtzidis } 2368739f7b7SArgyrios Kyrtzidis 2378739f7b7SArgyrios Kyrtzidis // If there were any wildcards, push any imported modules that were 2388739f7b7SArgyrios Kyrtzidis // re-exported by the wildcard restriction. 2398739f7b7SArgyrios Kyrtzidis if (!AnyWildcard) 2408739f7b7SArgyrios Kyrtzidis return; 2418739f7b7SArgyrios Kyrtzidis 2428739f7b7SArgyrios Kyrtzidis for (unsigned I = 0, N = Imports.size(); I != N; ++I) { 2438739f7b7SArgyrios Kyrtzidis Module *Mod = Imports[I]; 2448739f7b7SArgyrios Kyrtzidis bool Acceptable = UnrestrictedWildcard; 2458739f7b7SArgyrios Kyrtzidis if (!Acceptable) { 2468739f7b7SArgyrios Kyrtzidis // Check whether this module meets one of the restrictions. 2478739f7b7SArgyrios Kyrtzidis for (unsigned R = 0, NR = WildcardRestrictions.size(); R != NR; ++R) { 2488739f7b7SArgyrios Kyrtzidis Module *Restriction = WildcardRestrictions[R]; 2498739f7b7SArgyrios Kyrtzidis if (Mod == Restriction || Mod->isSubModuleOf(Restriction)) { 2508739f7b7SArgyrios Kyrtzidis Acceptable = true; 2518739f7b7SArgyrios Kyrtzidis break; 2528739f7b7SArgyrios Kyrtzidis } 2538739f7b7SArgyrios Kyrtzidis } 2548739f7b7SArgyrios Kyrtzidis } 2558739f7b7SArgyrios Kyrtzidis 2568739f7b7SArgyrios Kyrtzidis if (!Acceptable) 2578739f7b7SArgyrios Kyrtzidis continue; 2588739f7b7SArgyrios Kyrtzidis 2598739f7b7SArgyrios Kyrtzidis Exported.push_back(Mod); 2608739f7b7SArgyrios Kyrtzidis } 2618739f7b7SArgyrios Kyrtzidis } 2628739f7b7SArgyrios Kyrtzidis 2630e5d7b8cSRichard Smith void Module::buildVisibleModulesCache() const { 2640e5d7b8cSRichard Smith assert(VisibleModulesCache.empty() && "cache does not need building"); 2650e5d7b8cSRichard Smith 2660e5d7b8cSRichard Smith // This module is visible to itself. 2670e5d7b8cSRichard Smith VisibleModulesCache.insert(this); 2680e5d7b8cSRichard Smith 2690e5d7b8cSRichard Smith // Every imported module is visible. 270dde17e74SRichard Smith SmallVector<Module *, 16> Stack(Imports.begin(), Imports.end()); 271dc360d57SDmitri Gribenko while (!Stack.empty()) { 272dc360d57SDmitri Gribenko Module *CurrModule = Stack.pop_back_val(); 273dc360d57SDmitri Gribenko 274dde17e74SRichard Smith // Every module transitively exported by an imported module is visible. 275dde17e74SRichard Smith if (VisibleModulesCache.insert(CurrModule).second) 276dde17e74SRichard Smith CurrModule->getExportedModules(Stack); 2770e5d7b8cSRichard Smith } 2780e5d7b8cSRichard Smith } 2790e5d7b8cSRichard Smith 280f857950dSDmitri Gribenko void Module::print(raw_ostream &OS, unsigned Indent) const { 281de3ef502SDouglas Gregor OS.indent(Indent); 282de3ef502SDouglas Gregor if (IsFramework) 283de3ef502SDouglas Gregor OS << "framework "; 284de3ef502SDouglas Gregor if (IsExplicit) 285de3ef502SDouglas Gregor OS << "explicit "; 286a686e1b0SDouglas Gregor OS << "module " << Name; 287a686e1b0SDouglas Gregor 288a686e1b0SDouglas Gregor if (IsSystem) { 289a686e1b0SDouglas Gregor OS.indent(Indent + 2); 290a686e1b0SDouglas Gregor OS << " [system]"; 291a686e1b0SDouglas Gregor } 292a686e1b0SDouglas Gregor 293a686e1b0SDouglas Gregor OS << " {\n"; 294de3ef502SDouglas Gregor 295a3feee2aSRichard Smith if (!Requirements.empty()) { 2961fb5c3a6SDouglas Gregor OS.indent(Indent + 2); 2971fb5c3a6SDouglas Gregor OS << "requires "; 298a3feee2aSRichard Smith for (unsigned I = 0, N = Requirements.size(); I != N; ++I) { 2991fb5c3a6SDouglas Gregor if (I) 3001fb5c3a6SDouglas Gregor OS << ", "; 301a3feee2aSRichard Smith if (!Requirements[I].second) 302a3feee2aSRichard Smith OS << "!"; 303a3feee2aSRichard Smith OS << Requirements[I].first; 3041fb5c3a6SDouglas Gregor } 3051fb5c3a6SDouglas Gregor OS << "\n"; 3061fb5c3a6SDouglas Gregor } 3071fb5c3a6SDouglas Gregor 30873141fa9SDouglas Gregor if (const FileEntry *UmbrellaHeader = getUmbrellaHeader()) { 309de3ef502SDouglas Gregor OS.indent(Indent + 2); 310322f633cSDouglas Gregor OS << "umbrella header \""; 311de3ef502SDouglas Gregor OS.write_escaped(UmbrellaHeader->getName()); 312de3ef502SDouglas Gregor OS << "\"\n"; 313322f633cSDouglas Gregor } else if (const DirectoryEntry *UmbrellaDir = getUmbrellaDir()) { 314322f633cSDouglas Gregor OS.indent(Indent + 2); 315322f633cSDouglas Gregor OS << "umbrella \""; 316322f633cSDouglas Gregor OS.write_escaped(UmbrellaDir->getName()); 317322f633cSDouglas Gregor OS << "\"\n"; 318de3ef502SDouglas Gregor } 319de3ef502SDouglas Gregor 32035b13eceSDouglas Gregor if (!ConfigMacros.empty() || ConfigMacrosExhaustive) { 32135b13eceSDouglas Gregor OS.indent(Indent + 2); 32235b13eceSDouglas Gregor OS << "config_macros "; 32335b13eceSDouglas Gregor if (ConfigMacrosExhaustive) 3248d932427SDouglas Gregor OS << "[exhaustive]"; 32535b13eceSDouglas Gregor for (unsigned I = 0, N = ConfigMacros.size(); I != N; ++I) { 32635b13eceSDouglas Gregor if (I) 32735b13eceSDouglas Gregor OS << ", "; 32835b13eceSDouglas Gregor OS << ConfigMacros[I]; 32935b13eceSDouglas Gregor } 3308d932427SDouglas Gregor OS << "\n"; 33135b13eceSDouglas Gregor } 33235b13eceSDouglas Gregor 333b53e5483SLawrence Crowl for (unsigned I = 0, N = NormalHeaders.size(); I != N; ++I) { 334de3ef502SDouglas Gregor OS.indent(Indent + 2); 335de3ef502SDouglas Gregor OS << "header \""; 336b53e5483SLawrence Crowl OS.write_escaped(NormalHeaders[I]->getName()); 337de3ef502SDouglas Gregor OS << "\"\n"; 338de3ef502SDouglas Gregor } 33959527666SDouglas Gregor 34059527666SDouglas Gregor for (unsigned I = 0, N = ExcludedHeaders.size(); I != N; ++I) { 34159527666SDouglas Gregor OS.indent(Indent + 2); 34259527666SDouglas Gregor OS << "exclude header \""; 34359527666SDouglas Gregor OS.write_escaped(ExcludedHeaders[I]->getName()); 34459527666SDouglas Gregor OS << "\"\n"; 34559527666SDouglas Gregor } 346de3ef502SDouglas Gregor 347b53e5483SLawrence Crowl for (unsigned I = 0, N = PrivateHeaders.size(); I != N; ++I) { 348b53e5483SLawrence Crowl OS.indent(Indent + 2); 349b53e5483SLawrence Crowl OS << "private header \""; 350b53e5483SLawrence Crowl OS.write_escaped(PrivateHeaders[I]->getName()); 351b53e5483SLawrence Crowl OS << "\"\n"; 352b53e5483SLawrence Crowl } 353b53e5483SLawrence Crowl 354eb90e830SDouglas Gregor for (submodule_const_iterator MI = submodule_begin(), MIEnd = submodule_end(); 355de3ef502SDouglas Gregor MI != MIEnd; ++MI) 356eb90e830SDouglas Gregor (*MI)->print(OS, Indent + 2); 357de3ef502SDouglas Gregor 35824bb923aSDouglas Gregor for (unsigned I = 0, N = Exports.size(); I != N; ++I) { 35924bb923aSDouglas Gregor OS.indent(Indent + 2); 3608c7c8352SDouglas Gregor OS << "export "; 3618c7c8352SDouglas Gregor if (Module *Restriction = Exports[I].getPointer()) { 3628c7c8352SDouglas Gregor OS << Restriction->getFullModuleName(); 36324bb923aSDouglas Gregor if (Exports[I].getInt()) 36424bb923aSDouglas Gregor OS << ".*"; 3658c7c8352SDouglas Gregor } else { 3668c7c8352SDouglas Gregor OS << "*"; 3678c7c8352SDouglas Gregor } 36824bb923aSDouglas Gregor OS << "\n"; 36924bb923aSDouglas Gregor } 37024bb923aSDouglas Gregor 37124bb923aSDouglas Gregor for (unsigned I = 0, N = UnresolvedExports.size(); I != N; ++I) { 37224bb923aSDouglas Gregor OS.indent(Indent + 2); 37324bb923aSDouglas Gregor OS << "export "; 37424bb923aSDouglas Gregor printModuleId(OS, UnresolvedExports[I].Id); 3758c7c8352SDouglas Gregor if (UnresolvedExports[I].Wildcard) { 3768c7c8352SDouglas Gregor if (UnresolvedExports[I].Id.empty()) 3778c7c8352SDouglas Gregor OS << "*"; 3788c7c8352SDouglas Gregor else 37924bb923aSDouglas Gregor OS << ".*"; 3808c7c8352SDouglas Gregor } 38124bb923aSDouglas Gregor OS << "\n"; 38224bb923aSDouglas Gregor } 38324bb923aSDouglas Gregor 384ba7f2f71SDaniel Jasper for (unsigned I = 0, N = DirectUses.size(); I != N; ++I) { 385ba7f2f71SDaniel Jasper OS.indent(Indent + 2); 386ba7f2f71SDaniel Jasper OS << "use "; 387ba7f2f71SDaniel Jasper OS << DirectUses[I]->getFullModuleName(); 388ba7f2f71SDaniel Jasper OS << "\n"; 389ba7f2f71SDaniel Jasper } 390ba7f2f71SDaniel Jasper 391ba7f2f71SDaniel Jasper for (unsigned I = 0, N = UnresolvedDirectUses.size(); I != N; ++I) { 392ba7f2f71SDaniel Jasper OS.indent(Indent + 2); 393ba7f2f71SDaniel Jasper OS << "use "; 394ba7f2f71SDaniel Jasper printModuleId(OS, UnresolvedDirectUses[I]); 395ba7f2f71SDaniel Jasper OS << "\n"; 396ba7f2f71SDaniel Jasper } 397ba7f2f71SDaniel Jasper 3986ddfca91SDouglas Gregor for (unsigned I = 0, N = LinkLibraries.size(); I != N; ++I) { 3996ddfca91SDouglas Gregor OS.indent(Indent + 2); 4006ddfca91SDouglas Gregor OS << "link "; 4016ddfca91SDouglas Gregor if (LinkLibraries[I].IsFramework) 4026ddfca91SDouglas Gregor OS << "framework "; 4036ddfca91SDouglas Gregor OS << "\""; 4046ddfca91SDouglas Gregor OS.write_escaped(LinkLibraries[I].Library); 4056ddfca91SDouglas Gregor OS << "\""; 4066ddfca91SDouglas Gregor } 4076ddfca91SDouglas Gregor 408fb912657SDouglas Gregor for (unsigned I = 0, N = UnresolvedConflicts.size(); I != N; ++I) { 409fb912657SDouglas Gregor OS.indent(Indent + 2); 410fb912657SDouglas Gregor OS << "conflict "; 411fb912657SDouglas Gregor printModuleId(OS, UnresolvedConflicts[I].Id); 412fb912657SDouglas Gregor OS << ", \""; 413fb912657SDouglas Gregor OS.write_escaped(UnresolvedConflicts[I].Message); 414fb912657SDouglas Gregor OS << "\"\n"; 415fb912657SDouglas Gregor } 416fb912657SDouglas Gregor 417fb912657SDouglas Gregor for (unsigned I = 0, N = Conflicts.size(); I != N; ++I) { 418fb912657SDouglas Gregor OS.indent(Indent + 2); 419fb912657SDouglas Gregor OS << "conflict "; 420fb912657SDouglas Gregor OS << Conflicts[I].Other->getFullModuleName(); 421fb912657SDouglas Gregor OS << ", \""; 422fb912657SDouglas Gregor OS.write_escaped(Conflicts[I].Message); 423fb912657SDouglas Gregor OS << "\"\n"; 424fb912657SDouglas Gregor } 425fb912657SDouglas Gregor 42673441091SDouglas Gregor if (InferSubmodules) { 42773441091SDouglas Gregor OS.indent(Indent + 2); 42873441091SDouglas Gregor if (InferExplicitSubmodules) 42973441091SDouglas Gregor OS << "explicit "; 43073441091SDouglas Gregor OS << "module * {\n"; 43173441091SDouglas Gregor if (InferExportWildcard) { 43273441091SDouglas Gregor OS.indent(Indent + 4); 43373441091SDouglas Gregor OS << "export *\n"; 43473441091SDouglas Gregor } 43573441091SDouglas Gregor OS.indent(Indent + 2); 43673441091SDouglas Gregor OS << "}\n"; 43773441091SDouglas Gregor } 43873441091SDouglas Gregor 439de3ef502SDouglas Gregor OS.indent(Indent); 440de3ef502SDouglas Gregor OS << "}\n"; 441de3ef502SDouglas Gregor } 442de3ef502SDouglas Gregor 443de3ef502SDouglas Gregor void Module::dump() const { 444de3ef502SDouglas Gregor print(llvm::errs()); 445de3ef502SDouglas Gregor } 446de3ef502SDouglas Gregor 447de3ef502SDouglas Gregor 448