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 &Map;
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(&LToken, 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