1718292f2SDouglas Gregor //===--- ModuleMap.cpp - Describe the layout of modules ---------*- C++ -*-===// 2718292f2SDouglas Gregor // 3718292f2SDouglas Gregor // The LLVM Compiler Infrastructure 4718292f2SDouglas Gregor // 5718292f2SDouglas Gregor // This file is distributed under the University of Illinois Open Source 6718292f2SDouglas Gregor // License. See LICENSE.TXT for details. 7718292f2SDouglas Gregor // 8718292f2SDouglas Gregor //===----------------------------------------------------------------------===// 9718292f2SDouglas Gregor // 10718292f2SDouglas Gregor // This file defines the ModuleMap implementation, which describes the layout 11718292f2SDouglas Gregor // of a module as it relates to headers. 12718292f2SDouglas Gregor // 13718292f2SDouglas Gregor //===----------------------------------------------------------------------===// 14718292f2SDouglas Gregor #include "clang/Lex/ModuleMap.h" 15a7d03840SJordan Rose #include "clang/Basic/CharInfo.h" 16718292f2SDouglas Gregor #include "clang/Basic/Diagnostic.h" 17811db4eaSDouglas Gregor #include "clang/Basic/DiagnosticOptions.h" 18718292f2SDouglas Gregor #include "clang/Basic/FileManager.h" 19718292f2SDouglas Gregor #include "clang/Basic/TargetInfo.h" 20718292f2SDouglas Gregor #include "clang/Basic/TargetOptions.h" 21b146baabSArgyrios Kyrtzidis #include "clang/Lex/HeaderSearch.h" 223a02247dSChandler Carruth #include "clang/Lex/LexDiagnostic.h" 233a02247dSChandler Carruth #include "clang/Lex/Lexer.h" 243a02247dSChandler Carruth #include "clang/Lex/LiteralSupport.h" 253a02247dSChandler Carruth #include "llvm/ADT/StringRef.h" 263a02247dSChandler Carruth #include "llvm/ADT/StringSwitch.h" 27718292f2SDouglas Gregor #include "llvm/Support/Allocator.h" 28e89dbc1dSDouglas Gregor #include "llvm/Support/FileSystem.h" 29718292f2SDouglas Gregor #include "llvm/Support/Host.h" 30552c169eSRafael Espindola #include "llvm/Support/Path.h" 31718292f2SDouglas Gregor #include "llvm/Support/raw_ostream.h" 3207c22b78SDouglas Gregor #include <stdlib.h> 3301c7cfa2SDouglas Gregor #if defined(LLVM_ON_UNIX) 34eadae014SDmitri Gribenko #include <limits.h> 3501c7cfa2SDouglas Gregor #endif 36718292f2SDouglas Gregor using namespace clang; 37718292f2SDouglas Gregor 382b82c2a5SDouglas Gregor Module::ExportDecl 392b82c2a5SDouglas Gregor ModuleMap::resolveExport(Module *Mod, 402b82c2a5SDouglas Gregor const Module::UnresolvedExportDecl &Unresolved, 41e4412640SArgyrios Kyrtzidis bool Complain) const { 42f5eedd05SDouglas Gregor // We may have just a wildcard. 43f5eedd05SDouglas Gregor if (Unresolved.Id.empty()) { 44f5eedd05SDouglas Gregor assert(Unresolved.Wildcard && "Invalid unresolved export"); 45f5eedd05SDouglas Gregor return Module::ExportDecl(0, true); 46f5eedd05SDouglas Gregor } 47f5eedd05SDouglas Gregor 48fb912657SDouglas Gregor // Resolve the module-id. 49fb912657SDouglas Gregor Module *Context = resolveModuleId(Unresolved.Id, Mod, Complain); 50fb912657SDouglas Gregor if (!Context) 51fb912657SDouglas Gregor return Module::ExportDecl(); 52fb912657SDouglas Gregor 53fb912657SDouglas Gregor return Module::ExportDecl(Context, Unresolved.Wildcard); 54fb912657SDouglas Gregor } 55fb912657SDouglas Gregor 56fb912657SDouglas Gregor Module *ModuleMap::resolveModuleId(const ModuleId &Id, Module *Mod, 57fb912657SDouglas Gregor bool Complain) const { 582b82c2a5SDouglas Gregor // Find the starting module. 59fb912657SDouglas Gregor Module *Context = lookupModuleUnqualified(Id[0].first, Mod); 602b82c2a5SDouglas Gregor if (!Context) { 612b82c2a5SDouglas Gregor if (Complain) 620761a8a0SDaniel Jasper Diags.Report(Id[0].second, diag::err_mmap_missing_module_unqualified) 63fb912657SDouglas Gregor << Id[0].first << Mod->getFullModuleName(); 642b82c2a5SDouglas Gregor 65fb912657SDouglas Gregor return 0; 662b82c2a5SDouglas Gregor } 672b82c2a5SDouglas Gregor 682b82c2a5SDouglas Gregor // Dig into the module path. 69fb912657SDouglas Gregor for (unsigned I = 1, N = Id.size(); I != N; ++I) { 70fb912657SDouglas Gregor Module *Sub = lookupModuleQualified(Id[I].first, Context); 712b82c2a5SDouglas Gregor if (!Sub) { 722b82c2a5SDouglas Gregor if (Complain) 730761a8a0SDaniel Jasper Diags.Report(Id[I].second, diag::err_mmap_missing_module_qualified) 74fb912657SDouglas Gregor << Id[I].first << Context->getFullModuleName() 75fb912657SDouglas Gregor << SourceRange(Id[0].second, Id[I-1].second); 762b82c2a5SDouglas Gregor 77fb912657SDouglas Gregor return 0; 782b82c2a5SDouglas Gregor } 792b82c2a5SDouglas Gregor 802b82c2a5SDouglas Gregor Context = Sub; 812b82c2a5SDouglas Gregor } 822b82c2a5SDouglas Gregor 83fb912657SDouglas Gregor return Context; 842b82c2a5SDouglas Gregor } 852b82c2a5SDouglas Gregor 860761a8a0SDaniel Jasper ModuleMap::ModuleMap(SourceManager &SourceMgr, DiagnosticsEngine &Diags, 87b146baabSArgyrios Kyrtzidis const LangOptions &LangOpts, const TargetInfo *Target, 88b146baabSArgyrios Kyrtzidis HeaderSearch &HeaderInfo) 890761a8a0SDaniel Jasper : SourceMgr(SourceMgr), Diags(Diags), LangOpts(LangOpts), Target(Target), 901f76c4e8SManuel Klimek HeaderInfo(HeaderInfo), BuiltinIncludeDir(0), CompilingModule(0), 910761a8a0SDaniel Jasper SourceModule(0) {} 92718292f2SDouglas Gregor 93718292f2SDouglas Gregor ModuleMap::~ModuleMap() { 945acdf59eSDouglas Gregor for (llvm::StringMap<Module *>::iterator I = Modules.begin(), 955acdf59eSDouglas Gregor IEnd = Modules.end(); 965acdf59eSDouglas Gregor I != IEnd; ++I) { 975acdf59eSDouglas Gregor delete I->getValue(); 985acdf59eSDouglas Gregor } 99718292f2SDouglas Gregor } 100718292f2SDouglas Gregor 10189929282SDouglas Gregor void ModuleMap::setTarget(const TargetInfo &Target) { 10289929282SDouglas Gregor assert((!this->Target || this->Target == &Target) && 10389929282SDouglas Gregor "Improper target override"); 10489929282SDouglas Gregor this->Target = &Target; 10589929282SDouglas Gregor } 10689929282SDouglas Gregor 107056396aeSDouglas Gregor /// \brief "Sanitize" a filename so that it can be used as an identifier. 108056396aeSDouglas Gregor static StringRef sanitizeFilenameAsIdentifier(StringRef Name, 109056396aeSDouglas Gregor SmallVectorImpl<char> &Buffer) { 110056396aeSDouglas Gregor if (Name.empty()) 111056396aeSDouglas Gregor return Name; 112056396aeSDouglas Gregor 113a7d03840SJordan Rose if (!isValidIdentifier(Name)) { 114056396aeSDouglas Gregor // If we don't already have something with the form of an identifier, 115056396aeSDouglas Gregor // create a buffer with the sanitized name. 116056396aeSDouglas Gregor Buffer.clear(); 117a7d03840SJordan Rose if (isDigit(Name[0])) 118056396aeSDouglas Gregor Buffer.push_back('_'); 119056396aeSDouglas Gregor Buffer.reserve(Buffer.size() + Name.size()); 120056396aeSDouglas Gregor for (unsigned I = 0, N = Name.size(); I != N; ++I) { 121a7d03840SJordan Rose if (isIdentifierBody(Name[I])) 122056396aeSDouglas Gregor Buffer.push_back(Name[I]); 123056396aeSDouglas Gregor else 124056396aeSDouglas Gregor Buffer.push_back('_'); 125056396aeSDouglas Gregor } 126056396aeSDouglas Gregor 127056396aeSDouglas Gregor Name = StringRef(Buffer.data(), Buffer.size()); 128056396aeSDouglas Gregor } 129056396aeSDouglas Gregor 130056396aeSDouglas Gregor while (llvm::StringSwitch<bool>(Name) 131056396aeSDouglas Gregor #define KEYWORD(Keyword,Conditions) .Case(#Keyword, true) 132056396aeSDouglas Gregor #define ALIAS(Keyword, AliasOf, Conditions) .Case(Keyword, true) 133056396aeSDouglas Gregor #include "clang/Basic/TokenKinds.def" 134056396aeSDouglas Gregor .Default(false)) { 135056396aeSDouglas Gregor if (Name.data() != Buffer.data()) 136056396aeSDouglas Gregor Buffer.append(Name.begin(), Name.end()); 137056396aeSDouglas Gregor Buffer.push_back('_'); 138056396aeSDouglas Gregor Name = StringRef(Buffer.data(), Buffer.size()); 139056396aeSDouglas Gregor } 140056396aeSDouglas Gregor 141056396aeSDouglas Gregor return Name; 142056396aeSDouglas Gregor } 143056396aeSDouglas Gregor 14434d52749SDouglas Gregor /// \brief Determine whether the given file name is the name of a builtin 14534d52749SDouglas Gregor /// header, supplied by Clang to replace, override, or augment existing system 14634d52749SDouglas Gregor /// headers. 14734d52749SDouglas Gregor static bool isBuiltinHeader(StringRef FileName) { 14834d52749SDouglas Gregor return llvm::StringSwitch<bool>(FileName) 14934d52749SDouglas Gregor .Case("float.h", true) 15034d52749SDouglas Gregor .Case("iso646.h", true) 15134d52749SDouglas Gregor .Case("limits.h", true) 15234d52749SDouglas Gregor .Case("stdalign.h", true) 15334d52749SDouglas Gregor .Case("stdarg.h", true) 15434d52749SDouglas Gregor .Case("stdbool.h", true) 15534d52749SDouglas Gregor .Case("stddef.h", true) 15634d52749SDouglas Gregor .Case("stdint.h", true) 15734d52749SDouglas Gregor .Case("tgmath.h", true) 15834d52749SDouglas Gregor .Case("unwind.h", true) 15934d52749SDouglas Gregor .Default(false); 16034d52749SDouglas Gregor } 16134d52749SDouglas Gregor 16292669ee4SDaniel Jasper ModuleMap::HeadersMap::iterator 16392669ee4SDaniel Jasper ModuleMap::findKnownHeader(const FileEntry *File) { 16459527666SDouglas Gregor HeadersMap::iterator Known = Headers.find(File); 1654eaf0a6cSDaniel Jasper if (Known == Headers.end() && File->getDir() == BuiltinIncludeDir && 1664eaf0a6cSDaniel Jasper isBuiltinHeader(llvm::sys::path::filename(File->getName()))) { 1674eaf0a6cSDaniel Jasper HeaderInfo.loadTopLevelSystemModules(); 16892669ee4SDaniel Jasper return Headers.find(File); 1694eaf0a6cSDaniel Jasper } 17092669ee4SDaniel Jasper return Known; 17192669ee4SDaniel Jasper } 17292669ee4SDaniel Jasper 17392669ee4SDaniel Jasper // Returns 'true' if 'RequestingModule directly uses 'RequestedModule'. 17492669ee4SDaniel Jasper static bool directlyUses(const Module *RequestingModule, 17592669ee4SDaniel Jasper const Module *RequestedModule) { 17692669ee4SDaniel Jasper return std::find(RequestingModule->DirectUses.begin(), 17792669ee4SDaniel Jasper RequestingModule->DirectUses.end(), 17892669ee4SDaniel Jasper RequestedModule) != RequestingModule->DirectUses.end(); 17992669ee4SDaniel Jasper } 18092669ee4SDaniel Jasper 18192669ee4SDaniel Jasper static bool violatesPrivateInclude(Module *RequestingModule, 18292669ee4SDaniel Jasper const FileEntry *IncFileEnt, 18392669ee4SDaniel Jasper ModuleMap::ModuleHeaderRole Role, 18492669ee4SDaniel Jasper Module *RequestedModule) { 18592669ee4SDaniel Jasper #ifndef NDEBUG 18692669ee4SDaniel Jasper // Check for consistency between the module header role 18792669ee4SDaniel Jasper // as obtained from the lookup and as obtained from the module. 18892669ee4SDaniel Jasper // This check is not cheap, so enable it only for debugging. 18992669ee4SDaniel Jasper SmallVectorImpl<const FileEntry *> &PvtHdrs 19092669ee4SDaniel Jasper = RequestedModule->PrivateHeaders; 19192669ee4SDaniel Jasper SmallVectorImpl<const FileEntry *>::iterator Look 19292669ee4SDaniel Jasper = std::find(PvtHdrs.begin(), PvtHdrs.end(), IncFileEnt); 19392669ee4SDaniel Jasper bool IsPrivate = Look != PvtHdrs.end(); 19492669ee4SDaniel Jasper assert((IsPrivate && Role == ModuleMap::PrivateHeader) 19592669ee4SDaniel Jasper || (!IsPrivate && Role != ModuleMap::PrivateHeader)); 19692669ee4SDaniel Jasper #endif 19792669ee4SDaniel Jasper return Role == ModuleMap::PrivateHeader && 19892669ee4SDaniel Jasper RequestedModule->getTopLevelModule() != RequestingModule; 19992669ee4SDaniel Jasper } 20092669ee4SDaniel Jasper 20192669ee4SDaniel Jasper void ModuleMap::diagnoseHeaderInclusion(Module *RequestingModule, 20292669ee4SDaniel Jasper SourceLocation FilenameLoc, 20392669ee4SDaniel Jasper StringRef Filename, 20492669ee4SDaniel Jasper const FileEntry *File) { 20592669ee4SDaniel Jasper // No errors for indirect modules. This may be a bit of a problem for modules 20692669ee4SDaniel Jasper // with no source files. 20792669ee4SDaniel Jasper if (RequestingModule != SourceModule) 20892669ee4SDaniel Jasper return; 20992669ee4SDaniel Jasper 21092669ee4SDaniel Jasper if (RequestingModule) 21192669ee4SDaniel Jasper resolveUses(RequestingModule, /*Complain=*/false); 21292669ee4SDaniel Jasper 21392669ee4SDaniel Jasper HeadersMap::iterator Known = findKnownHeader(File); 21492669ee4SDaniel Jasper if (Known == Headers.end()) 21592669ee4SDaniel Jasper return; 21692669ee4SDaniel Jasper 21792669ee4SDaniel Jasper Module *Private = NULL; 21892669ee4SDaniel Jasper Module *NotUsed = NULL; 21992669ee4SDaniel Jasper for (SmallVectorImpl<KnownHeader>::iterator I = Known->second.begin(), 22092669ee4SDaniel Jasper E = Known->second.end(); 22192669ee4SDaniel Jasper I != E; ++I) { 22292669ee4SDaniel Jasper // Excluded headers don't really belong to a module. 22392669ee4SDaniel Jasper if (I->getRole() == ModuleMap::ExcludedHeader) 22492669ee4SDaniel Jasper continue; 22592669ee4SDaniel Jasper 22692669ee4SDaniel Jasper // If 'File' is part of 'RequestingModule' we can definitely include it. 22792669ee4SDaniel Jasper if (I->getModule() == RequestingModule) 22892669ee4SDaniel Jasper return; 22992669ee4SDaniel Jasper 23092669ee4SDaniel Jasper // Remember private headers for later printing of a diagnostic. 23192669ee4SDaniel Jasper if (violatesPrivateInclude(RequestingModule, File, I->getRole(), 23292669ee4SDaniel Jasper I->getModule())) { 23392669ee4SDaniel Jasper Private = I->getModule(); 23492669ee4SDaniel Jasper continue; 23592669ee4SDaniel Jasper } 23692669ee4SDaniel Jasper 23792669ee4SDaniel Jasper // If uses need to be specified explicitly, we are only allowed to return 23892669ee4SDaniel Jasper // modules that are explicitly used by the requesting module. 23992669ee4SDaniel Jasper if (RequestingModule && LangOpts.ModulesDeclUse && 24092669ee4SDaniel Jasper !directlyUses(RequestingModule, I->getModule())) { 24192669ee4SDaniel Jasper NotUsed = I->getModule(); 24292669ee4SDaniel Jasper continue; 24392669ee4SDaniel Jasper } 24492669ee4SDaniel Jasper 24592669ee4SDaniel Jasper // We have found a module that we can happily use. 24692669ee4SDaniel Jasper return; 24792669ee4SDaniel Jasper } 24892669ee4SDaniel Jasper 24992669ee4SDaniel Jasper // We have found a header, but it is private. 25092669ee4SDaniel Jasper if (Private != NULL) { 25192669ee4SDaniel Jasper Diags.Report(FilenameLoc, diag::error_use_of_private_header_outside_module) 25292669ee4SDaniel Jasper << Filename; 25392669ee4SDaniel Jasper return; 25492669ee4SDaniel Jasper } 25592669ee4SDaniel Jasper 25692669ee4SDaniel Jasper // We have found a module, but we don't use it. 25792669ee4SDaniel Jasper if (NotUsed != NULL) { 25892669ee4SDaniel Jasper Diags.Report(FilenameLoc, diag::error_undeclared_use_of_module) 25992669ee4SDaniel Jasper << RequestingModule->getFullModuleName() << Filename; 26092669ee4SDaniel Jasper return; 26192669ee4SDaniel Jasper } 26292669ee4SDaniel Jasper 26392669ee4SDaniel Jasper // Headers for which we have not found a module are fine to include. 26492669ee4SDaniel Jasper } 26592669ee4SDaniel Jasper 26692669ee4SDaniel Jasper ModuleMap::KnownHeader 26792669ee4SDaniel Jasper ModuleMap::findModuleForHeader(const FileEntry *File, 26892669ee4SDaniel Jasper Module *RequestingModule) { 26992669ee4SDaniel Jasper HeadersMap::iterator Known = findKnownHeader(File); 2704eaf0a6cSDaniel Jasper 2711fb5c3a6SDouglas Gregor if (Known != Headers.end()) { 27297da9178SDaniel Jasper ModuleMap::KnownHeader Result = KnownHeader(); 2731fb5c3a6SDouglas Gregor 27497da9178SDaniel Jasper // Iterate over all modules that 'File' is part of to find the best fit. 27597da9178SDaniel Jasper for (SmallVectorImpl<KnownHeader>::iterator I = Known->second.begin(), 27697da9178SDaniel Jasper E = Known->second.end(); 27797da9178SDaniel Jasper I != E; ++I) { 2784eaf0a6cSDaniel Jasper // Cannot use a module if the header is excluded in it. 2794eaf0a6cSDaniel Jasper if (I->getRole() == ModuleMap::ExcludedHeader) 2804eaf0a6cSDaniel Jasper continue; 2814eaf0a6cSDaniel Jasper 2824eaf0a6cSDaniel Jasper // Cannot use a module if it is unavailable. 2834eaf0a6cSDaniel Jasper if (!I->getModule()->isAvailable()) 28497da9178SDaniel Jasper continue; 28597da9178SDaniel Jasper 28697da9178SDaniel Jasper // If 'File' is part of 'RequestingModule', 'RequestingModule' is the 28797da9178SDaniel Jasper // module we are looking for. 28897da9178SDaniel Jasper if (I->getModule() == RequestingModule) 28997da9178SDaniel Jasper return *I; 29097da9178SDaniel Jasper 29197da9178SDaniel Jasper // If uses need to be specified explicitly, we are only allowed to return 29297da9178SDaniel Jasper // modules that are explicitly used by the requesting module. 29397da9178SDaniel Jasper if (RequestingModule && LangOpts.ModulesDeclUse && 29492669ee4SDaniel Jasper !directlyUses(RequestingModule, I->getModule())) 29597da9178SDaniel Jasper continue; 2964eaf0a6cSDaniel Jasper 29797da9178SDaniel Jasper Result = *I; 29897da9178SDaniel Jasper // If 'File' is a public header of this module, this is as good as we 29997da9178SDaniel Jasper // are going to get. 3008c71eba1SRichard Smith // FIXME: If we have a RequestingModule, we should prefer the header from 3018c71eba1SRichard Smith // that module. 30297da9178SDaniel Jasper if (I->getRole() == ModuleMap::NormalHeader) 30397da9178SDaniel Jasper break; 30497da9178SDaniel Jasper } 30597da9178SDaniel Jasper return Result; 3061fb5c3a6SDouglas Gregor } 307ab0c8a84SDouglas Gregor 308b65dbfffSDouglas Gregor const DirectoryEntry *Dir = File->getDir(); 309f857950dSDmitri Gribenko SmallVector<const DirectoryEntry *, 2> SkippedDirs; 310e00c8b20SDouglas Gregor 31174260502SDouglas Gregor // Note: as an egregious but useful hack we use the real path here, because 31274260502SDouglas Gregor // frameworks moving from top-level frameworks to embedded frameworks tend 31374260502SDouglas Gregor // to be symlinked from the top-level location to the embedded location, 31474260502SDouglas Gregor // and we need to resolve lookups as if we had found the embedded location. 3151f76c4e8SManuel Klimek StringRef DirName = SourceMgr.getFileManager().getCanonicalName(Dir); 316a89c5ac4SDouglas Gregor 317a89c5ac4SDouglas Gregor // Keep walking up the directory hierarchy, looking for a directory with 318a89c5ac4SDouglas Gregor // an umbrella header. 319b65dbfffSDouglas Gregor do { 320a89c5ac4SDouglas Gregor llvm::DenseMap<const DirectoryEntry *, Module *>::iterator KnownDir 321a89c5ac4SDouglas Gregor = UmbrellaDirs.find(Dir); 322a89c5ac4SDouglas Gregor if (KnownDir != UmbrellaDirs.end()) { 323a89c5ac4SDouglas Gregor Module *Result = KnownDir->second; 324930a85ccSDouglas Gregor 325930a85ccSDouglas Gregor // Search up the module stack until we find a module with an umbrella 32673141fa9SDouglas Gregor // directory. 327930a85ccSDouglas Gregor Module *UmbrellaModule = Result; 32873141fa9SDouglas Gregor while (!UmbrellaModule->getUmbrellaDir() && UmbrellaModule->Parent) 329930a85ccSDouglas Gregor UmbrellaModule = UmbrellaModule->Parent; 330930a85ccSDouglas Gregor 331930a85ccSDouglas Gregor if (UmbrellaModule->InferSubmodules) { 332a89c5ac4SDouglas Gregor // Infer submodules for each of the directories we found between 333a89c5ac4SDouglas Gregor // the directory of the umbrella header and the directory where 334a89c5ac4SDouglas Gregor // the actual header is located. 3359458f82dSDouglas Gregor bool Explicit = UmbrellaModule->InferExplicitSubmodules; 3369458f82dSDouglas Gregor 3377033127bSDouglas Gregor for (unsigned I = SkippedDirs.size(); I != 0; --I) { 338a89c5ac4SDouglas Gregor // Find or create the module that corresponds to this directory name. 339056396aeSDouglas Gregor SmallString<32> NameBuf; 340056396aeSDouglas Gregor StringRef Name = sanitizeFilenameAsIdentifier( 341056396aeSDouglas Gregor llvm::sys::path::stem(SkippedDirs[I-1]->getName()), 342056396aeSDouglas Gregor NameBuf); 343a89c5ac4SDouglas Gregor Result = findOrCreateModule(Name, Result, /*IsFramework=*/false, 3449458f82dSDouglas Gregor Explicit).first; 345a89c5ac4SDouglas Gregor 346a89c5ac4SDouglas Gregor // Associate the module and the directory. 347a89c5ac4SDouglas Gregor UmbrellaDirs[SkippedDirs[I-1]] = Result; 348a89c5ac4SDouglas Gregor 349a89c5ac4SDouglas Gregor // If inferred submodules export everything they import, add a 350a89c5ac4SDouglas Gregor // wildcard to the set of exports. 351930a85ccSDouglas Gregor if (UmbrellaModule->InferExportWildcard && Result->Exports.empty()) 352a89c5ac4SDouglas Gregor Result->Exports.push_back(Module::ExportDecl(0, true)); 353a89c5ac4SDouglas Gregor } 354a89c5ac4SDouglas Gregor 355a89c5ac4SDouglas Gregor // Infer a submodule with the same name as this header file. 356056396aeSDouglas Gregor SmallString<32> NameBuf; 357056396aeSDouglas Gregor StringRef Name = sanitizeFilenameAsIdentifier( 358056396aeSDouglas Gregor llvm::sys::path::stem(File->getName()), NameBuf); 359a89c5ac4SDouglas Gregor Result = findOrCreateModule(Name, Result, /*IsFramework=*/false, 3609458f82dSDouglas Gregor Explicit).first; 3613c5305c1SArgyrios Kyrtzidis Result->addTopHeader(File); 362a89c5ac4SDouglas Gregor 363a89c5ac4SDouglas Gregor // If inferred submodules export everything they import, add a 364a89c5ac4SDouglas Gregor // wildcard to the set of exports. 365930a85ccSDouglas Gregor if (UmbrellaModule->InferExportWildcard && Result->Exports.empty()) 366a89c5ac4SDouglas Gregor Result->Exports.push_back(Module::ExportDecl(0, true)); 367a89c5ac4SDouglas Gregor } else { 368a89c5ac4SDouglas Gregor // Record each of the directories we stepped through as being part of 369a89c5ac4SDouglas Gregor // the module we found, since the umbrella header covers them all. 370a89c5ac4SDouglas Gregor for (unsigned I = 0, N = SkippedDirs.size(); I != N; ++I) 371a89c5ac4SDouglas Gregor UmbrellaDirs[SkippedDirs[I]] = Result; 372a89c5ac4SDouglas Gregor } 373a89c5ac4SDouglas Gregor 37497da9178SDaniel Jasper Headers[File].push_back(KnownHeader(Result, NormalHeader)); 3751fb5c3a6SDouglas Gregor 3761fb5c3a6SDouglas Gregor // If a header corresponds to an unavailable module, don't report 3771fb5c3a6SDouglas Gregor // that it maps to anything. 3781fb5c3a6SDouglas Gregor if (!Result->isAvailable()) 379b53e5483SLawrence Crowl return KnownHeader(); 3801fb5c3a6SDouglas Gregor 38197da9178SDaniel Jasper return Headers[File].back(); 382a89c5ac4SDouglas Gregor } 383a89c5ac4SDouglas Gregor 384a89c5ac4SDouglas Gregor SkippedDirs.push_back(Dir); 385a89c5ac4SDouglas Gregor 386b65dbfffSDouglas Gregor // Retrieve our parent path. 387b65dbfffSDouglas Gregor DirName = llvm::sys::path::parent_path(DirName); 388b65dbfffSDouglas Gregor if (DirName.empty()) 389b65dbfffSDouglas Gregor break; 390b65dbfffSDouglas Gregor 391b65dbfffSDouglas Gregor // Resolve the parent path to a directory entry. 3921f76c4e8SManuel Klimek Dir = SourceMgr.getFileManager().getDirectory(DirName); 393a89c5ac4SDouglas Gregor } while (Dir); 394b65dbfffSDouglas Gregor 395b53e5483SLawrence Crowl return KnownHeader(); 396ab0c8a84SDouglas Gregor } 397ab0c8a84SDouglas Gregor 398e4412640SArgyrios Kyrtzidis bool ModuleMap::isHeaderInUnavailableModule(const FileEntry *Header) const { 399*50996ce1SRichard Smith return isHeaderUnavailableInModule(Header, 0); 400*50996ce1SRichard Smith } 401*50996ce1SRichard Smith 402*50996ce1SRichard Smith bool ModuleMap::isHeaderUnavailableInModule(const FileEntry *Header, 403*50996ce1SRichard Smith Module *RequestingModule) const { 404e4412640SArgyrios Kyrtzidis HeadersMap::const_iterator Known = Headers.find(Header); 40597da9178SDaniel Jasper if (Known != Headers.end()) { 40697da9178SDaniel Jasper for (SmallVectorImpl<KnownHeader>::const_iterator 40797da9178SDaniel Jasper I = Known->second.begin(), 40897da9178SDaniel Jasper E = Known->second.end(); 40997da9178SDaniel Jasper I != E; ++I) { 410*50996ce1SRichard Smith if (I->isAvailable() && (!RequestingModule || 411*50996ce1SRichard Smith I->getModule()->isSubModuleOf(RequestingModule))) 41297da9178SDaniel Jasper return false; 41397da9178SDaniel Jasper } 41497da9178SDaniel Jasper return true; 41597da9178SDaniel Jasper } 4161fb5c3a6SDouglas Gregor 4171fb5c3a6SDouglas Gregor const DirectoryEntry *Dir = Header->getDir(); 418f857950dSDmitri Gribenko SmallVector<const DirectoryEntry *, 2> SkippedDirs; 4191fb5c3a6SDouglas Gregor StringRef DirName = Dir->getName(); 4201fb5c3a6SDouglas Gregor 421*50996ce1SRichard Smith auto IsUnavailable = [&](const Module *M) { 422*50996ce1SRichard Smith return !M->isAvailable() && (!RequestingModule || 423*50996ce1SRichard Smith M->isSubModuleOf(RequestingModule)); 424*50996ce1SRichard Smith }; 425*50996ce1SRichard Smith 4261fb5c3a6SDouglas Gregor // Keep walking up the directory hierarchy, looking for a directory with 4271fb5c3a6SDouglas Gregor // an umbrella header. 4281fb5c3a6SDouglas Gregor do { 429e4412640SArgyrios Kyrtzidis llvm::DenseMap<const DirectoryEntry *, Module *>::const_iterator KnownDir 4301fb5c3a6SDouglas Gregor = UmbrellaDirs.find(Dir); 4311fb5c3a6SDouglas Gregor if (KnownDir != UmbrellaDirs.end()) { 4321fb5c3a6SDouglas Gregor Module *Found = KnownDir->second; 433*50996ce1SRichard Smith if (IsUnavailable(Found)) 4341fb5c3a6SDouglas Gregor return true; 4351fb5c3a6SDouglas Gregor 4361fb5c3a6SDouglas Gregor // Search up the module stack until we find a module with an umbrella 4371fb5c3a6SDouglas Gregor // directory. 4381fb5c3a6SDouglas Gregor Module *UmbrellaModule = Found; 4391fb5c3a6SDouglas Gregor while (!UmbrellaModule->getUmbrellaDir() && UmbrellaModule->Parent) 4401fb5c3a6SDouglas Gregor UmbrellaModule = UmbrellaModule->Parent; 4411fb5c3a6SDouglas Gregor 4421fb5c3a6SDouglas Gregor if (UmbrellaModule->InferSubmodules) { 4431fb5c3a6SDouglas Gregor for (unsigned I = SkippedDirs.size(); I != 0; --I) { 4441fb5c3a6SDouglas Gregor // Find or create the module that corresponds to this directory name. 445056396aeSDouglas Gregor SmallString<32> NameBuf; 446056396aeSDouglas Gregor StringRef Name = sanitizeFilenameAsIdentifier( 447056396aeSDouglas Gregor llvm::sys::path::stem(SkippedDirs[I-1]->getName()), 448056396aeSDouglas Gregor NameBuf); 4491fb5c3a6SDouglas Gregor Found = lookupModuleQualified(Name, Found); 4501fb5c3a6SDouglas Gregor if (!Found) 4511fb5c3a6SDouglas Gregor return false; 452*50996ce1SRichard Smith if (IsUnavailable(Found)) 4531fb5c3a6SDouglas Gregor return true; 4541fb5c3a6SDouglas Gregor } 4551fb5c3a6SDouglas Gregor 4561fb5c3a6SDouglas Gregor // Infer a submodule with the same name as this header file. 457056396aeSDouglas Gregor SmallString<32> NameBuf; 458056396aeSDouglas Gregor StringRef Name = sanitizeFilenameAsIdentifier( 459056396aeSDouglas Gregor llvm::sys::path::stem(Header->getName()), 460056396aeSDouglas Gregor NameBuf); 4611fb5c3a6SDouglas Gregor Found = lookupModuleQualified(Name, Found); 4621fb5c3a6SDouglas Gregor if (!Found) 4631fb5c3a6SDouglas Gregor return false; 4641fb5c3a6SDouglas Gregor } 4651fb5c3a6SDouglas Gregor 466*50996ce1SRichard Smith return IsUnavailable(Found); 4671fb5c3a6SDouglas Gregor } 4681fb5c3a6SDouglas Gregor 4691fb5c3a6SDouglas Gregor SkippedDirs.push_back(Dir); 4701fb5c3a6SDouglas Gregor 4711fb5c3a6SDouglas Gregor // Retrieve our parent path. 4721fb5c3a6SDouglas Gregor DirName = llvm::sys::path::parent_path(DirName); 4731fb5c3a6SDouglas Gregor if (DirName.empty()) 4741fb5c3a6SDouglas Gregor break; 4751fb5c3a6SDouglas Gregor 4761fb5c3a6SDouglas Gregor // Resolve the parent path to a directory entry. 4771f76c4e8SManuel Klimek Dir = SourceMgr.getFileManager().getDirectory(DirName); 4781fb5c3a6SDouglas Gregor } while (Dir); 4791fb5c3a6SDouglas Gregor 4801fb5c3a6SDouglas Gregor return false; 4811fb5c3a6SDouglas Gregor } 4821fb5c3a6SDouglas Gregor 483e4412640SArgyrios Kyrtzidis Module *ModuleMap::findModule(StringRef Name) const { 484e4412640SArgyrios Kyrtzidis llvm::StringMap<Module *>::const_iterator Known = Modules.find(Name); 48588bdfb0eSDouglas Gregor if (Known != Modules.end()) 48688bdfb0eSDouglas Gregor return Known->getValue(); 48788bdfb0eSDouglas Gregor 48888bdfb0eSDouglas Gregor return 0; 48988bdfb0eSDouglas Gregor } 49088bdfb0eSDouglas Gregor 491e4412640SArgyrios Kyrtzidis Module *ModuleMap::lookupModuleUnqualified(StringRef Name, 492e4412640SArgyrios Kyrtzidis Module *Context) const { 4932b82c2a5SDouglas Gregor for(; Context; Context = Context->Parent) { 4942b82c2a5SDouglas Gregor if (Module *Sub = lookupModuleQualified(Name, Context)) 4952b82c2a5SDouglas Gregor return Sub; 4962b82c2a5SDouglas Gregor } 4972b82c2a5SDouglas Gregor 4982b82c2a5SDouglas Gregor return findModule(Name); 4992b82c2a5SDouglas Gregor } 5002b82c2a5SDouglas Gregor 501e4412640SArgyrios Kyrtzidis Module *ModuleMap::lookupModuleQualified(StringRef Name, Module *Context) const{ 5022b82c2a5SDouglas Gregor if (!Context) 5032b82c2a5SDouglas Gregor return findModule(Name); 5042b82c2a5SDouglas Gregor 505eb90e830SDouglas Gregor return Context->findSubmodule(Name); 5062b82c2a5SDouglas Gregor } 5072b82c2a5SDouglas Gregor 508de3ef502SDouglas Gregor std::pair<Module *, bool> 50969021974SDouglas Gregor ModuleMap::findOrCreateModule(StringRef Name, Module *Parent, bool IsFramework, 51069021974SDouglas Gregor bool IsExplicit) { 51169021974SDouglas Gregor // Try to find an existing module with this name. 512eb90e830SDouglas Gregor if (Module *Sub = lookupModuleQualified(Name, Parent)) 513eb90e830SDouglas Gregor return std::make_pair(Sub, false); 51469021974SDouglas Gregor 51569021974SDouglas Gregor // Create a new module with this name. 51669021974SDouglas Gregor Module *Result = new Module(Name, SourceLocation(), Parent, IsFramework, 51769021974SDouglas Gregor IsExplicit); 518ba7f2f71SDaniel Jasper if (LangOpts.CurrentModule == Name) { 519ba7f2f71SDaniel Jasper SourceModule = Result; 520ba7f2f71SDaniel Jasper SourceModuleName = Name; 521ba7f2f71SDaniel Jasper } 5226f722b4eSArgyrios Kyrtzidis if (!Parent) { 52369021974SDouglas Gregor Modules[Name] = Result; 5246f722b4eSArgyrios Kyrtzidis if (!LangOpts.CurrentModule.empty() && !CompilingModule && 5256f722b4eSArgyrios Kyrtzidis Name == LangOpts.CurrentModule) { 5266f722b4eSArgyrios Kyrtzidis CompilingModule = Result; 5276f722b4eSArgyrios Kyrtzidis } 5286f722b4eSArgyrios Kyrtzidis } 52969021974SDouglas Gregor return std::make_pair(Result, true); 53069021974SDouglas Gregor } 53169021974SDouglas Gregor 5329194a91dSDouglas Gregor bool ModuleMap::canInferFrameworkModule(const DirectoryEntry *ParentDir, 533e4412640SArgyrios Kyrtzidis StringRef Name, bool &IsSystem) const { 5349194a91dSDouglas Gregor // Check whether we have already looked into the parent directory 5359194a91dSDouglas Gregor // for a module map. 536e4412640SArgyrios Kyrtzidis llvm::DenseMap<const DirectoryEntry *, InferredDirectory>::const_iterator 5379194a91dSDouglas Gregor inferred = InferredDirectories.find(ParentDir); 5389194a91dSDouglas Gregor if (inferred == InferredDirectories.end()) 5399194a91dSDouglas Gregor return false; 5409194a91dSDouglas Gregor 5419194a91dSDouglas Gregor if (!inferred->second.InferModules) 5429194a91dSDouglas Gregor return false; 5439194a91dSDouglas Gregor 5449194a91dSDouglas Gregor // We're allowed to infer for this directory, but make sure it's okay 5459194a91dSDouglas Gregor // to infer this particular module. 5469194a91dSDouglas Gregor bool canInfer = std::find(inferred->second.ExcludedModules.begin(), 5479194a91dSDouglas Gregor inferred->second.ExcludedModules.end(), 5489194a91dSDouglas Gregor Name) == inferred->second.ExcludedModules.end(); 5499194a91dSDouglas Gregor 5509194a91dSDouglas Gregor if (canInfer && inferred->second.InferSystemModules) 5519194a91dSDouglas Gregor IsSystem = true; 5529194a91dSDouglas Gregor 5539194a91dSDouglas Gregor return canInfer; 5549194a91dSDouglas Gregor } 5559194a91dSDouglas Gregor 55611dfe6feSDouglas Gregor /// \brief For a framework module, infer the framework against which we 55711dfe6feSDouglas Gregor /// should link. 55811dfe6feSDouglas Gregor static void inferFrameworkLink(Module *Mod, const DirectoryEntry *FrameworkDir, 55911dfe6feSDouglas Gregor FileManager &FileMgr) { 56011dfe6feSDouglas Gregor assert(Mod->IsFramework && "Can only infer linking for framework modules"); 56111dfe6feSDouglas Gregor assert(!Mod->isSubFramework() && 56211dfe6feSDouglas Gregor "Can only infer linking for top-level frameworks"); 56311dfe6feSDouglas Gregor 56411dfe6feSDouglas Gregor SmallString<128> LibName; 56511dfe6feSDouglas Gregor LibName += FrameworkDir->getName(); 56611dfe6feSDouglas Gregor llvm::sys::path::append(LibName, Mod->Name); 56711dfe6feSDouglas Gregor if (FileMgr.getFile(LibName)) { 56811dfe6feSDouglas Gregor Mod->LinkLibraries.push_back(Module::LinkLibrary(Mod->Name, 56911dfe6feSDouglas Gregor /*IsFramework=*/true)); 57011dfe6feSDouglas Gregor } 57111dfe6feSDouglas Gregor } 57211dfe6feSDouglas Gregor 573de3ef502SDouglas Gregor Module * 57456c64013SDouglas Gregor ModuleMap::inferFrameworkModule(StringRef ModuleName, 575e89dbc1dSDouglas Gregor const DirectoryEntry *FrameworkDir, 576a686e1b0SDouglas Gregor bool IsSystem, 577e89dbc1dSDouglas Gregor Module *Parent) { 57856c64013SDouglas Gregor // Check whether we've already found this module. 579e89dbc1dSDouglas Gregor if (Module *Mod = lookupModuleQualified(ModuleName, Parent)) 580e89dbc1dSDouglas Gregor return Mod; 581e89dbc1dSDouglas Gregor 5821f76c4e8SManuel Klimek FileManager &FileMgr = SourceMgr.getFileManager(); 58356c64013SDouglas Gregor 5849194a91dSDouglas Gregor // If the framework has a parent path from which we're allowed to infer 5859194a91dSDouglas Gregor // a framework module, do so. 5869194a91dSDouglas Gregor if (!Parent) { 5874ddf2221SDouglas Gregor // Determine whether we're allowed to infer a module map. 588e00c8b20SDouglas Gregor 5894ddf2221SDouglas Gregor // Note: as an egregious but useful hack we use the real path here, because 5904ddf2221SDouglas Gregor // we might be looking at an embedded framework that symlinks out to a 5914ddf2221SDouglas Gregor // top-level framework, and we need to infer as if we were naming the 5924ddf2221SDouglas Gregor // top-level framework. 593e00c8b20SDouglas Gregor StringRef FrameworkDirName 5941f76c4e8SManuel Klimek = SourceMgr.getFileManager().getCanonicalName(FrameworkDir); 5954ddf2221SDouglas Gregor 5969194a91dSDouglas Gregor bool canInfer = false; 5974ddf2221SDouglas Gregor if (llvm::sys::path::has_parent_path(FrameworkDirName)) { 5989194a91dSDouglas Gregor // Figure out the parent path. 5994ddf2221SDouglas Gregor StringRef Parent = llvm::sys::path::parent_path(FrameworkDirName); 6009194a91dSDouglas Gregor if (const DirectoryEntry *ParentDir = FileMgr.getDirectory(Parent)) { 6019194a91dSDouglas Gregor // Check whether we have already looked into the parent directory 6029194a91dSDouglas Gregor // for a module map. 603e4412640SArgyrios Kyrtzidis llvm::DenseMap<const DirectoryEntry *, InferredDirectory>::const_iterator 6049194a91dSDouglas Gregor inferred = InferredDirectories.find(ParentDir); 6059194a91dSDouglas Gregor if (inferred == InferredDirectories.end()) { 6069194a91dSDouglas Gregor // We haven't looked here before. Load a module map, if there is 6079194a91dSDouglas Gregor // one. 608984e1df7SBen Langmuir bool IsFrameworkDir = Parent.endswith(".framework"); 609984e1df7SBen Langmuir if (const FileEntry *ModMapFile = 610984e1df7SBen Langmuir HeaderInfo.lookupModuleMapFile(ParentDir, IsFrameworkDir)) { 611963c5535SDouglas Gregor parseModuleMapFile(ModMapFile, IsSystem); 6129194a91dSDouglas Gregor inferred = InferredDirectories.find(ParentDir); 6139194a91dSDouglas Gregor } 6149194a91dSDouglas Gregor 6159194a91dSDouglas Gregor if (inferred == InferredDirectories.end()) 6169194a91dSDouglas Gregor inferred = InferredDirectories.insert( 6179194a91dSDouglas Gregor std::make_pair(ParentDir, InferredDirectory())).first; 6189194a91dSDouglas Gregor } 6199194a91dSDouglas Gregor 6209194a91dSDouglas Gregor if (inferred->second.InferModules) { 6219194a91dSDouglas Gregor // We're allowed to infer for this directory, but make sure it's okay 6229194a91dSDouglas Gregor // to infer this particular module. 6234ddf2221SDouglas Gregor StringRef Name = llvm::sys::path::stem(FrameworkDirName); 6249194a91dSDouglas Gregor canInfer = std::find(inferred->second.ExcludedModules.begin(), 6259194a91dSDouglas Gregor inferred->second.ExcludedModules.end(), 6269194a91dSDouglas Gregor Name) == inferred->second.ExcludedModules.end(); 6279194a91dSDouglas Gregor 6289194a91dSDouglas Gregor if (inferred->second.InferSystemModules) 6299194a91dSDouglas Gregor IsSystem = true; 6309194a91dSDouglas Gregor } 6319194a91dSDouglas Gregor } 6329194a91dSDouglas Gregor } 6339194a91dSDouglas Gregor 6349194a91dSDouglas Gregor // If we're not allowed to infer a framework module, don't. 6359194a91dSDouglas Gregor if (!canInfer) 6369194a91dSDouglas Gregor return 0; 6379194a91dSDouglas Gregor } 6389194a91dSDouglas Gregor 6399194a91dSDouglas Gregor 64056c64013SDouglas Gregor // Look for an umbrella header. 6412c1dd271SDylan Noblesmith SmallString<128> UmbrellaName = StringRef(FrameworkDir->getName()); 64217381a06SBenjamin Kramer llvm::sys::path::append(UmbrellaName, "Headers", ModuleName + ".h"); 643e89dbc1dSDouglas Gregor const FileEntry *UmbrellaHeader = FileMgr.getFile(UmbrellaName); 64456c64013SDouglas Gregor 64556c64013SDouglas Gregor // FIXME: If there's no umbrella header, we could probably scan the 64656c64013SDouglas Gregor // framework to load *everything*. But, it's not clear that this is a good 64756c64013SDouglas Gregor // idea. 64856c64013SDouglas Gregor if (!UmbrellaHeader) 64956c64013SDouglas Gregor return 0; 65056c64013SDouglas Gregor 651e89dbc1dSDouglas Gregor Module *Result = new Module(ModuleName, SourceLocation(), Parent, 652e89dbc1dSDouglas Gregor /*IsFramework=*/true, /*IsExplicit=*/false); 653ba7f2f71SDaniel Jasper if (LangOpts.CurrentModule == ModuleName) { 654ba7f2f71SDaniel Jasper SourceModule = Result; 655ba7f2f71SDaniel Jasper SourceModuleName = ModuleName; 656ba7f2f71SDaniel Jasper } 657a686e1b0SDouglas Gregor if (IsSystem) 658a686e1b0SDouglas Gregor Result->IsSystem = IsSystem; 659a686e1b0SDouglas Gregor 660eb90e830SDouglas Gregor if (!Parent) 661e89dbc1dSDouglas Gregor Modules[ModuleName] = Result; 662e89dbc1dSDouglas Gregor 663322f633cSDouglas Gregor // umbrella header "umbrella-header-name" 66473141fa9SDouglas Gregor Result->Umbrella = UmbrellaHeader; 66597da9178SDaniel Jasper Headers[UmbrellaHeader].push_back(KnownHeader(Result, NormalHeader)); 6664dc71835SDouglas Gregor UmbrellaDirs[UmbrellaHeader->getDir()] = Result; 667d8bd7537SDouglas Gregor 668d8bd7537SDouglas Gregor // export * 669d8bd7537SDouglas Gregor Result->Exports.push_back(Module::ExportDecl(0, true)); 670d8bd7537SDouglas Gregor 671a89c5ac4SDouglas Gregor // module * { export * } 672a89c5ac4SDouglas Gregor Result->InferSubmodules = true; 673a89c5ac4SDouglas Gregor Result->InferExportWildcard = true; 674a89c5ac4SDouglas Gregor 675e89dbc1dSDouglas Gregor // Look for subframeworks. 676e89dbc1dSDouglas Gregor llvm::error_code EC; 6772c1dd271SDylan Noblesmith SmallString<128> SubframeworksDirName 678ddaa69cbSDouglas Gregor = StringRef(FrameworkDir->getName()); 679e89dbc1dSDouglas Gregor llvm::sys::path::append(SubframeworksDirName, "Frameworks"); 6802d4d8cb3SBenjamin Kramer llvm::sys::path::native(SubframeworksDirName); 681ddaa69cbSDouglas Gregor for (llvm::sys::fs::directory_iterator 6822d4d8cb3SBenjamin Kramer Dir(SubframeworksDirName.str(), EC), DirEnd; 683e89dbc1dSDouglas Gregor Dir != DirEnd && !EC; Dir.increment(EC)) { 684e89dbc1dSDouglas Gregor if (!StringRef(Dir->path()).endswith(".framework")) 685e89dbc1dSDouglas Gregor continue; 686f2161a70SDouglas Gregor 687e89dbc1dSDouglas Gregor if (const DirectoryEntry *SubframeworkDir 688e89dbc1dSDouglas Gregor = FileMgr.getDirectory(Dir->path())) { 68907c22b78SDouglas Gregor // Note: as an egregious but useful hack, we use the real path here and 69007c22b78SDouglas Gregor // check whether it is actually a subdirectory of the parent directory. 69107c22b78SDouglas Gregor // This will not be the case if the 'subframework' is actually a symlink 69207c22b78SDouglas Gregor // out to a top-level framework. 693e00c8b20SDouglas Gregor StringRef SubframeworkDirName = FileMgr.getCanonicalName(SubframeworkDir); 69407c22b78SDouglas Gregor bool FoundParent = false; 69507c22b78SDouglas Gregor do { 69607c22b78SDouglas Gregor // Get the parent directory name. 69707c22b78SDouglas Gregor SubframeworkDirName 69807c22b78SDouglas Gregor = llvm::sys::path::parent_path(SubframeworkDirName); 69907c22b78SDouglas Gregor if (SubframeworkDirName.empty()) 70007c22b78SDouglas Gregor break; 70107c22b78SDouglas Gregor 70207c22b78SDouglas Gregor if (FileMgr.getDirectory(SubframeworkDirName) == FrameworkDir) { 70307c22b78SDouglas Gregor FoundParent = true; 70407c22b78SDouglas Gregor break; 70507c22b78SDouglas Gregor } 70607c22b78SDouglas Gregor } while (true); 70707c22b78SDouglas Gregor 70807c22b78SDouglas Gregor if (!FoundParent) 70907c22b78SDouglas Gregor continue; 71007c22b78SDouglas Gregor 711e89dbc1dSDouglas Gregor // FIXME: Do we want to warn about subframeworks without umbrella headers? 712056396aeSDouglas Gregor SmallString<32> NameBuf; 713056396aeSDouglas Gregor inferFrameworkModule(sanitizeFilenameAsIdentifier( 714056396aeSDouglas Gregor llvm::sys::path::stem(Dir->path()), NameBuf), 715056396aeSDouglas Gregor SubframeworkDir, IsSystem, Result); 716e89dbc1dSDouglas Gregor } 717e89dbc1dSDouglas Gregor } 718e89dbc1dSDouglas Gregor 71911dfe6feSDouglas Gregor // If the module is a top-level framework, automatically link against the 72011dfe6feSDouglas Gregor // framework. 72111dfe6feSDouglas Gregor if (!Result->isSubFramework()) { 72211dfe6feSDouglas Gregor inferFrameworkLink(Result, FrameworkDir, FileMgr); 72311dfe6feSDouglas Gregor } 72411dfe6feSDouglas Gregor 72556c64013SDouglas Gregor return Result; 72656c64013SDouglas Gregor } 72756c64013SDouglas Gregor 728a89c5ac4SDouglas Gregor void ModuleMap::setUmbrellaHeader(Module *Mod, const FileEntry *UmbrellaHeader){ 72997da9178SDaniel Jasper Headers[UmbrellaHeader].push_back(KnownHeader(Mod, NormalHeader)); 73073141fa9SDouglas Gregor Mod->Umbrella = UmbrellaHeader; 7317033127bSDouglas Gregor UmbrellaDirs[UmbrellaHeader->getDir()] = Mod; 732a89c5ac4SDouglas Gregor } 733a89c5ac4SDouglas Gregor 734524e33e1SDouglas Gregor void ModuleMap::setUmbrellaDir(Module *Mod, const DirectoryEntry *UmbrellaDir) { 735524e33e1SDouglas Gregor Mod->Umbrella = UmbrellaDir; 736524e33e1SDouglas Gregor UmbrellaDirs[UmbrellaDir] = Mod; 737524e33e1SDouglas Gregor } 738524e33e1SDouglas Gregor 73959527666SDouglas Gregor void ModuleMap::addHeader(Module *Mod, const FileEntry *Header, 740b53e5483SLawrence Crowl ModuleHeaderRole Role) { 741b53e5483SLawrence Crowl if (Role == ExcludedHeader) { 74259527666SDouglas Gregor Mod->ExcludedHeaders.push_back(Header); 743b146baabSArgyrios Kyrtzidis } else { 744b53e5483SLawrence Crowl if (Role == PrivateHeader) 745b53e5483SLawrence Crowl Mod->PrivateHeaders.push_back(Header); 746b53e5483SLawrence Crowl else 747b53e5483SLawrence Crowl Mod->NormalHeaders.push_back(Header); 7486f722b4eSArgyrios Kyrtzidis bool isCompilingModuleHeader = Mod->getTopLevelModule() == CompilingModule; 749b53e5483SLawrence Crowl HeaderInfo.MarkFileModuleHeader(Header, Role, isCompilingModuleHeader); 750b146baabSArgyrios Kyrtzidis } 75197da9178SDaniel Jasper Headers[Header].push_back(KnownHeader(Mod, Role)); 752a89c5ac4SDouglas Gregor } 753a89c5ac4SDouglas Gregor 754514b636aSDouglas Gregor const FileEntry * 755e4412640SArgyrios Kyrtzidis ModuleMap::getContainingModuleMapFile(Module *Module) const { 7561f76c4e8SManuel Klimek if (Module->DefinitionLoc.isInvalid()) 757514b636aSDouglas Gregor return 0; 758514b636aSDouglas Gregor 7591f76c4e8SManuel Klimek return SourceMgr.getFileEntryForID( 7601f76c4e8SManuel Klimek SourceMgr.getFileID(Module->DefinitionLoc)); 761514b636aSDouglas Gregor } 762514b636aSDouglas Gregor 763718292f2SDouglas Gregor void ModuleMap::dump() { 764718292f2SDouglas Gregor llvm::errs() << "Modules:"; 765718292f2SDouglas Gregor for (llvm::StringMap<Module *>::iterator M = Modules.begin(), 766718292f2SDouglas Gregor MEnd = Modules.end(); 767718292f2SDouglas Gregor M != MEnd; ++M) 768d28d1b8dSDouglas Gregor M->getValue()->print(llvm::errs(), 2); 769718292f2SDouglas Gregor 770718292f2SDouglas Gregor llvm::errs() << "Headers:"; 77159527666SDouglas Gregor for (HeadersMap::iterator H = Headers.begin(), HEnd = Headers.end(); 772718292f2SDouglas Gregor H != HEnd; ++H) { 77397da9178SDaniel Jasper llvm::errs() << " \"" << H->first->getName() << "\" -> "; 77497da9178SDaniel Jasper for (SmallVectorImpl<KnownHeader>::const_iterator I = H->second.begin(), 77597da9178SDaniel Jasper E = H->second.end(); 77697da9178SDaniel Jasper I != E; ++I) { 77797da9178SDaniel Jasper if (I != H->second.begin()) 77897da9178SDaniel Jasper llvm::errs() << ","; 77997da9178SDaniel Jasper llvm::errs() << I->getModule()->getFullModuleName(); 78097da9178SDaniel Jasper } 78197da9178SDaniel Jasper llvm::errs() << "\n"; 782718292f2SDouglas Gregor } 783718292f2SDouglas Gregor } 784718292f2SDouglas Gregor 7852b82c2a5SDouglas Gregor bool ModuleMap::resolveExports(Module *Mod, bool Complain) { 7862b82c2a5SDouglas Gregor bool HadError = false; 7872b82c2a5SDouglas Gregor for (unsigned I = 0, N = Mod->UnresolvedExports.size(); I != N; ++I) { 7882b82c2a5SDouglas Gregor Module::ExportDecl Export = resolveExport(Mod, Mod->UnresolvedExports[I], 7892b82c2a5SDouglas Gregor Complain); 790f5eedd05SDouglas Gregor if (Export.getPointer() || Export.getInt()) 7912b82c2a5SDouglas Gregor Mod->Exports.push_back(Export); 7922b82c2a5SDouglas Gregor else 7932b82c2a5SDouglas Gregor HadError = true; 7942b82c2a5SDouglas Gregor } 7952b82c2a5SDouglas Gregor Mod->UnresolvedExports.clear(); 7962b82c2a5SDouglas Gregor return HadError; 7972b82c2a5SDouglas Gregor } 7982b82c2a5SDouglas Gregor 799ba7f2f71SDaniel Jasper bool ModuleMap::resolveUses(Module *Mod, bool Complain) { 800ba7f2f71SDaniel Jasper bool HadError = false; 801ba7f2f71SDaniel Jasper for (unsigned I = 0, N = Mod->UnresolvedDirectUses.size(); I != N; ++I) { 802ba7f2f71SDaniel Jasper Module *DirectUse = 803ba7f2f71SDaniel Jasper resolveModuleId(Mod->UnresolvedDirectUses[I], Mod, Complain); 804ba7f2f71SDaniel Jasper if (DirectUse) 805ba7f2f71SDaniel Jasper Mod->DirectUses.push_back(DirectUse); 806ba7f2f71SDaniel Jasper else 807ba7f2f71SDaniel Jasper HadError = true; 808ba7f2f71SDaniel Jasper } 809ba7f2f71SDaniel Jasper Mod->UnresolvedDirectUses.clear(); 810ba7f2f71SDaniel Jasper return HadError; 811ba7f2f71SDaniel Jasper } 812ba7f2f71SDaniel Jasper 813fb912657SDouglas Gregor bool ModuleMap::resolveConflicts(Module *Mod, bool Complain) { 814fb912657SDouglas Gregor bool HadError = false; 815fb912657SDouglas Gregor for (unsigned I = 0, N = Mod->UnresolvedConflicts.size(); I != N; ++I) { 816fb912657SDouglas Gregor Module *OtherMod = resolveModuleId(Mod->UnresolvedConflicts[I].Id, 817fb912657SDouglas Gregor Mod, Complain); 818fb912657SDouglas Gregor if (!OtherMod) { 819fb912657SDouglas Gregor HadError = true; 820fb912657SDouglas Gregor continue; 821fb912657SDouglas Gregor } 822fb912657SDouglas Gregor 823fb912657SDouglas Gregor Module::Conflict Conflict; 824fb912657SDouglas Gregor Conflict.Other = OtherMod; 825fb912657SDouglas Gregor Conflict.Message = Mod->UnresolvedConflicts[I].Message; 826fb912657SDouglas Gregor Mod->Conflicts.push_back(Conflict); 827fb912657SDouglas Gregor } 828fb912657SDouglas Gregor Mod->UnresolvedConflicts.clear(); 829fb912657SDouglas Gregor return HadError; 830fb912657SDouglas Gregor } 831fb912657SDouglas Gregor 8320093b3c7SDouglas Gregor Module *ModuleMap::inferModuleFromLocation(FullSourceLoc Loc) { 8330093b3c7SDouglas Gregor if (Loc.isInvalid()) 8340093b3c7SDouglas Gregor return 0; 8350093b3c7SDouglas Gregor 8360093b3c7SDouglas Gregor // Use the expansion location to determine which module we're in. 8370093b3c7SDouglas Gregor FullSourceLoc ExpansionLoc = Loc.getExpansionLoc(); 8380093b3c7SDouglas Gregor if (!ExpansionLoc.isFileID()) 8390093b3c7SDouglas Gregor return 0; 8400093b3c7SDouglas Gregor 8410093b3c7SDouglas Gregor 8420093b3c7SDouglas Gregor const SourceManager &SrcMgr = Loc.getManager(); 8430093b3c7SDouglas Gregor FileID ExpansionFileID = ExpansionLoc.getFileID(); 844224d8a74SDouglas Gregor 845224d8a74SDouglas Gregor while (const FileEntry *ExpansionFile 846224d8a74SDouglas Gregor = SrcMgr.getFileEntryForID(ExpansionFileID)) { 847224d8a74SDouglas Gregor // Find the module that owns this header (if any). 848b53e5483SLawrence Crowl if (Module *Mod = findModuleForHeader(ExpansionFile).getModule()) 849224d8a74SDouglas Gregor return Mod; 850224d8a74SDouglas Gregor 851224d8a74SDouglas Gregor // No module owns this header, so look up the inclusion chain to see if 852224d8a74SDouglas Gregor // any included header has an associated module. 853224d8a74SDouglas Gregor SourceLocation IncludeLoc = SrcMgr.getIncludeLoc(ExpansionFileID); 854224d8a74SDouglas Gregor if (IncludeLoc.isInvalid()) 8550093b3c7SDouglas Gregor return 0; 8560093b3c7SDouglas Gregor 857224d8a74SDouglas Gregor ExpansionFileID = SrcMgr.getFileID(IncludeLoc); 858224d8a74SDouglas Gregor } 859224d8a74SDouglas Gregor 860224d8a74SDouglas Gregor return 0; 8610093b3c7SDouglas Gregor } 8620093b3c7SDouglas Gregor 863718292f2SDouglas Gregor //----------------------------------------------------------------------------// 864718292f2SDouglas Gregor // Module map file parser 865718292f2SDouglas Gregor //----------------------------------------------------------------------------// 866718292f2SDouglas Gregor 867718292f2SDouglas Gregor namespace clang { 868718292f2SDouglas Gregor /// \brief A token in a module map file. 869718292f2SDouglas Gregor struct MMToken { 870718292f2SDouglas Gregor enum TokenKind { 8711fb5c3a6SDouglas Gregor Comma, 87235b13eceSDouglas Gregor ConfigMacros, 873fb912657SDouglas Gregor Conflict, 874718292f2SDouglas Gregor EndOfFile, 875718292f2SDouglas Gregor HeaderKeyword, 876718292f2SDouglas Gregor Identifier, 877a3feee2aSRichard Smith Exclaim, 87859527666SDouglas Gregor ExcludeKeyword, 879718292f2SDouglas Gregor ExplicitKeyword, 8802b82c2a5SDouglas Gregor ExportKeyword, 88197292843SDaniel Jasper ExternKeyword, 882755b2055SDouglas Gregor FrameworkKeyword, 8836ddfca91SDouglas Gregor LinkKeyword, 884718292f2SDouglas Gregor ModuleKeyword, 8852b82c2a5SDouglas Gregor Period, 886b53e5483SLawrence Crowl PrivateKeyword, 887718292f2SDouglas Gregor UmbrellaKeyword, 888ba7f2f71SDaniel Jasper UseKeyword, 8891fb5c3a6SDouglas Gregor RequiresKeyword, 8902b82c2a5SDouglas Gregor Star, 891718292f2SDouglas Gregor StringLiteral, 892718292f2SDouglas Gregor LBrace, 893a686e1b0SDouglas Gregor RBrace, 894a686e1b0SDouglas Gregor LSquare, 895a686e1b0SDouglas Gregor RSquare 896718292f2SDouglas Gregor } Kind; 897718292f2SDouglas Gregor 898718292f2SDouglas Gregor unsigned Location; 899718292f2SDouglas Gregor unsigned StringLength; 900718292f2SDouglas Gregor const char *StringData; 901718292f2SDouglas Gregor 902718292f2SDouglas Gregor void clear() { 903718292f2SDouglas Gregor Kind = EndOfFile; 904718292f2SDouglas Gregor Location = 0; 905718292f2SDouglas Gregor StringLength = 0; 906718292f2SDouglas Gregor StringData = 0; 907718292f2SDouglas Gregor } 908718292f2SDouglas Gregor 909718292f2SDouglas Gregor bool is(TokenKind K) const { return Kind == K; } 910718292f2SDouglas Gregor 911718292f2SDouglas Gregor SourceLocation getLocation() const { 912718292f2SDouglas Gregor return SourceLocation::getFromRawEncoding(Location); 913718292f2SDouglas Gregor } 914718292f2SDouglas Gregor 915718292f2SDouglas Gregor StringRef getString() const { 916718292f2SDouglas Gregor return StringRef(StringData, StringLength); 917718292f2SDouglas Gregor } 918718292f2SDouglas Gregor }; 919718292f2SDouglas Gregor 9209194a91dSDouglas Gregor /// \brief The set of attributes that can be attached to a module. 9214442605fSBill Wendling struct Attributes { 92277944868SRichard Smith Attributes() : IsSystem(), IsExternC(), IsExhaustive() { } 9239194a91dSDouglas Gregor 9249194a91dSDouglas Gregor /// \brief Whether this is a system module. 9259194a91dSDouglas Gregor unsigned IsSystem : 1; 92635b13eceSDouglas Gregor 92777944868SRichard Smith /// \brief Whether this is an extern "C" module. 92877944868SRichard Smith unsigned IsExternC : 1; 92977944868SRichard Smith 93035b13eceSDouglas Gregor /// \brief Whether this is an exhaustive set of configuration macros. 93135b13eceSDouglas Gregor unsigned IsExhaustive : 1; 9329194a91dSDouglas Gregor }; 9339194a91dSDouglas Gregor 9349194a91dSDouglas Gregor 935718292f2SDouglas Gregor class ModuleMapParser { 936718292f2SDouglas Gregor Lexer &L; 937718292f2SDouglas Gregor SourceManager &SourceMgr; 938bc10b9fbSDouglas Gregor 939bc10b9fbSDouglas Gregor /// \brief Default target information, used only for string literal 940bc10b9fbSDouglas Gregor /// parsing. 941bc10b9fbSDouglas Gregor const TargetInfo *Target; 942bc10b9fbSDouglas Gregor 943718292f2SDouglas Gregor DiagnosticsEngine &Diags; 944718292f2SDouglas Gregor ModuleMap ⤅ 945718292f2SDouglas Gregor 9465257fc63SDouglas Gregor /// \brief The directory that this module map resides in. 9475257fc63SDouglas Gregor const DirectoryEntry *Directory; 9485257fc63SDouglas Gregor 9493ec6663bSDouglas Gregor /// \brief The directory containing Clang-supplied headers. 9503ec6663bSDouglas Gregor const DirectoryEntry *BuiltinIncludeDir; 9513ec6663bSDouglas Gregor 952963c5535SDouglas Gregor /// \brief Whether this module map is in a system header directory. 953963c5535SDouglas Gregor bool IsSystem; 954963c5535SDouglas Gregor 955718292f2SDouglas Gregor /// \brief Whether an error occurred. 956718292f2SDouglas Gregor bool HadError; 957718292f2SDouglas Gregor 958718292f2SDouglas Gregor /// \brief Stores string data for the various string literals referenced 959718292f2SDouglas Gregor /// during parsing. 960718292f2SDouglas Gregor llvm::BumpPtrAllocator StringData; 961718292f2SDouglas Gregor 962718292f2SDouglas Gregor /// \brief The current token. 963718292f2SDouglas Gregor MMToken Tok; 964718292f2SDouglas Gregor 965718292f2SDouglas Gregor /// \brief The active module. 966de3ef502SDouglas Gregor Module *ActiveModule; 967718292f2SDouglas Gregor 968718292f2SDouglas Gregor /// \brief Consume the current token and return its location. 969718292f2SDouglas Gregor SourceLocation consumeToken(); 970718292f2SDouglas Gregor 971718292f2SDouglas Gregor /// \brief Skip tokens until we reach the a token with the given kind 972718292f2SDouglas Gregor /// (or the end of the file). 973718292f2SDouglas Gregor void skipUntil(MMToken::TokenKind K); 974718292f2SDouglas Gregor 975f857950dSDmitri Gribenko typedef SmallVector<std::pair<std::string, SourceLocation>, 2> ModuleId; 976e7ab3669SDouglas Gregor bool parseModuleId(ModuleId &Id); 977718292f2SDouglas Gregor void parseModuleDecl(); 97897292843SDaniel Jasper void parseExternModuleDecl(); 9791fb5c3a6SDouglas Gregor void parseRequiresDecl(); 980b53e5483SLawrence Crowl void parseHeaderDecl(clang::MMToken::TokenKind, 981b53e5483SLawrence Crowl SourceLocation LeadingLoc); 982524e33e1SDouglas Gregor void parseUmbrellaDirDecl(SourceLocation UmbrellaLoc); 9832b82c2a5SDouglas Gregor void parseExportDecl(); 984ba7f2f71SDaniel Jasper void parseUseDecl(); 9856ddfca91SDouglas Gregor void parseLinkDecl(); 98635b13eceSDouglas Gregor void parseConfigMacros(); 987fb912657SDouglas Gregor void parseConflict(); 9889194a91dSDouglas Gregor void parseInferredModuleDecl(bool Framework, bool Explicit); 9894442605fSBill Wendling bool parseOptionalAttributes(Attributes &Attrs); 990718292f2SDouglas Gregor 9917033127bSDouglas Gregor const DirectoryEntry *getOverriddenHeaderSearchDir(); 9927033127bSDouglas Gregor 993718292f2SDouglas Gregor public: 994718292f2SDouglas Gregor explicit ModuleMapParser(Lexer &L, SourceManager &SourceMgr, 995bc10b9fbSDouglas Gregor const TargetInfo *Target, 996718292f2SDouglas Gregor DiagnosticsEngine &Diags, 9975257fc63SDouglas Gregor ModuleMap &Map, 9983ec6663bSDouglas Gregor const DirectoryEntry *Directory, 999963c5535SDouglas Gregor const DirectoryEntry *BuiltinIncludeDir, 1000963c5535SDouglas Gregor bool IsSystem) 1001bc10b9fbSDouglas Gregor : L(L), SourceMgr(SourceMgr), Target(Target), Diags(Diags), Map(Map), 10023ec6663bSDouglas Gregor Directory(Directory), BuiltinIncludeDir(BuiltinIncludeDir), 1003963c5535SDouglas Gregor IsSystem(IsSystem), HadError(false), ActiveModule(0) 1004718292f2SDouglas Gregor { 1005718292f2SDouglas Gregor Tok.clear(); 1006718292f2SDouglas Gregor consumeToken(); 1007718292f2SDouglas Gregor } 1008718292f2SDouglas Gregor 1009718292f2SDouglas Gregor bool parseModuleMapFile(); 1010718292f2SDouglas Gregor }; 1011718292f2SDouglas Gregor } 1012718292f2SDouglas Gregor 1013718292f2SDouglas Gregor SourceLocation ModuleMapParser::consumeToken() { 1014718292f2SDouglas Gregor retry: 1015718292f2SDouglas Gregor SourceLocation Result = Tok.getLocation(); 1016718292f2SDouglas Gregor Tok.clear(); 1017718292f2SDouglas Gregor 1018718292f2SDouglas Gregor Token LToken; 1019718292f2SDouglas Gregor L.LexFromRawLexer(LToken); 1020718292f2SDouglas Gregor Tok.Location = LToken.getLocation().getRawEncoding(); 1021718292f2SDouglas Gregor switch (LToken.getKind()) { 1022718292f2SDouglas Gregor case tok::raw_identifier: 1023718292f2SDouglas Gregor Tok.StringData = LToken.getRawIdentifierData(); 1024718292f2SDouglas Gregor Tok.StringLength = LToken.getLength(); 1025718292f2SDouglas Gregor Tok.Kind = llvm::StringSwitch<MMToken::TokenKind>(Tok.getString()) 102635b13eceSDouglas Gregor .Case("config_macros", MMToken::ConfigMacros) 1027fb912657SDouglas Gregor .Case("conflict", MMToken::Conflict) 102859527666SDouglas Gregor .Case("exclude", MMToken::ExcludeKeyword) 1029718292f2SDouglas Gregor .Case("explicit", MMToken::ExplicitKeyword) 10302b82c2a5SDouglas Gregor .Case("export", MMToken::ExportKeyword) 103197292843SDaniel Jasper .Case("extern", MMToken::ExternKeyword) 1032755b2055SDouglas Gregor .Case("framework", MMToken::FrameworkKeyword) 103335b13eceSDouglas Gregor .Case("header", MMToken::HeaderKeyword) 10346ddfca91SDouglas Gregor .Case("link", MMToken::LinkKeyword) 1035718292f2SDouglas Gregor .Case("module", MMToken::ModuleKeyword) 1036b53e5483SLawrence Crowl .Case("private", MMToken::PrivateKeyword) 10371fb5c3a6SDouglas Gregor .Case("requires", MMToken::RequiresKeyword) 1038718292f2SDouglas Gregor .Case("umbrella", MMToken::UmbrellaKeyword) 1039ba7f2f71SDaniel Jasper .Case("use", MMToken::UseKeyword) 1040718292f2SDouglas Gregor .Default(MMToken::Identifier); 1041718292f2SDouglas Gregor break; 1042718292f2SDouglas Gregor 10431fb5c3a6SDouglas Gregor case tok::comma: 10441fb5c3a6SDouglas Gregor Tok.Kind = MMToken::Comma; 10451fb5c3a6SDouglas Gregor break; 10461fb5c3a6SDouglas Gregor 1047718292f2SDouglas Gregor case tok::eof: 1048718292f2SDouglas Gregor Tok.Kind = MMToken::EndOfFile; 1049718292f2SDouglas Gregor break; 1050718292f2SDouglas Gregor 1051718292f2SDouglas Gregor case tok::l_brace: 1052718292f2SDouglas Gregor Tok.Kind = MMToken::LBrace; 1053718292f2SDouglas Gregor break; 1054718292f2SDouglas Gregor 1055a686e1b0SDouglas Gregor case tok::l_square: 1056a686e1b0SDouglas Gregor Tok.Kind = MMToken::LSquare; 1057a686e1b0SDouglas Gregor break; 1058a686e1b0SDouglas Gregor 10592b82c2a5SDouglas Gregor case tok::period: 10602b82c2a5SDouglas Gregor Tok.Kind = MMToken::Period; 10612b82c2a5SDouglas Gregor break; 10622b82c2a5SDouglas Gregor 1063718292f2SDouglas Gregor case tok::r_brace: 1064718292f2SDouglas Gregor Tok.Kind = MMToken::RBrace; 1065718292f2SDouglas Gregor break; 1066718292f2SDouglas Gregor 1067a686e1b0SDouglas Gregor case tok::r_square: 1068a686e1b0SDouglas Gregor Tok.Kind = MMToken::RSquare; 1069a686e1b0SDouglas Gregor break; 1070a686e1b0SDouglas Gregor 10712b82c2a5SDouglas Gregor case tok::star: 10722b82c2a5SDouglas Gregor Tok.Kind = MMToken::Star; 10732b82c2a5SDouglas Gregor break; 10742b82c2a5SDouglas Gregor 1075a3feee2aSRichard Smith case tok::exclaim: 1076a3feee2aSRichard Smith Tok.Kind = MMToken::Exclaim; 1077a3feee2aSRichard Smith break; 1078a3feee2aSRichard Smith 1079718292f2SDouglas Gregor case tok::string_literal: { 1080d67aea28SRichard Smith if (LToken.hasUDSuffix()) { 1081d67aea28SRichard Smith Diags.Report(LToken.getLocation(), diag::err_invalid_string_udl); 1082d67aea28SRichard Smith HadError = true; 1083d67aea28SRichard Smith goto retry; 1084d67aea28SRichard Smith } 1085d67aea28SRichard Smith 1086718292f2SDouglas Gregor // Parse the string literal. 1087718292f2SDouglas Gregor LangOptions LangOpts; 1088718292f2SDouglas Gregor StringLiteralParser StringLiteral(<oken, 1, SourceMgr, LangOpts, *Target); 1089718292f2SDouglas Gregor if (StringLiteral.hadError) 1090718292f2SDouglas Gregor goto retry; 1091718292f2SDouglas Gregor 1092718292f2SDouglas Gregor // Copy the string literal into our string data allocator. 1093718292f2SDouglas Gregor unsigned Length = StringLiteral.GetStringLength(); 1094718292f2SDouglas Gregor char *Saved = StringData.Allocate<char>(Length + 1); 1095718292f2SDouglas Gregor memcpy(Saved, StringLiteral.GetString().data(), Length); 1096718292f2SDouglas Gregor Saved[Length] = 0; 1097718292f2SDouglas Gregor 1098718292f2SDouglas Gregor // Form the token. 1099718292f2SDouglas Gregor Tok.Kind = MMToken::StringLiteral; 1100718292f2SDouglas Gregor Tok.StringData = Saved; 1101718292f2SDouglas Gregor Tok.StringLength = Length; 1102718292f2SDouglas Gregor break; 1103718292f2SDouglas Gregor } 1104718292f2SDouglas Gregor 1105718292f2SDouglas Gregor case tok::comment: 1106718292f2SDouglas Gregor goto retry; 1107718292f2SDouglas Gregor 1108718292f2SDouglas Gregor default: 1109718292f2SDouglas Gregor Diags.Report(LToken.getLocation(), diag::err_mmap_unknown_token); 1110718292f2SDouglas Gregor HadError = true; 1111718292f2SDouglas Gregor goto retry; 1112718292f2SDouglas Gregor } 1113718292f2SDouglas Gregor 1114718292f2SDouglas Gregor return Result; 1115718292f2SDouglas Gregor } 1116718292f2SDouglas Gregor 1117718292f2SDouglas Gregor void ModuleMapParser::skipUntil(MMToken::TokenKind K) { 1118718292f2SDouglas Gregor unsigned braceDepth = 0; 1119a686e1b0SDouglas Gregor unsigned squareDepth = 0; 1120718292f2SDouglas Gregor do { 1121718292f2SDouglas Gregor switch (Tok.Kind) { 1122718292f2SDouglas Gregor case MMToken::EndOfFile: 1123718292f2SDouglas Gregor return; 1124718292f2SDouglas Gregor 1125718292f2SDouglas Gregor case MMToken::LBrace: 1126a686e1b0SDouglas Gregor if (Tok.is(K) && braceDepth == 0 && squareDepth == 0) 1127718292f2SDouglas Gregor return; 1128718292f2SDouglas Gregor 1129718292f2SDouglas Gregor ++braceDepth; 1130718292f2SDouglas Gregor break; 1131718292f2SDouglas Gregor 1132a686e1b0SDouglas Gregor case MMToken::LSquare: 1133a686e1b0SDouglas Gregor if (Tok.is(K) && braceDepth == 0 && squareDepth == 0) 1134a686e1b0SDouglas Gregor return; 1135a686e1b0SDouglas Gregor 1136a686e1b0SDouglas Gregor ++squareDepth; 1137a686e1b0SDouglas Gregor break; 1138a686e1b0SDouglas Gregor 1139718292f2SDouglas Gregor case MMToken::RBrace: 1140718292f2SDouglas Gregor if (braceDepth > 0) 1141718292f2SDouglas Gregor --braceDepth; 1142718292f2SDouglas Gregor else if (Tok.is(K)) 1143718292f2SDouglas Gregor return; 1144718292f2SDouglas Gregor break; 1145718292f2SDouglas Gregor 1146a686e1b0SDouglas Gregor case MMToken::RSquare: 1147a686e1b0SDouglas Gregor if (squareDepth > 0) 1148a686e1b0SDouglas Gregor --squareDepth; 1149a686e1b0SDouglas Gregor else if (Tok.is(K)) 1150a686e1b0SDouglas Gregor return; 1151a686e1b0SDouglas Gregor break; 1152a686e1b0SDouglas Gregor 1153718292f2SDouglas Gregor default: 1154a686e1b0SDouglas Gregor if (braceDepth == 0 && squareDepth == 0 && Tok.is(K)) 1155718292f2SDouglas Gregor return; 1156718292f2SDouglas Gregor break; 1157718292f2SDouglas Gregor } 1158718292f2SDouglas Gregor 1159718292f2SDouglas Gregor consumeToken(); 1160718292f2SDouglas Gregor } while (true); 1161718292f2SDouglas Gregor } 1162718292f2SDouglas Gregor 1163e7ab3669SDouglas Gregor /// \brief Parse a module-id. 1164e7ab3669SDouglas Gregor /// 1165e7ab3669SDouglas Gregor /// module-id: 1166e7ab3669SDouglas Gregor /// identifier 1167e7ab3669SDouglas Gregor /// identifier '.' module-id 1168e7ab3669SDouglas Gregor /// 1169e7ab3669SDouglas Gregor /// \returns true if an error occurred, false otherwise. 1170e7ab3669SDouglas Gregor bool ModuleMapParser::parseModuleId(ModuleId &Id) { 1171e7ab3669SDouglas Gregor Id.clear(); 1172e7ab3669SDouglas Gregor do { 11733cd34c76SDaniel Jasper if (Tok.is(MMToken::Identifier) || Tok.is(MMToken::StringLiteral)) { 1174e7ab3669SDouglas Gregor Id.push_back(std::make_pair(Tok.getString(), Tok.getLocation())); 1175e7ab3669SDouglas Gregor consumeToken(); 1176e7ab3669SDouglas Gregor } else { 1177e7ab3669SDouglas Gregor Diags.Report(Tok.getLocation(), diag::err_mmap_expected_module_name); 1178e7ab3669SDouglas Gregor return true; 1179e7ab3669SDouglas Gregor } 1180e7ab3669SDouglas Gregor 1181e7ab3669SDouglas Gregor if (!Tok.is(MMToken::Period)) 1182e7ab3669SDouglas Gregor break; 1183e7ab3669SDouglas Gregor 1184e7ab3669SDouglas Gregor consumeToken(); 1185e7ab3669SDouglas Gregor } while (true); 1186e7ab3669SDouglas Gregor 1187e7ab3669SDouglas Gregor return false; 1188e7ab3669SDouglas Gregor } 1189e7ab3669SDouglas Gregor 1190a686e1b0SDouglas Gregor namespace { 1191a686e1b0SDouglas Gregor /// \brief Enumerates the known attributes. 1192a686e1b0SDouglas Gregor enum AttributeKind { 1193a686e1b0SDouglas Gregor /// \brief An unknown attribute. 1194a686e1b0SDouglas Gregor AT_unknown, 1195a686e1b0SDouglas Gregor /// \brief The 'system' attribute. 119635b13eceSDouglas Gregor AT_system, 119777944868SRichard Smith /// \brief The 'extern_c' attribute. 119877944868SRichard Smith AT_extern_c, 119935b13eceSDouglas Gregor /// \brief The 'exhaustive' attribute. 120035b13eceSDouglas Gregor AT_exhaustive 1201a686e1b0SDouglas Gregor }; 1202a686e1b0SDouglas Gregor } 1203a686e1b0SDouglas Gregor 1204718292f2SDouglas Gregor /// \brief Parse a module declaration. 1205718292f2SDouglas Gregor /// 1206718292f2SDouglas Gregor /// module-declaration: 120797292843SDaniel Jasper /// 'extern' 'module' module-id string-literal 1208a686e1b0SDouglas Gregor /// 'explicit'[opt] 'framework'[opt] 'module' module-id attributes[opt] 1209a686e1b0SDouglas Gregor /// { module-member* } 1210a686e1b0SDouglas Gregor /// 1211718292f2SDouglas Gregor /// module-member: 12121fb5c3a6SDouglas Gregor /// requires-declaration 1213718292f2SDouglas Gregor /// header-declaration 1214e7ab3669SDouglas Gregor /// submodule-declaration 12152b82c2a5SDouglas Gregor /// export-declaration 12166ddfca91SDouglas Gregor /// link-declaration 121773441091SDouglas Gregor /// 121873441091SDouglas Gregor /// submodule-declaration: 121973441091SDouglas Gregor /// module-declaration 122073441091SDouglas Gregor /// inferred-submodule-declaration 1221718292f2SDouglas Gregor void ModuleMapParser::parseModuleDecl() { 1222755b2055SDouglas Gregor assert(Tok.is(MMToken::ExplicitKeyword) || Tok.is(MMToken::ModuleKeyword) || 122397292843SDaniel Jasper Tok.is(MMToken::FrameworkKeyword) || Tok.is(MMToken::ExternKeyword)); 122497292843SDaniel Jasper if (Tok.is(MMToken::ExternKeyword)) { 122597292843SDaniel Jasper parseExternModuleDecl(); 122697292843SDaniel Jasper return; 122797292843SDaniel Jasper } 122897292843SDaniel Jasper 1229f2161a70SDouglas Gregor // Parse 'explicit' or 'framework' keyword, if present. 1230e7ab3669SDouglas Gregor SourceLocation ExplicitLoc; 1231718292f2SDouglas Gregor bool Explicit = false; 1232f2161a70SDouglas Gregor bool Framework = false; 1233755b2055SDouglas Gregor 1234f2161a70SDouglas Gregor // Parse 'explicit' keyword, if present. 1235f2161a70SDouglas Gregor if (Tok.is(MMToken::ExplicitKeyword)) { 1236e7ab3669SDouglas Gregor ExplicitLoc = consumeToken(); 1237f2161a70SDouglas Gregor Explicit = true; 1238f2161a70SDouglas Gregor } 1239f2161a70SDouglas Gregor 1240f2161a70SDouglas Gregor // Parse 'framework' keyword, if present. 1241755b2055SDouglas Gregor if (Tok.is(MMToken::FrameworkKeyword)) { 1242755b2055SDouglas Gregor consumeToken(); 1243755b2055SDouglas Gregor Framework = true; 1244755b2055SDouglas Gregor } 1245718292f2SDouglas Gregor 1246718292f2SDouglas Gregor // Parse 'module' keyword. 1247718292f2SDouglas Gregor if (!Tok.is(MMToken::ModuleKeyword)) { 1248d6343c99SDouglas Gregor Diags.Report(Tok.getLocation(), diag::err_mmap_expected_module); 1249718292f2SDouglas Gregor consumeToken(); 1250718292f2SDouglas Gregor HadError = true; 1251718292f2SDouglas Gregor return; 1252718292f2SDouglas Gregor } 1253718292f2SDouglas Gregor consumeToken(); // 'module' keyword 1254718292f2SDouglas Gregor 125573441091SDouglas Gregor // If we have a wildcard for the module name, this is an inferred submodule. 125673441091SDouglas Gregor // Parse it. 125773441091SDouglas Gregor if (Tok.is(MMToken::Star)) 12589194a91dSDouglas Gregor return parseInferredModuleDecl(Framework, Explicit); 125973441091SDouglas Gregor 1260718292f2SDouglas Gregor // Parse the module name. 1261e7ab3669SDouglas Gregor ModuleId Id; 1262e7ab3669SDouglas Gregor if (parseModuleId(Id)) { 1263718292f2SDouglas Gregor HadError = true; 1264718292f2SDouglas Gregor return; 1265718292f2SDouglas Gregor } 1266e7ab3669SDouglas Gregor 1267e7ab3669SDouglas Gregor if (ActiveModule) { 1268e7ab3669SDouglas Gregor if (Id.size() > 1) { 1269e7ab3669SDouglas Gregor Diags.Report(Id.front().second, diag::err_mmap_nested_submodule_id) 1270e7ab3669SDouglas Gregor << SourceRange(Id.front().second, Id.back().second); 1271e7ab3669SDouglas Gregor 1272e7ab3669SDouglas Gregor HadError = true; 1273e7ab3669SDouglas Gregor return; 1274e7ab3669SDouglas Gregor } 1275e7ab3669SDouglas Gregor } else if (Id.size() == 1 && Explicit) { 1276e7ab3669SDouglas Gregor // Top-level modules can't be explicit. 1277e7ab3669SDouglas Gregor Diags.Report(ExplicitLoc, diag::err_mmap_explicit_top_level); 1278e7ab3669SDouglas Gregor Explicit = false; 1279e7ab3669SDouglas Gregor ExplicitLoc = SourceLocation(); 1280e7ab3669SDouglas Gregor HadError = true; 1281e7ab3669SDouglas Gregor } 1282e7ab3669SDouglas Gregor 1283e7ab3669SDouglas Gregor Module *PreviousActiveModule = ActiveModule; 1284e7ab3669SDouglas Gregor if (Id.size() > 1) { 1285e7ab3669SDouglas Gregor // This module map defines a submodule. Go find the module of which it 1286e7ab3669SDouglas Gregor // is a submodule. 1287e7ab3669SDouglas Gregor ActiveModule = 0; 1288e7ab3669SDouglas Gregor for (unsigned I = 0, N = Id.size() - 1; I != N; ++I) { 1289e7ab3669SDouglas Gregor if (Module *Next = Map.lookupModuleQualified(Id[I].first, ActiveModule)) { 1290e7ab3669SDouglas Gregor ActiveModule = Next; 1291e7ab3669SDouglas Gregor continue; 1292e7ab3669SDouglas Gregor } 1293e7ab3669SDouglas Gregor 1294e7ab3669SDouglas Gregor if (ActiveModule) { 1295e7ab3669SDouglas Gregor Diags.Report(Id[I].second, diag::err_mmap_missing_module_qualified) 12965b5d21eaSRichard Smith << Id[I].first 12975b5d21eaSRichard Smith << ActiveModule->getTopLevelModule()->getFullModuleName(); 1298e7ab3669SDouglas Gregor } else { 1299e7ab3669SDouglas Gregor Diags.Report(Id[I].second, diag::err_mmap_expected_module_name); 1300e7ab3669SDouglas Gregor } 1301e7ab3669SDouglas Gregor HadError = true; 1302e7ab3669SDouglas Gregor return; 1303e7ab3669SDouglas Gregor } 1304e7ab3669SDouglas Gregor } 1305e7ab3669SDouglas Gregor 1306e7ab3669SDouglas Gregor StringRef ModuleName = Id.back().first; 1307e7ab3669SDouglas Gregor SourceLocation ModuleNameLoc = Id.back().second; 1308718292f2SDouglas Gregor 1309a686e1b0SDouglas Gregor // Parse the optional attribute list. 13104442605fSBill Wendling Attributes Attrs; 13119194a91dSDouglas Gregor parseOptionalAttributes(Attrs); 1312a686e1b0SDouglas Gregor 1313718292f2SDouglas Gregor // Parse the opening brace. 1314718292f2SDouglas Gregor if (!Tok.is(MMToken::LBrace)) { 1315718292f2SDouglas Gregor Diags.Report(Tok.getLocation(), diag::err_mmap_expected_lbrace) 1316718292f2SDouglas Gregor << ModuleName; 1317718292f2SDouglas Gregor HadError = true; 1318718292f2SDouglas Gregor return; 1319718292f2SDouglas Gregor } 1320718292f2SDouglas Gregor SourceLocation LBraceLoc = consumeToken(); 1321718292f2SDouglas Gregor 1322718292f2SDouglas Gregor // Determine whether this (sub)module has already been defined. 1323eb90e830SDouglas Gregor if (Module *Existing = Map.lookupModuleQualified(ModuleName, ActiveModule)) { 1324fcc54a3bSDouglas Gregor if (Existing->DefinitionLoc.isInvalid() && !ActiveModule) { 1325fcc54a3bSDouglas Gregor // Skip the module definition. 1326fcc54a3bSDouglas Gregor skipUntil(MMToken::RBrace); 1327fcc54a3bSDouglas Gregor if (Tok.is(MMToken::RBrace)) 1328fcc54a3bSDouglas Gregor consumeToken(); 1329fcc54a3bSDouglas Gregor else { 1330fcc54a3bSDouglas Gregor Diags.Report(Tok.getLocation(), diag::err_mmap_expected_rbrace); 1331fcc54a3bSDouglas Gregor Diags.Report(LBraceLoc, diag::note_mmap_lbrace_match); 1332fcc54a3bSDouglas Gregor HadError = true; 1333fcc54a3bSDouglas Gregor } 1334fcc54a3bSDouglas Gregor return; 1335fcc54a3bSDouglas Gregor } 1336fcc54a3bSDouglas Gregor 1337718292f2SDouglas Gregor Diags.Report(ModuleNameLoc, diag::err_mmap_module_redefinition) 1338718292f2SDouglas Gregor << ModuleName; 1339eb90e830SDouglas Gregor Diags.Report(Existing->DefinitionLoc, diag::note_mmap_prev_definition); 1340718292f2SDouglas Gregor 1341718292f2SDouglas Gregor // Skip the module definition. 1342718292f2SDouglas Gregor skipUntil(MMToken::RBrace); 1343718292f2SDouglas Gregor if (Tok.is(MMToken::RBrace)) 1344718292f2SDouglas Gregor consumeToken(); 1345718292f2SDouglas Gregor 1346718292f2SDouglas Gregor HadError = true; 1347718292f2SDouglas Gregor return; 1348718292f2SDouglas Gregor } 1349718292f2SDouglas Gregor 1350718292f2SDouglas Gregor // Start defining this module. 1351eb90e830SDouglas Gregor ActiveModule = Map.findOrCreateModule(ModuleName, ActiveModule, Framework, 1352eb90e830SDouglas Gregor Explicit).first; 1353eb90e830SDouglas Gregor ActiveModule->DefinitionLoc = ModuleNameLoc; 1354963c5535SDouglas Gregor if (Attrs.IsSystem || IsSystem) 1355a686e1b0SDouglas Gregor ActiveModule->IsSystem = true; 135677944868SRichard Smith if (Attrs.IsExternC) 135777944868SRichard Smith ActiveModule->IsExternC = true; 1358718292f2SDouglas Gregor 1359718292f2SDouglas Gregor bool Done = false; 1360718292f2SDouglas Gregor do { 1361718292f2SDouglas Gregor switch (Tok.Kind) { 1362718292f2SDouglas Gregor case MMToken::EndOfFile: 1363718292f2SDouglas Gregor case MMToken::RBrace: 1364718292f2SDouglas Gregor Done = true; 1365718292f2SDouglas Gregor break; 1366718292f2SDouglas Gregor 136735b13eceSDouglas Gregor case MMToken::ConfigMacros: 136835b13eceSDouglas Gregor parseConfigMacros(); 136935b13eceSDouglas Gregor break; 137035b13eceSDouglas Gregor 1371fb912657SDouglas Gregor case MMToken::Conflict: 1372fb912657SDouglas Gregor parseConflict(); 1373fb912657SDouglas Gregor break; 1374fb912657SDouglas Gregor 1375718292f2SDouglas Gregor case MMToken::ExplicitKeyword: 137697292843SDaniel Jasper case MMToken::ExternKeyword: 1377f2161a70SDouglas Gregor case MMToken::FrameworkKeyword: 1378718292f2SDouglas Gregor case MMToken::ModuleKeyword: 1379718292f2SDouglas Gregor parseModuleDecl(); 1380718292f2SDouglas Gregor break; 1381718292f2SDouglas Gregor 13822b82c2a5SDouglas Gregor case MMToken::ExportKeyword: 13832b82c2a5SDouglas Gregor parseExportDecl(); 13842b82c2a5SDouglas Gregor break; 13852b82c2a5SDouglas Gregor 1386ba7f2f71SDaniel Jasper case MMToken::UseKeyword: 1387ba7f2f71SDaniel Jasper parseUseDecl(); 1388ba7f2f71SDaniel Jasper break; 1389ba7f2f71SDaniel Jasper 13901fb5c3a6SDouglas Gregor case MMToken::RequiresKeyword: 13911fb5c3a6SDouglas Gregor parseRequiresDecl(); 13921fb5c3a6SDouglas Gregor break; 13931fb5c3a6SDouglas Gregor 1394524e33e1SDouglas Gregor case MMToken::UmbrellaKeyword: { 1395524e33e1SDouglas Gregor SourceLocation UmbrellaLoc = consumeToken(); 1396524e33e1SDouglas Gregor if (Tok.is(MMToken::HeaderKeyword)) 1397b53e5483SLawrence Crowl parseHeaderDecl(MMToken::UmbrellaKeyword, UmbrellaLoc); 1398524e33e1SDouglas Gregor else 1399524e33e1SDouglas Gregor parseUmbrellaDirDecl(UmbrellaLoc); 1400718292f2SDouglas Gregor break; 1401524e33e1SDouglas Gregor } 1402718292f2SDouglas Gregor 140359527666SDouglas Gregor case MMToken::ExcludeKeyword: { 140459527666SDouglas Gregor SourceLocation ExcludeLoc = consumeToken(); 140559527666SDouglas Gregor if (Tok.is(MMToken::HeaderKeyword)) { 1406b53e5483SLawrence Crowl parseHeaderDecl(MMToken::ExcludeKeyword, ExcludeLoc); 140759527666SDouglas Gregor } else { 140859527666SDouglas Gregor Diags.Report(Tok.getLocation(), diag::err_mmap_expected_header) 140959527666SDouglas Gregor << "exclude"; 141059527666SDouglas Gregor } 141159527666SDouglas Gregor break; 141259527666SDouglas Gregor } 141359527666SDouglas Gregor 1414b53e5483SLawrence Crowl case MMToken::PrivateKeyword: { 1415b53e5483SLawrence Crowl SourceLocation PrivateLoc = consumeToken(); 1416b53e5483SLawrence Crowl if (Tok.is(MMToken::HeaderKeyword)) { 1417b53e5483SLawrence Crowl parseHeaderDecl(MMToken::PrivateKeyword, PrivateLoc); 1418b53e5483SLawrence Crowl } else { 1419b53e5483SLawrence Crowl Diags.Report(Tok.getLocation(), diag::err_mmap_expected_header) 1420b53e5483SLawrence Crowl << "private"; 1421b53e5483SLawrence Crowl } 1422b53e5483SLawrence Crowl break; 1423b53e5483SLawrence Crowl } 1424b53e5483SLawrence Crowl 1425322f633cSDouglas Gregor case MMToken::HeaderKeyword: 1426b53e5483SLawrence Crowl parseHeaderDecl(MMToken::HeaderKeyword, SourceLocation()); 1427718292f2SDouglas Gregor break; 1428718292f2SDouglas Gregor 14296ddfca91SDouglas Gregor case MMToken::LinkKeyword: 14306ddfca91SDouglas Gregor parseLinkDecl(); 14316ddfca91SDouglas Gregor break; 14326ddfca91SDouglas Gregor 1433718292f2SDouglas Gregor default: 1434718292f2SDouglas Gregor Diags.Report(Tok.getLocation(), diag::err_mmap_expected_member); 1435718292f2SDouglas Gregor consumeToken(); 1436718292f2SDouglas Gregor break; 1437718292f2SDouglas Gregor } 1438718292f2SDouglas Gregor } while (!Done); 1439718292f2SDouglas Gregor 1440718292f2SDouglas Gregor if (Tok.is(MMToken::RBrace)) 1441718292f2SDouglas Gregor consumeToken(); 1442718292f2SDouglas Gregor else { 1443718292f2SDouglas Gregor Diags.Report(Tok.getLocation(), diag::err_mmap_expected_rbrace); 1444718292f2SDouglas Gregor Diags.Report(LBraceLoc, diag::note_mmap_lbrace_match); 1445718292f2SDouglas Gregor HadError = true; 1446718292f2SDouglas Gregor } 1447718292f2SDouglas Gregor 144811dfe6feSDouglas Gregor // If the active module is a top-level framework, and there are no link 144911dfe6feSDouglas Gregor // libraries, automatically link against the framework. 145011dfe6feSDouglas Gregor if (ActiveModule->IsFramework && !ActiveModule->isSubFramework() && 145111dfe6feSDouglas Gregor ActiveModule->LinkLibraries.empty()) { 145211dfe6feSDouglas Gregor inferFrameworkLink(ActiveModule, Directory, SourceMgr.getFileManager()); 145311dfe6feSDouglas Gregor } 145411dfe6feSDouglas Gregor 1455e7ab3669SDouglas Gregor // We're done parsing this module. Pop back to the previous module. 1456e7ab3669SDouglas Gregor ActiveModule = PreviousActiveModule; 1457718292f2SDouglas Gregor } 1458718292f2SDouglas Gregor 145997292843SDaniel Jasper /// \brief Parse an extern module declaration. 146097292843SDaniel Jasper /// 146197292843SDaniel Jasper /// extern module-declaration: 146297292843SDaniel Jasper /// 'extern' 'module' module-id string-literal 146397292843SDaniel Jasper void ModuleMapParser::parseExternModuleDecl() { 146497292843SDaniel Jasper assert(Tok.is(MMToken::ExternKeyword)); 146597292843SDaniel Jasper consumeToken(); // 'extern' keyword 146697292843SDaniel Jasper 146797292843SDaniel Jasper // Parse 'module' keyword. 146897292843SDaniel Jasper if (!Tok.is(MMToken::ModuleKeyword)) { 146997292843SDaniel Jasper Diags.Report(Tok.getLocation(), diag::err_mmap_expected_module); 147097292843SDaniel Jasper consumeToken(); 147197292843SDaniel Jasper HadError = true; 147297292843SDaniel Jasper return; 147397292843SDaniel Jasper } 147497292843SDaniel Jasper consumeToken(); // 'module' keyword 147597292843SDaniel Jasper 147697292843SDaniel Jasper // Parse the module name. 147797292843SDaniel Jasper ModuleId Id; 147897292843SDaniel Jasper if (parseModuleId(Id)) { 147997292843SDaniel Jasper HadError = true; 148097292843SDaniel Jasper return; 148197292843SDaniel Jasper } 148297292843SDaniel Jasper 148397292843SDaniel Jasper // Parse the referenced module map file name. 148497292843SDaniel Jasper if (!Tok.is(MMToken::StringLiteral)) { 148597292843SDaniel Jasper Diags.Report(Tok.getLocation(), diag::err_mmap_expected_mmap_file); 148697292843SDaniel Jasper HadError = true; 148797292843SDaniel Jasper return; 148897292843SDaniel Jasper } 148997292843SDaniel Jasper std::string FileName = Tok.getString(); 149097292843SDaniel Jasper consumeToken(); // filename 149197292843SDaniel Jasper 149297292843SDaniel Jasper StringRef FileNameRef = FileName; 149397292843SDaniel Jasper SmallString<128> ModuleMapFileName; 149497292843SDaniel Jasper if (llvm::sys::path::is_relative(FileNameRef)) { 149597292843SDaniel Jasper ModuleMapFileName += Directory->getName(); 149697292843SDaniel Jasper llvm::sys::path::append(ModuleMapFileName, FileName); 149797292843SDaniel Jasper FileNameRef = ModuleMapFileName.str(); 149897292843SDaniel Jasper } 149997292843SDaniel Jasper if (const FileEntry *File = SourceMgr.getFileManager().getFile(FileNameRef)) 150097292843SDaniel Jasper Map.parseModuleMapFile(File, /*IsSystem=*/false); 150197292843SDaniel Jasper } 150297292843SDaniel Jasper 15031fb5c3a6SDouglas Gregor /// \brief Parse a requires declaration. 15041fb5c3a6SDouglas Gregor /// 15051fb5c3a6SDouglas Gregor /// requires-declaration: 15061fb5c3a6SDouglas Gregor /// 'requires' feature-list 15071fb5c3a6SDouglas Gregor /// 15081fb5c3a6SDouglas Gregor /// feature-list: 1509a3feee2aSRichard Smith /// feature ',' feature-list 1510a3feee2aSRichard Smith /// feature 1511a3feee2aSRichard Smith /// 1512a3feee2aSRichard Smith /// feature: 1513a3feee2aSRichard Smith /// '!'[opt] identifier 15141fb5c3a6SDouglas Gregor void ModuleMapParser::parseRequiresDecl() { 15151fb5c3a6SDouglas Gregor assert(Tok.is(MMToken::RequiresKeyword)); 15161fb5c3a6SDouglas Gregor 15171fb5c3a6SDouglas Gregor // Parse 'requires' keyword. 15181fb5c3a6SDouglas Gregor consumeToken(); 15191fb5c3a6SDouglas Gregor 15201fb5c3a6SDouglas Gregor // Parse the feature-list. 15211fb5c3a6SDouglas Gregor do { 1522a3feee2aSRichard Smith bool RequiredState = true; 1523a3feee2aSRichard Smith if (Tok.is(MMToken::Exclaim)) { 1524a3feee2aSRichard Smith RequiredState = false; 1525a3feee2aSRichard Smith consumeToken(); 1526a3feee2aSRichard Smith } 1527a3feee2aSRichard Smith 15281fb5c3a6SDouglas Gregor if (!Tok.is(MMToken::Identifier)) { 15291fb5c3a6SDouglas Gregor Diags.Report(Tok.getLocation(), diag::err_mmap_expected_feature); 15301fb5c3a6SDouglas Gregor HadError = true; 15311fb5c3a6SDouglas Gregor return; 15321fb5c3a6SDouglas Gregor } 15331fb5c3a6SDouglas Gregor 15341fb5c3a6SDouglas Gregor // Consume the feature name. 15351fb5c3a6SDouglas Gregor std::string Feature = Tok.getString(); 15361fb5c3a6SDouglas Gregor consumeToken(); 15371fb5c3a6SDouglas Gregor 15381fb5c3a6SDouglas Gregor // Add this feature. 1539a3feee2aSRichard Smith ActiveModule->addRequirement(Feature, RequiredState, 1540a3feee2aSRichard Smith Map.LangOpts, *Map.Target); 15411fb5c3a6SDouglas Gregor 15421fb5c3a6SDouglas Gregor if (!Tok.is(MMToken::Comma)) 15431fb5c3a6SDouglas Gregor break; 15441fb5c3a6SDouglas Gregor 15451fb5c3a6SDouglas Gregor // Consume the comma. 15461fb5c3a6SDouglas Gregor consumeToken(); 15471fb5c3a6SDouglas Gregor } while (true); 15481fb5c3a6SDouglas Gregor } 15491fb5c3a6SDouglas Gregor 1550f2161a70SDouglas Gregor /// \brief Append to \p Paths the set of paths needed to get to the 1551f2161a70SDouglas Gregor /// subframework in which the given module lives. 1552bf8da9d7SBenjamin Kramer static void appendSubframeworkPaths(Module *Mod, 1553f857950dSDmitri Gribenko SmallVectorImpl<char> &Path) { 1554f2161a70SDouglas Gregor // Collect the framework names from the given module to the top-level module. 1555f857950dSDmitri Gribenko SmallVector<StringRef, 2> Paths; 1556f2161a70SDouglas Gregor for (; Mod; Mod = Mod->Parent) { 1557f2161a70SDouglas Gregor if (Mod->IsFramework) 1558f2161a70SDouglas Gregor Paths.push_back(Mod->Name); 1559f2161a70SDouglas Gregor } 1560f2161a70SDouglas Gregor 1561f2161a70SDouglas Gregor if (Paths.empty()) 1562f2161a70SDouglas Gregor return; 1563f2161a70SDouglas Gregor 1564f2161a70SDouglas Gregor // Add Frameworks/Name.framework for each subframework. 156517381a06SBenjamin Kramer for (unsigned I = Paths.size() - 1; I != 0; --I) 156617381a06SBenjamin Kramer llvm::sys::path::append(Path, "Frameworks", Paths[I-1] + ".framework"); 1567f2161a70SDouglas Gregor } 1568f2161a70SDouglas Gregor 1569718292f2SDouglas Gregor /// \brief Parse a header declaration. 1570718292f2SDouglas Gregor /// 1571718292f2SDouglas Gregor /// header-declaration: 1572322f633cSDouglas Gregor /// 'umbrella'[opt] 'header' string-literal 157359527666SDouglas Gregor /// 'exclude'[opt] 'header' string-literal 1574b53e5483SLawrence Crowl void ModuleMapParser::parseHeaderDecl(MMToken::TokenKind LeadingToken, 1575b53e5483SLawrence Crowl SourceLocation LeadingLoc) { 1576718292f2SDouglas Gregor assert(Tok.is(MMToken::HeaderKeyword)); 15771871ed3dSBenjamin Kramer consumeToken(); 1578718292f2SDouglas Gregor 1579718292f2SDouglas Gregor // Parse the header name. 1580718292f2SDouglas Gregor if (!Tok.is(MMToken::StringLiteral)) { 1581718292f2SDouglas Gregor Diags.Report(Tok.getLocation(), diag::err_mmap_expected_header) 1582718292f2SDouglas Gregor << "header"; 1583718292f2SDouglas Gregor HadError = true; 1584718292f2SDouglas Gregor return; 1585718292f2SDouglas Gregor } 15860761a8a0SDaniel Jasper Module::HeaderDirective Header; 15870761a8a0SDaniel Jasper Header.FileName = Tok.getString(); 15880761a8a0SDaniel Jasper Header.FileNameLoc = consumeToken(); 1589718292f2SDouglas Gregor 1590524e33e1SDouglas Gregor // Check whether we already have an umbrella. 1591b53e5483SLawrence Crowl if (LeadingToken == MMToken::UmbrellaKeyword && ActiveModule->Umbrella) { 15920761a8a0SDaniel Jasper Diags.Report(Header.FileNameLoc, diag::err_mmap_umbrella_clash) 1593524e33e1SDouglas Gregor << ActiveModule->getFullModuleName(); 1594322f633cSDouglas Gregor HadError = true; 1595322f633cSDouglas Gregor return; 1596322f633cSDouglas Gregor } 1597322f633cSDouglas Gregor 15985257fc63SDouglas Gregor // Look for this file. 1599e7ab3669SDouglas Gregor const FileEntry *File = 0; 16003ec6663bSDouglas Gregor const FileEntry *BuiltinFile = 0; 16012c1dd271SDylan Noblesmith SmallString<128> PathName; 16020761a8a0SDaniel Jasper if (llvm::sys::path::is_absolute(Header.FileName)) { 16030761a8a0SDaniel Jasper PathName = Header.FileName; 1604e7ab3669SDouglas Gregor File = SourceMgr.getFileManager().getFile(PathName); 16057033127bSDouglas Gregor } else if (const DirectoryEntry *Dir = getOverriddenHeaderSearchDir()) { 16067033127bSDouglas Gregor PathName = Dir->getName(); 16070761a8a0SDaniel Jasper llvm::sys::path::append(PathName, Header.FileName); 16087033127bSDouglas Gregor File = SourceMgr.getFileManager().getFile(PathName); 1609e7ab3669SDouglas Gregor } else { 1610e7ab3669SDouglas Gregor // Search for the header file within the search directory. 16117033127bSDouglas Gregor PathName = Directory->getName(); 1612e7ab3669SDouglas Gregor unsigned PathLength = PathName.size(); 1613755b2055SDouglas Gregor 1614f2161a70SDouglas Gregor if (ActiveModule->isPartOfFramework()) { 1615f2161a70SDouglas Gregor appendSubframeworkPaths(ActiveModule, PathName); 1616755b2055SDouglas Gregor 1617e7ab3669SDouglas Gregor // Check whether this file is in the public headers. 16180761a8a0SDaniel Jasper llvm::sys::path::append(PathName, "Headers", Header.FileName); 1619e7ab3669SDouglas Gregor File = SourceMgr.getFileManager().getFile(PathName); 1620e7ab3669SDouglas Gregor 1621e7ab3669SDouglas Gregor if (!File) { 1622e7ab3669SDouglas Gregor // Check whether this file is in the private headers. 1623e7ab3669SDouglas Gregor PathName.resize(PathLength); 16240761a8a0SDaniel Jasper llvm::sys::path::append(PathName, "PrivateHeaders", Header.FileName); 1625e7ab3669SDouglas Gregor File = SourceMgr.getFileManager().getFile(PathName); 1626e7ab3669SDouglas Gregor } 1627e7ab3669SDouglas Gregor } else { 1628e7ab3669SDouglas Gregor // Lookup for normal headers. 16290761a8a0SDaniel Jasper llvm::sys::path::append(PathName, Header.FileName); 1630e7ab3669SDouglas Gregor File = SourceMgr.getFileManager().getFile(PathName); 16313ec6663bSDouglas Gregor 16323ec6663bSDouglas Gregor // If this is a system module with a top-level header, this header 16333ec6663bSDouglas Gregor // may have a counterpart (or replacement) in the set of headers 16343ec6663bSDouglas Gregor // supplied by Clang. Find that builtin header. 1635b53e5483SLawrence Crowl if (ActiveModule->IsSystem && LeadingToken != MMToken::UmbrellaKeyword && 1636b53e5483SLawrence Crowl BuiltinIncludeDir && BuiltinIncludeDir != Directory && 16370761a8a0SDaniel Jasper isBuiltinHeader(Header.FileName)) { 16382c1dd271SDylan Noblesmith SmallString<128> BuiltinPathName(BuiltinIncludeDir->getName()); 16390761a8a0SDaniel Jasper llvm::sys::path::append(BuiltinPathName, Header.FileName); 16403ec6663bSDouglas Gregor BuiltinFile = SourceMgr.getFileManager().getFile(BuiltinPathName); 16413ec6663bSDouglas Gregor 16423ec6663bSDouglas Gregor // If Clang supplies this header but the underlying system does not, 16433ec6663bSDouglas Gregor // just silently swap in our builtin version. Otherwise, we'll end 16443ec6663bSDouglas Gregor // up adding both (later). 16453ec6663bSDouglas Gregor if (!File && BuiltinFile) { 16463ec6663bSDouglas Gregor File = BuiltinFile; 16473ec6663bSDouglas Gregor BuiltinFile = 0; 16483ec6663bSDouglas Gregor } 16493ec6663bSDouglas Gregor } 1650e7ab3669SDouglas Gregor } 1651e7ab3669SDouglas Gregor } 16525257fc63SDouglas Gregor 16535257fc63SDouglas Gregor // FIXME: We shouldn't be eagerly stat'ing every file named in a module map. 16545257fc63SDouglas Gregor // Come up with a lazy way to do this. 1655e7ab3669SDouglas Gregor if (File) { 165697da9178SDaniel Jasper if (LeadingToken == MMToken::UmbrellaKeyword) { 1657322f633cSDouglas Gregor const DirectoryEntry *UmbrellaDir = File->getDir(); 165859527666SDouglas Gregor if (Module *UmbrellaModule = Map.UmbrellaDirs[UmbrellaDir]) { 1659b53e5483SLawrence Crowl Diags.Report(LeadingLoc, diag::err_mmap_umbrella_clash) 166059527666SDouglas Gregor << UmbrellaModule->getFullModuleName(); 1661322f633cSDouglas Gregor HadError = true; 16625257fc63SDouglas Gregor } else { 1663322f633cSDouglas Gregor // Record this umbrella header. 1664322f633cSDouglas Gregor Map.setUmbrellaHeader(ActiveModule, File); 1665322f633cSDouglas Gregor } 1666322f633cSDouglas Gregor } else { 1667322f633cSDouglas Gregor // Record this header. 1668b53e5483SLawrence Crowl ModuleMap::ModuleHeaderRole Role = ModuleMap::NormalHeader; 1669b53e5483SLawrence Crowl if (LeadingToken == MMToken::ExcludeKeyword) 1670b53e5483SLawrence Crowl Role = ModuleMap::ExcludedHeader; 1671b53e5483SLawrence Crowl else if (LeadingToken == MMToken::PrivateKeyword) 1672b53e5483SLawrence Crowl Role = ModuleMap::PrivateHeader; 1673b53e5483SLawrence Crowl else 1674b53e5483SLawrence Crowl assert(LeadingToken == MMToken::HeaderKeyword); 1675b53e5483SLawrence Crowl 1676b53e5483SLawrence Crowl Map.addHeader(ActiveModule, File, Role); 16773ec6663bSDouglas Gregor 16783ec6663bSDouglas Gregor // If there is a builtin counterpart to this file, add it now. 16793ec6663bSDouglas Gregor if (BuiltinFile) 1680b53e5483SLawrence Crowl Map.addHeader(ActiveModule, BuiltinFile, Role); 16815257fc63SDouglas Gregor } 1682b53e5483SLawrence Crowl } else if (LeadingToken != MMToken::ExcludeKeyword) { 16834b27a64bSDouglas Gregor // Ignore excluded header files. They're optional anyway. 16844b27a64bSDouglas Gregor 16850761a8a0SDaniel Jasper // If we find a module that has a missing header, we mark this module as 16860761a8a0SDaniel Jasper // unavailable and store the header directive for displaying diagnostics. 16870761a8a0SDaniel Jasper // Other submodules in the same module can still be used. 16880761a8a0SDaniel Jasper Header.IsUmbrella = LeadingToken == MMToken::UmbrellaKeyword; 16890761a8a0SDaniel Jasper ActiveModule->IsAvailable = false; 16900761a8a0SDaniel Jasper ActiveModule->MissingHeaders.push_back(Header); 16915257fc63SDouglas Gregor } 1692718292f2SDouglas Gregor } 1693718292f2SDouglas Gregor 1694524e33e1SDouglas Gregor /// \brief Parse an umbrella directory declaration. 1695524e33e1SDouglas Gregor /// 1696524e33e1SDouglas Gregor /// umbrella-dir-declaration: 1697524e33e1SDouglas Gregor /// umbrella string-literal 1698524e33e1SDouglas Gregor void ModuleMapParser::parseUmbrellaDirDecl(SourceLocation UmbrellaLoc) { 1699524e33e1SDouglas Gregor // Parse the directory name. 1700524e33e1SDouglas Gregor if (!Tok.is(MMToken::StringLiteral)) { 1701524e33e1SDouglas Gregor Diags.Report(Tok.getLocation(), diag::err_mmap_expected_header) 1702524e33e1SDouglas Gregor << "umbrella"; 1703524e33e1SDouglas Gregor HadError = true; 1704524e33e1SDouglas Gregor return; 1705524e33e1SDouglas Gregor } 1706524e33e1SDouglas Gregor 1707524e33e1SDouglas Gregor std::string DirName = Tok.getString(); 1708524e33e1SDouglas Gregor SourceLocation DirNameLoc = consumeToken(); 1709524e33e1SDouglas Gregor 1710524e33e1SDouglas Gregor // Check whether we already have an umbrella. 1711524e33e1SDouglas Gregor if (ActiveModule->Umbrella) { 1712524e33e1SDouglas Gregor Diags.Report(DirNameLoc, diag::err_mmap_umbrella_clash) 1713524e33e1SDouglas Gregor << ActiveModule->getFullModuleName(); 1714524e33e1SDouglas Gregor HadError = true; 1715524e33e1SDouglas Gregor return; 1716524e33e1SDouglas Gregor } 1717524e33e1SDouglas Gregor 1718524e33e1SDouglas Gregor // Look for this file. 1719524e33e1SDouglas Gregor const DirectoryEntry *Dir = 0; 1720524e33e1SDouglas Gregor if (llvm::sys::path::is_absolute(DirName)) 1721524e33e1SDouglas Gregor Dir = SourceMgr.getFileManager().getDirectory(DirName); 1722524e33e1SDouglas Gregor else { 17232c1dd271SDylan Noblesmith SmallString<128> PathName; 1724524e33e1SDouglas Gregor PathName = Directory->getName(); 1725524e33e1SDouglas Gregor llvm::sys::path::append(PathName, DirName); 1726524e33e1SDouglas Gregor Dir = SourceMgr.getFileManager().getDirectory(PathName); 1727524e33e1SDouglas Gregor } 1728524e33e1SDouglas Gregor 1729524e33e1SDouglas Gregor if (!Dir) { 1730524e33e1SDouglas Gregor Diags.Report(DirNameLoc, diag::err_mmap_umbrella_dir_not_found) 1731524e33e1SDouglas Gregor << DirName; 1732524e33e1SDouglas Gregor HadError = true; 1733524e33e1SDouglas Gregor return; 1734524e33e1SDouglas Gregor } 1735524e33e1SDouglas Gregor 1736524e33e1SDouglas Gregor if (Module *OwningModule = Map.UmbrellaDirs[Dir]) { 1737524e33e1SDouglas Gregor Diags.Report(UmbrellaLoc, diag::err_mmap_umbrella_clash) 1738524e33e1SDouglas Gregor << OwningModule->getFullModuleName(); 1739524e33e1SDouglas Gregor HadError = true; 1740524e33e1SDouglas Gregor return; 1741524e33e1SDouglas Gregor } 1742524e33e1SDouglas Gregor 1743524e33e1SDouglas Gregor // Record this umbrella directory. 1744524e33e1SDouglas Gregor Map.setUmbrellaDir(ActiveModule, Dir); 1745524e33e1SDouglas Gregor } 1746524e33e1SDouglas Gregor 17472b82c2a5SDouglas Gregor /// \brief Parse a module export declaration. 17482b82c2a5SDouglas Gregor /// 17492b82c2a5SDouglas Gregor /// export-declaration: 17502b82c2a5SDouglas Gregor /// 'export' wildcard-module-id 17512b82c2a5SDouglas Gregor /// 17522b82c2a5SDouglas Gregor /// wildcard-module-id: 17532b82c2a5SDouglas Gregor /// identifier 17542b82c2a5SDouglas Gregor /// '*' 17552b82c2a5SDouglas Gregor /// identifier '.' wildcard-module-id 17562b82c2a5SDouglas Gregor void ModuleMapParser::parseExportDecl() { 17572b82c2a5SDouglas Gregor assert(Tok.is(MMToken::ExportKeyword)); 17582b82c2a5SDouglas Gregor SourceLocation ExportLoc = consumeToken(); 17592b82c2a5SDouglas Gregor 17602b82c2a5SDouglas Gregor // Parse the module-id with an optional wildcard at the end. 17612b82c2a5SDouglas Gregor ModuleId ParsedModuleId; 17622b82c2a5SDouglas Gregor bool Wildcard = false; 17632b82c2a5SDouglas Gregor do { 17642b82c2a5SDouglas Gregor if (Tok.is(MMToken::Identifier)) { 17652b82c2a5SDouglas Gregor ParsedModuleId.push_back(std::make_pair(Tok.getString(), 17662b82c2a5SDouglas Gregor Tok.getLocation())); 17672b82c2a5SDouglas Gregor consumeToken(); 17682b82c2a5SDouglas Gregor 17692b82c2a5SDouglas Gregor if (Tok.is(MMToken::Period)) { 17702b82c2a5SDouglas Gregor consumeToken(); 17712b82c2a5SDouglas Gregor continue; 17722b82c2a5SDouglas Gregor } 17732b82c2a5SDouglas Gregor 17742b82c2a5SDouglas Gregor break; 17752b82c2a5SDouglas Gregor } 17762b82c2a5SDouglas Gregor 17772b82c2a5SDouglas Gregor if(Tok.is(MMToken::Star)) { 17782b82c2a5SDouglas Gregor Wildcard = true; 1779f5eedd05SDouglas Gregor consumeToken(); 17802b82c2a5SDouglas Gregor break; 17812b82c2a5SDouglas Gregor } 17822b82c2a5SDouglas Gregor 1783ba7f2f71SDaniel Jasper Diags.Report(Tok.getLocation(), diag::err_mmap_module_id); 17842b82c2a5SDouglas Gregor HadError = true; 17852b82c2a5SDouglas Gregor return; 17862b82c2a5SDouglas Gregor } while (true); 17872b82c2a5SDouglas Gregor 17882b82c2a5SDouglas Gregor Module::UnresolvedExportDecl Unresolved = { 17892b82c2a5SDouglas Gregor ExportLoc, ParsedModuleId, Wildcard 17902b82c2a5SDouglas Gregor }; 17912b82c2a5SDouglas Gregor ActiveModule->UnresolvedExports.push_back(Unresolved); 17922b82c2a5SDouglas Gregor } 17932b82c2a5SDouglas Gregor 1794ba7f2f71SDaniel Jasper /// \brief Parse a module uses declaration. 1795ba7f2f71SDaniel Jasper /// 1796ba7f2f71SDaniel Jasper /// uses-declaration: 1797ba7f2f71SDaniel Jasper /// 'uses' wildcard-module-id 1798ba7f2f71SDaniel Jasper void ModuleMapParser::parseUseDecl() { 1799ba7f2f71SDaniel Jasper assert(Tok.is(MMToken::UseKeyword)); 1800ba7f2f71SDaniel Jasper consumeToken(); 1801ba7f2f71SDaniel Jasper // Parse the module-id. 1802ba7f2f71SDaniel Jasper ModuleId ParsedModuleId; 18033cd34c76SDaniel Jasper parseModuleId(ParsedModuleId); 1804ba7f2f71SDaniel Jasper 1805ba7f2f71SDaniel Jasper ActiveModule->UnresolvedDirectUses.push_back(ParsedModuleId); 1806ba7f2f71SDaniel Jasper } 1807ba7f2f71SDaniel Jasper 18086ddfca91SDouglas Gregor /// \brief Parse a link declaration. 18096ddfca91SDouglas Gregor /// 18106ddfca91SDouglas Gregor /// module-declaration: 18116ddfca91SDouglas Gregor /// 'link' 'framework'[opt] string-literal 18126ddfca91SDouglas Gregor void ModuleMapParser::parseLinkDecl() { 18136ddfca91SDouglas Gregor assert(Tok.is(MMToken::LinkKeyword)); 18146ddfca91SDouglas Gregor SourceLocation LinkLoc = consumeToken(); 18156ddfca91SDouglas Gregor 18166ddfca91SDouglas Gregor // Parse the optional 'framework' keyword. 18176ddfca91SDouglas Gregor bool IsFramework = false; 18186ddfca91SDouglas Gregor if (Tok.is(MMToken::FrameworkKeyword)) { 18196ddfca91SDouglas Gregor consumeToken(); 18206ddfca91SDouglas Gregor IsFramework = true; 18216ddfca91SDouglas Gregor } 18226ddfca91SDouglas Gregor 18236ddfca91SDouglas Gregor // Parse the library name 18246ddfca91SDouglas Gregor if (!Tok.is(MMToken::StringLiteral)) { 18256ddfca91SDouglas Gregor Diags.Report(Tok.getLocation(), diag::err_mmap_expected_library_name) 18266ddfca91SDouglas Gregor << IsFramework << SourceRange(LinkLoc); 18276ddfca91SDouglas Gregor HadError = true; 18286ddfca91SDouglas Gregor return; 18296ddfca91SDouglas Gregor } 18306ddfca91SDouglas Gregor 18316ddfca91SDouglas Gregor std::string LibraryName = Tok.getString(); 18326ddfca91SDouglas Gregor consumeToken(); 18336ddfca91SDouglas Gregor ActiveModule->LinkLibraries.push_back(Module::LinkLibrary(LibraryName, 18346ddfca91SDouglas Gregor IsFramework)); 18356ddfca91SDouglas Gregor } 18366ddfca91SDouglas Gregor 183735b13eceSDouglas Gregor /// \brief Parse a configuration macro declaration. 183835b13eceSDouglas Gregor /// 183935b13eceSDouglas Gregor /// module-declaration: 184035b13eceSDouglas Gregor /// 'config_macros' attributes[opt] config-macro-list? 184135b13eceSDouglas Gregor /// 184235b13eceSDouglas Gregor /// config-macro-list: 184335b13eceSDouglas Gregor /// identifier (',' identifier)? 184435b13eceSDouglas Gregor void ModuleMapParser::parseConfigMacros() { 184535b13eceSDouglas Gregor assert(Tok.is(MMToken::ConfigMacros)); 184635b13eceSDouglas Gregor SourceLocation ConfigMacrosLoc = consumeToken(); 184735b13eceSDouglas Gregor 184835b13eceSDouglas Gregor // Only top-level modules can have configuration macros. 184935b13eceSDouglas Gregor if (ActiveModule->Parent) { 185035b13eceSDouglas Gregor Diags.Report(ConfigMacrosLoc, diag::err_mmap_config_macro_submodule); 185135b13eceSDouglas Gregor } 185235b13eceSDouglas Gregor 185335b13eceSDouglas Gregor // Parse the optional attributes. 185435b13eceSDouglas Gregor Attributes Attrs; 185535b13eceSDouglas Gregor parseOptionalAttributes(Attrs); 185635b13eceSDouglas Gregor if (Attrs.IsExhaustive && !ActiveModule->Parent) { 185735b13eceSDouglas Gregor ActiveModule->ConfigMacrosExhaustive = true; 185835b13eceSDouglas Gregor } 185935b13eceSDouglas Gregor 186035b13eceSDouglas Gregor // If we don't have an identifier, we're done. 186135b13eceSDouglas Gregor if (!Tok.is(MMToken::Identifier)) 186235b13eceSDouglas Gregor return; 186335b13eceSDouglas Gregor 186435b13eceSDouglas Gregor // Consume the first identifier. 186535b13eceSDouglas Gregor if (!ActiveModule->Parent) { 186635b13eceSDouglas Gregor ActiveModule->ConfigMacros.push_back(Tok.getString().str()); 186735b13eceSDouglas Gregor } 186835b13eceSDouglas Gregor consumeToken(); 186935b13eceSDouglas Gregor 187035b13eceSDouglas Gregor do { 187135b13eceSDouglas Gregor // If there's a comma, consume it. 187235b13eceSDouglas Gregor if (!Tok.is(MMToken::Comma)) 187335b13eceSDouglas Gregor break; 187435b13eceSDouglas Gregor consumeToken(); 187535b13eceSDouglas Gregor 187635b13eceSDouglas Gregor // We expect to see a macro name here. 187735b13eceSDouglas Gregor if (!Tok.is(MMToken::Identifier)) { 187835b13eceSDouglas Gregor Diags.Report(Tok.getLocation(), diag::err_mmap_expected_config_macro); 187935b13eceSDouglas Gregor break; 188035b13eceSDouglas Gregor } 188135b13eceSDouglas Gregor 188235b13eceSDouglas Gregor // Consume the macro name. 188335b13eceSDouglas Gregor if (!ActiveModule->Parent) { 188435b13eceSDouglas Gregor ActiveModule->ConfigMacros.push_back(Tok.getString().str()); 188535b13eceSDouglas Gregor } 188635b13eceSDouglas Gregor consumeToken(); 188735b13eceSDouglas Gregor } while (true); 188835b13eceSDouglas Gregor } 188935b13eceSDouglas Gregor 1890fb912657SDouglas Gregor /// \brief Format a module-id into a string. 1891fb912657SDouglas Gregor static std::string formatModuleId(const ModuleId &Id) { 1892fb912657SDouglas Gregor std::string result; 1893fb912657SDouglas Gregor { 1894fb912657SDouglas Gregor llvm::raw_string_ostream OS(result); 1895fb912657SDouglas Gregor 1896fb912657SDouglas Gregor for (unsigned I = 0, N = Id.size(); I != N; ++I) { 1897fb912657SDouglas Gregor if (I) 1898fb912657SDouglas Gregor OS << "."; 1899fb912657SDouglas Gregor OS << Id[I].first; 1900fb912657SDouglas Gregor } 1901fb912657SDouglas Gregor } 1902fb912657SDouglas Gregor 1903fb912657SDouglas Gregor return result; 1904fb912657SDouglas Gregor } 1905fb912657SDouglas Gregor 1906fb912657SDouglas Gregor /// \brief Parse a conflict declaration. 1907fb912657SDouglas Gregor /// 1908fb912657SDouglas Gregor /// module-declaration: 1909fb912657SDouglas Gregor /// 'conflict' module-id ',' string-literal 1910fb912657SDouglas Gregor void ModuleMapParser::parseConflict() { 1911fb912657SDouglas Gregor assert(Tok.is(MMToken::Conflict)); 1912fb912657SDouglas Gregor SourceLocation ConflictLoc = consumeToken(); 1913fb912657SDouglas Gregor Module::UnresolvedConflict Conflict; 1914fb912657SDouglas Gregor 1915fb912657SDouglas Gregor // Parse the module-id. 1916fb912657SDouglas Gregor if (parseModuleId(Conflict.Id)) 1917fb912657SDouglas Gregor return; 1918fb912657SDouglas Gregor 1919fb912657SDouglas Gregor // Parse the ','. 1920fb912657SDouglas Gregor if (!Tok.is(MMToken::Comma)) { 1921fb912657SDouglas Gregor Diags.Report(Tok.getLocation(), diag::err_mmap_expected_conflicts_comma) 1922fb912657SDouglas Gregor << SourceRange(ConflictLoc); 1923fb912657SDouglas Gregor return; 1924fb912657SDouglas Gregor } 1925fb912657SDouglas Gregor consumeToken(); 1926fb912657SDouglas Gregor 1927fb912657SDouglas Gregor // Parse the message. 1928fb912657SDouglas Gregor if (!Tok.is(MMToken::StringLiteral)) { 1929fb912657SDouglas Gregor Diags.Report(Tok.getLocation(), diag::err_mmap_expected_conflicts_message) 1930fb912657SDouglas Gregor << formatModuleId(Conflict.Id); 1931fb912657SDouglas Gregor return; 1932fb912657SDouglas Gregor } 1933fb912657SDouglas Gregor Conflict.Message = Tok.getString().str(); 1934fb912657SDouglas Gregor consumeToken(); 1935fb912657SDouglas Gregor 1936fb912657SDouglas Gregor // Add this unresolved conflict. 1937fb912657SDouglas Gregor ActiveModule->UnresolvedConflicts.push_back(Conflict); 1938fb912657SDouglas Gregor } 1939fb912657SDouglas Gregor 19406ddfca91SDouglas Gregor /// \brief Parse an inferred module declaration (wildcard modules). 19419194a91dSDouglas Gregor /// 19429194a91dSDouglas Gregor /// module-declaration: 19439194a91dSDouglas Gregor /// 'explicit'[opt] 'framework'[opt] 'module' * attributes[opt] 19449194a91dSDouglas Gregor /// { inferred-module-member* } 19459194a91dSDouglas Gregor /// 19469194a91dSDouglas Gregor /// inferred-module-member: 19479194a91dSDouglas Gregor /// 'export' '*' 19489194a91dSDouglas Gregor /// 'exclude' identifier 19499194a91dSDouglas Gregor void ModuleMapParser::parseInferredModuleDecl(bool Framework, bool Explicit) { 195073441091SDouglas Gregor assert(Tok.is(MMToken::Star)); 195173441091SDouglas Gregor SourceLocation StarLoc = consumeToken(); 195273441091SDouglas Gregor bool Failed = false; 195373441091SDouglas Gregor 195473441091SDouglas Gregor // Inferred modules must be submodules. 19559194a91dSDouglas Gregor if (!ActiveModule && !Framework) { 195673441091SDouglas Gregor Diags.Report(StarLoc, diag::err_mmap_top_level_inferred_submodule); 195773441091SDouglas Gregor Failed = true; 195873441091SDouglas Gregor } 195973441091SDouglas Gregor 19609194a91dSDouglas Gregor if (ActiveModule) { 1961524e33e1SDouglas Gregor // Inferred modules must have umbrella directories. 1962524e33e1SDouglas Gregor if (!Failed && !ActiveModule->getUmbrellaDir()) { 196373441091SDouglas Gregor Diags.Report(StarLoc, diag::err_mmap_inferred_no_umbrella); 196473441091SDouglas Gregor Failed = true; 196573441091SDouglas Gregor } 196673441091SDouglas Gregor 196773441091SDouglas Gregor // Check for redefinition of an inferred module. 1968dd005f69SDouglas Gregor if (!Failed && ActiveModule->InferSubmodules) { 196973441091SDouglas Gregor Diags.Report(StarLoc, diag::err_mmap_inferred_redef); 1970dd005f69SDouglas Gregor if (ActiveModule->InferredSubmoduleLoc.isValid()) 1971dd005f69SDouglas Gregor Diags.Report(ActiveModule->InferredSubmoduleLoc, 197273441091SDouglas Gregor diag::note_mmap_prev_definition); 197373441091SDouglas Gregor Failed = true; 197473441091SDouglas Gregor } 197573441091SDouglas Gregor 19769194a91dSDouglas Gregor // Check for the 'framework' keyword, which is not permitted here. 19779194a91dSDouglas Gregor if (Framework) { 19789194a91dSDouglas Gregor Diags.Report(StarLoc, diag::err_mmap_inferred_framework_submodule); 19799194a91dSDouglas Gregor Framework = false; 19809194a91dSDouglas Gregor } 19819194a91dSDouglas Gregor } else if (Explicit) { 19829194a91dSDouglas Gregor Diags.Report(StarLoc, diag::err_mmap_explicit_inferred_framework); 19839194a91dSDouglas Gregor Explicit = false; 19849194a91dSDouglas Gregor } 19859194a91dSDouglas Gregor 198673441091SDouglas Gregor // If there were any problems with this inferred submodule, skip its body. 198773441091SDouglas Gregor if (Failed) { 198873441091SDouglas Gregor if (Tok.is(MMToken::LBrace)) { 198973441091SDouglas Gregor consumeToken(); 199073441091SDouglas Gregor skipUntil(MMToken::RBrace); 199173441091SDouglas Gregor if (Tok.is(MMToken::RBrace)) 199273441091SDouglas Gregor consumeToken(); 199373441091SDouglas Gregor } 199473441091SDouglas Gregor HadError = true; 199573441091SDouglas Gregor return; 199673441091SDouglas Gregor } 199773441091SDouglas Gregor 19989194a91dSDouglas Gregor // Parse optional attributes. 19994442605fSBill Wendling Attributes Attrs; 20009194a91dSDouglas Gregor parseOptionalAttributes(Attrs); 20019194a91dSDouglas Gregor 20029194a91dSDouglas Gregor if (ActiveModule) { 200373441091SDouglas Gregor // Note that we have an inferred submodule. 2004dd005f69SDouglas Gregor ActiveModule->InferSubmodules = true; 2005dd005f69SDouglas Gregor ActiveModule->InferredSubmoduleLoc = StarLoc; 2006dd005f69SDouglas Gregor ActiveModule->InferExplicitSubmodules = Explicit; 20079194a91dSDouglas Gregor } else { 20089194a91dSDouglas Gregor // We'll be inferring framework modules for this directory. 20099194a91dSDouglas Gregor Map.InferredDirectories[Directory].InferModules = true; 20109194a91dSDouglas Gregor Map.InferredDirectories[Directory].InferSystemModules = Attrs.IsSystem; 2011131daca0SRichard Smith // FIXME: Handle the 'framework' keyword. 20129194a91dSDouglas Gregor } 201373441091SDouglas Gregor 201473441091SDouglas Gregor // Parse the opening brace. 201573441091SDouglas Gregor if (!Tok.is(MMToken::LBrace)) { 201673441091SDouglas Gregor Diags.Report(Tok.getLocation(), diag::err_mmap_expected_lbrace_wildcard); 201773441091SDouglas Gregor HadError = true; 201873441091SDouglas Gregor return; 201973441091SDouglas Gregor } 202073441091SDouglas Gregor SourceLocation LBraceLoc = consumeToken(); 202173441091SDouglas Gregor 202273441091SDouglas Gregor // Parse the body of the inferred submodule. 202373441091SDouglas Gregor bool Done = false; 202473441091SDouglas Gregor do { 202573441091SDouglas Gregor switch (Tok.Kind) { 202673441091SDouglas Gregor case MMToken::EndOfFile: 202773441091SDouglas Gregor case MMToken::RBrace: 202873441091SDouglas Gregor Done = true; 202973441091SDouglas Gregor break; 203073441091SDouglas Gregor 20319194a91dSDouglas Gregor case MMToken::ExcludeKeyword: { 20329194a91dSDouglas Gregor if (ActiveModule) { 20339194a91dSDouglas Gregor Diags.Report(Tok.getLocation(), diag::err_mmap_expected_inferred_member) 2034162405daSDouglas Gregor << (ActiveModule != 0); 20359194a91dSDouglas Gregor consumeToken(); 20369194a91dSDouglas Gregor break; 20379194a91dSDouglas Gregor } 20389194a91dSDouglas Gregor 20399194a91dSDouglas Gregor consumeToken(); 20409194a91dSDouglas Gregor if (!Tok.is(MMToken::Identifier)) { 20419194a91dSDouglas Gregor Diags.Report(Tok.getLocation(), diag::err_mmap_missing_exclude_name); 20429194a91dSDouglas Gregor break; 20439194a91dSDouglas Gregor } 20449194a91dSDouglas Gregor 20459194a91dSDouglas Gregor Map.InferredDirectories[Directory].ExcludedModules 20469194a91dSDouglas Gregor .push_back(Tok.getString()); 20479194a91dSDouglas Gregor consumeToken(); 20489194a91dSDouglas Gregor break; 20499194a91dSDouglas Gregor } 20509194a91dSDouglas Gregor 20519194a91dSDouglas Gregor case MMToken::ExportKeyword: 20529194a91dSDouglas Gregor if (!ActiveModule) { 20539194a91dSDouglas Gregor Diags.Report(Tok.getLocation(), diag::err_mmap_expected_inferred_member) 2054162405daSDouglas Gregor << (ActiveModule != 0); 20559194a91dSDouglas Gregor consumeToken(); 20569194a91dSDouglas Gregor break; 20579194a91dSDouglas Gregor } 20589194a91dSDouglas Gregor 205973441091SDouglas Gregor consumeToken(); 206073441091SDouglas Gregor if (Tok.is(MMToken::Star)) 2061dd005f69SDouglas Gregor ActiveModule->InferExportWildcard = true; 206273441091SDouglas Gregor else 206373441091SDouglas Gregor Diags.Report(Tok.getLocation(), 206473441091SDouglas Gregor diag::err_mmap_expected_export_wildcard); 206573441091SDouglas Gregor consumeToken(); 206673441091SDouglas Gregor break; 206773441091SDouglas Gregor 206873441091SDouglas Gregor case MMToken::ExplicitKeyword: 206973441091SDouglas Gregor case MMToken::ModuleKeyword: 207073441091SDouglas Gregor case MMToken::HeaderKeyword: 2071b53e5483SLawrence Crowl case MMToken::PrivateKeyword: 207273441091SDouglas Gregor case MMToken::UmbrellaKeyword: 207373441091SDouglas Gregor default: 20749194a91dSDouglas Gregor Diags.Report(Tok.getLocation(), diag::err_mmap_expected_inferred_member) 2075162405daSDouglas Gregor << (ActiveModule != 0); 207673441091SDouglas Gregor consumeToken(); 207773441091SDouglas Gregor break; 207873441091SDouglas Gregor } 207973441091SDouglas Gregor } while (!Done); 208073441091SDouglas Gregor 208173441091SDouglas Gregor if (Tok.is(MMToken::RBrace)) 208273441091SDouglas Gregor consumeToken(); 208373441091SDouglas Gregor else { 208473441091SDouglas Gregor Diags.Report(Tok.getLocation(), diag::err_mmap_expected_rbrace); 208573441091SDouglas Gregor Diags.Report(LBraceLoc, diag::note_mmap_lbrace_match); 208673441091SDouglas Gregor HadError = true; 208773441091SDouglas Gregor } 208873441091SDouglas Gregor } 208973441091SDouglas Gregor 20909194a91dSDouglas Gregor /// \brief Parse optional attributes. 20919194a91dSDouglas Gregor /// 20929194a91dSDouglas Gregor /// attributes: 20939194a91dSDouglas Gregor /// attribute attributes 20949194a91dSDouglas Gregor /// attribute 20959194a91dSDouglas Gregor /// 20969194a91dSDouglas Gregor /// attribute: 20979194a91dSDouglas Gregor /// [ identifier ] 20989194a91dSDouglas Gregor /// 20999194a91dSDouglas Gregor /// \param Attrs Will be filled in with the parsed attributes. 21009194a91dSDouglas Gregor /// 21019194a91dSDouglas Gregor /// \returns true if an error occurred, false otherwise. 21024442605fSBill Wendling bool ModuleMapParser::parseOptionalAttributes(Attributes &Attrs) { 21039194a91dSDouglas Gregor bool HadError = false; 21049194a91dSDouglas Gregor 21059194a91dSDouglas Gregor while (Tok.is(MMToken::LSquare)) { 21069194a91dSDouglas Gregor // Consume the '['. 21079194a91dSDouglas Gregor SourceLocation LSquareLoc = consumeToken(); 21089194a91dSDouglas Gregor 21099194a91dSDouglas Gregor // Check whether we have an attribute name here. 21109194a91dSDouglas Gregor if (!Tok.is(MMToken::Identifier)) { 21119194a91dSDouglas Gregor Diags.Report(Tok.getLocation(), diag::err_mmap_expected_attribute); 21129194a91dSDouglas Gregor skipUntil(MMToken::RSquare); 21139194a91dSDouglas Gregor if (Tok.is(MMToken::RSquare)) 21149194a91dSDouglas Gregor consumeToken(); 21159194a91dSDouglas Gregor HadError = true; 21169194a91dSDouglas Gregor } 21179194a91dSDouglas Gregor 21189194a91dSDouglas Gregor // Decode the attribute name. 21199194a91dSDouglas Gregor AttributeKind Attribute 21209194a91dSDouglas Gregor = llvm::StringSwitch<AttributeKind>(Tok.getString()) 212135b13eceSDouglas Gregor .Case("exhaustive", AT_exhaustive) 212277944868SRichard Smith .Case("extern_c", AT_extern_c) 21239194a91dSDouglas Gregor .Case("system", AT_system) 21249194a91dSDouglas Gregor .Default(AT_unknown); 21259194a91dSDouglas Gregor switch (Attribute) { 21269194a91dSDouglas Gregor case AT_unknown: 21279194a91dSDouglas Gregor Diags.Report(Tok.getLocation(), diag::warn_mmap_unknown_attribute) 21289194a91dSDouglas Gregor << Tok.getString(); 21299194a91dSDouglas Gregor break; 21309194a91dSDouglas Gregor 21319194a91dSDouglas Gregor case AT_system: 21329194a91dSDouglas Gregor Attrs.IsSystem = true; 21339194a91dSDouglas Gregor break; 213435b13eceSDouglas Gregor 213577944868SRichard Smith case AT_extern_c: 213677944868SRichard Smith Attrs.IsExternC = true; 213777944868SRichard Smith break; 213877944868SRichard Smith 213935b13eceSDouglas Gregor case AT_exhaustive: 214035b13eceSDouglas Gregor Attrs.IsExhaustive = true; 214135b13eceSDouglas Gregor break; 21429194a91dSDouglas Gregor } 21439194a91dSDouglas Gregor consumeToken(); 21449194a91dSDouglas Gregor 21459194a91dSDouglas Gregor // Consume the ']'. 21469194a91dSDouglas Gregor if (!Tok.is(MMToken::RSquare)) { 21479194a91dSDouglas Gregor Diags.Report(Tok.getLocation(), diag::err_mmap_expected_rsquare); 21489194a91dSDouglas Gregor Diags.Report(LSquareLoc, diag::note_mmap_lsquare_match); 21499194a91dSDouglas Gregor skipUntil(MMToken::RSquare); 21509194a91dSDouglas Gregor HadError = true; 21519194a91dSDouglas Gregor } 21529194a91dSDouglas Gregor 21539194a91dSDouglas Gregor if (Tok.is(MMToken::RSquare)) 21549194a91dSDouglas Gregor consumeToken(); 21559194a91dSDouglas Gregor } 21569194a91dSDouglas Gregor 21579194a91dSDouglas Gregor return HadError; 21589194a91dSDouglas Gregor } 21599194a91dSDouglas Gregor 21607033127bSDouglas Gregor /// \brief If there is a specific header search directory due the presence 21617033127bSDouglas Gregor /// of an umbrella directory, retrieve that directory. Otherwise, returns null. 21627033127bSDouglas Gregor const DirectoryEntry *ModuleMapParser::getOverriddenHeaderSearchDir() { 21637033127bSDouglas Gregor for (Module *Mod = ActiveModule; Mod; Mod = Mod->Parent) { 21647033127bSDouglas Gregor // If we have an umbrella directory, use that. 21657033127bSDouglas Gregor if (Mod->hasUmbrellaDir()) 21667033127bSDouglas Gregor return Mod->getUmbrellaDir(); 21677033127bSDouglas Gregor 21687033127bSDouglas Gregor // If we have a framework directory, stop looking. 21697033127bSDouglas Gregor if (Mod->IsFramework) 21707033127bSDouglas Gregor return 0; 21717033127bSDouglas Gregor } 21727033127bSDouglas Gregor 21737033127bSDouglas Gregor return 0; 21747033127bSDouglas Gregor } 21757033127bSDouglas Gregor 2176718292f2SDouglas Gregor /// \brief Parse a module map file. 2177718292f2SDouglas Gregor /// 2178718292f2SDouglas Gregor /// module-map-file: 2179718292f2SDouglas Gregor /// module-declaration* 2180718292f2SDouglas Gregor bool ModuleMapParser::parseModuleMapFile() { 2181718292f2SDouglas Gregor do { 2182718292f2SDouglas Gregor switch (Tok.Kind) { 2183718292f2SDouglas Gregor case MMToken::EndOfFile: 2184718292f2SDouglas Gregor return HadError; 2185718292f2SDouglas Gregor 2186e7ab3669SDouglas Gregor case MMToken::ExplicitKeyword: 218797292843SDaniel Jasper case MMToken::ExternKeyword: 2188718292f2SDouglas Gregor case MMToken::ModuleKeyword: 2189755b2055SDouglas Gregor case MMToken::FrameworkKeyword: 2190718292f2SDouglas Gregor parseModuleDecl(); 2191718292f2SDouglas Gregor break; 2192718292f2SDouglas Gregor 21931fb5c3a6SDouglas Gregor case MMToken::Comma: 219435b13eceSDouglas Gregor case MMToken::ConfigMacros: 2195fb912657SDouglas Gregor case MMToken::Conflict: 2196a3feee2aSRichard Smith case MMToken::Exclaim: 219759527666SDouglas Gregor case MMToken::ExcludeKeyword: 21982b82c2a5SDouglas Gregor case MMToken::ExportKeyword: 2199718292f2SDouglas Gregor case MMToken::HeaderKeyword: 2200718292f2SDouglas Gregor case MMToken::Identifier: 2201718292f2SDouglas Gregor case MMToken::LBrace: 22026ddfca91SDouglas Gregor case MMToken::LinkKeyword: 2203a686e1b0SDouglas Gregor case MMToken::LSquare: 22042b82c2a5SDouglas Gregor case MMToken::Period: 2205b53e5483SLawrence Crowl case MMToken::PrivateKeyword: 2206718292f2SDouglas Gregor case MMToken::RBrace: 2207a686e1b0SDouglas Gregor case MMToken::RSquare: 22081fb5c3a6SDouglas Gregor case MMToken::RequiresKeyword: 22092b82c2a5SDouglas Gregor case MMToken::Star: 2210718292f2SDouglas Gregor case MMToken::StringLiteral: 2211718292f2SDouglas Gregor case MMToken::UmbrellaKeyword: 2212ba7f2f71SDaniel Jasper case MMToken::UseKeyword: 2213718292f2SDouglas Gregor Diags.Report(Tok.getLocation(), diag::err_mmap_expected_module); 2214718292f2SDouglas Gregor HadError = true; 2215718292f2SDouglas Gregor consumeToken(); 2216718292f2SDouglas Gregor break; 2217718292f2SDouglas Gregor } 2218718292f2SDouglas Gregor } while (true); 2219718292f2SDouglas Gregor } 2220718292f2SDouglas Gregor 2221963c5535SDouglas Gregor bool ModuleMap::parseModuleMapFile(const FileEntry *File, bool IsSystem) { 22224ddf2221SDouglas Gregor llvm::DenseMap<const FileEntry *, bool>::iterator Known 22234ddf2221SDouglas Gregor = ParsedModuleMap.find(File); 22244ddf2221SDouglas Gregor if (Known != ParsedModuleMap.end()) 22254ddf2221SDouglas Gregor return Known->second; 22264ddf2221SDouglas Gregor 222789929282SDouglas Gregor assert(Target != 0 && "Missing target information"); 2228cb69b57bSBen Langmuir auto FileCharacter = IsSystem ? SrcMgr::C_System : SrcMgr::C_User; 2229cb69b57bSBen Langmuir FileID ID = SourceMgr.createFileID(File, SourceLocation(), FileCharacter); 22301f76c4e8SManuel Klimek const llvm::MemoryBuffer *Buffer = SourceMgr.getBuffer(ID); 2231718292f2SDouglas Gregor if (!Buffer) 22324ddf2221SDouglas Gregor return ParsedModuleMap[File] = true; 2233718292f2SDouglas Gregor 2234984e1df7SBen Langmuir // Find the directory for the module. For frameworks, that may require going 2235984e1df7SBen Langmuir // up from the 'Modules' directory. 2236984e1df7SBen Langmuir const DirectoryEntry *Dir = File->getDir(); 2237984e1df7SBen Langmuir StringRef DirName(Dir->getName()); 2238984e1df7SBen Langmuir if (llvm::sys::path::filename(DirName) == "Modules") { 2239984e1df7SBen Langmuir DirName = llvm::sys::path::parent_path(DirName); 2240984e1df7SBen Langmuir if (DirName.endswith(".framework")) 2241984e1df7SBen Langmuir Dir = SourceMgr.getFileManager().getDirectory(DirName); 2242984e1df7SBen Langmuir assert(Dir && "parent must exist"); 2243984e1df7SBen Langmuir } 2244984e1df7SBen Langmuir 2245718292f2SDouglas Gregor // Parse this module map file. 22461f76c4e8SManuel Klimek Lexer L(ID, SourceMgr.getBuffer(ID), SourceMgr, MMapLangOpts); 2247984e1df7SBen Langmuir ModuleMapParser Parser(L, SourceMgr, Target, Diags, *this, Dir, 2248963c5535SDouglas Gregor BuiltinIncludeDir, IsSystem); 2249718292f2SDouglas Gregor bool Result = Parser.parseModuleMapFile(); 22504ddf2221SDouglas Gregor ParsedModuleMap[File] = Result; 2251718292f2SDouglas Gregor return Result; 2252718292f2SDouglas Gregor } 2253