//===--- ModuleMap.cpp - Describe the layout of modules ---------*- C++ -*-===// // // The LLVM Compiler Infrastructure // // This file is distributed under the University of Illinois Open Source // License. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// // // This file defines the ModuleMap implementation, which describes the layout // of a module as it relates to headers. // //===----------------------------------------------------------------------===// #include "clang/Lex/ModuleMap.h" #include "clang/Lex/Lexer.h" #include "clang/Lex/LiteralSupport.h" #include "clang/Lex/LexDiagnostic.h" #include "clang/Basic/Diagnostic.h" #include "clang/Basic/FileManager.h" #include "clang/Basic/TargetInfo.h" #include "clang/Basic/TargetOptions.h" #include "llvm/Support/Allocator.h" #include "llvm/Support/Host.h" #include "llvm/Support/PathV2.h" #include "llvm/Support/raw_ostream.h" #include "llvm/ADT/StringRef.h" #include "llvm/ADT/StringSwitch.h" using namespace clang; Module::ExportDecl ModuleMap::resolveExport(Module *Mod, const Module::UnresolvedExportDecl &Unresolved, bool Complain) { // We may have just a wildcard. if (Unresolved.Id.empty()) { assert(Unresolved.Wildcard && "Invalid unresolved export"); return Module::ExportDecl(0, true); } // Find the starting module. Module *Context = lookupModuleUnqualified(Unresolved.Id[0].first, Mod); if (!Context) { if (Complain) Diags->Report(Unresolved.Id[0].second, diag::err_mmap_missing_module_unqualified) << Unresolved.Id[0].first << Mod->getFullModuleName(); return Module::ExportDecl(); } // Dig into the module path. for (unsigned I = 1, N = Unresolved.Id.size(); I != N; ++I) { Module *Sub = lookupModuleQualified(Unresolved.Id[I].first, Context); if (!Sub) { if (Complain) Diags->Report(Unresolved.Id[I].second, diag::err_mmap_missing_module_qualified) << Unresolved.Id[I].first << Context->getFullModuleName() << SourceRange(Unresolved.Id[0].second, Unresolved.Id[I-1].second); return Module::ExportDecl(); } Context = Sub; } return Module::ExportDecl(Context, Unresolved.Wildcard); } ModuleMap::ModuleMap(FileManager &FileMgr, const DiagnosticConsumer &DC) { llvm::IntrusiveRefCntPtr DiagIDs(new DiagnosticIDs); Diags = llvm::IntrusiveRefCntPtr( new DiagnosticsEngine(DiagIDs)); Diags->setClient(DC.clone(*Diags), /*ShouldOwnClient=*/true); SourceMgr = new SourceManager(*Diags, FileMgr); } ModuleMap::~ModuleMap() { for (llvm::StringMap::iterator I = Modules.begin(), IEnd = Modules.end(); I != IEnd; ++I) { delete I->getValue(); } delete SourceMgr; } Module *ModuleMap::findModuleForHeader(const FileEntry *File) { llvm::DenseMap::iterator Known = Headers.find(File); if (Known != Headers.end()) return Known->second; const DirectoryEntry *Dir = File->getDir(); llvm::SmallVector SkippedDirs; StringRef DirName = Dir->getName(); // Keep walking up the directory hierarchy, looking for a directory with // an umbrella header. do { llvm::DenseMap::iterator KnownDir = UmbrellaDirs.find(Dir); if (KnownDir != UmbrellaDirs.end()) { Module *Result = KnownDir->second; // Search up the module stack until we find a module with an umbrella // header. Module *UmbrellaModule = Result; while (!UmbrellaModule->UmbrellaHeader && UmbrellaModule->Parent) UmbrellaModule = UmbrellaModule->Parent; if (UmbrellaModule->InferSubmodules) { // Infer submodules for each of the directories we found between // the directory of the umbrella header and the directory where // the actual header is located. // For a framework module, the umbrella directory is the framework // directory, so strip off the "Headers" or "PrivateHeaders". // FIXME: Should we tack on an "explicit" for PrivateHeaders? That // might be what we want, but it feels like a hack. unsigned LastSkippedDir = SkippedDirs.size(); if (LastSkippedDir && UmbrellaModule->IsFramework) --LastSkippedDir; for (unsigned I = LastSkippedDir; I != 0; --I) { // Find or create the module that corresponds to this directory name. StringRef Name = llvm::sys::path::stem(SkippedDirs[I-1]->getName()); Result = findOrCreateModule(Name, Result, /*IsFramework=*/false, UmbrellaModule->InferExplicitSubmodules).first; // Associate the module and the directory. UmbrellaDirs[SkippedDirs[I-1]] = Result; // If inferred submodules export everything they import, add a // wildcard to the set of exports. if (UmbrellaModule->InferExportWildcard && Result->Exports.empty()) Result->Exports.push_back(Module::ExportDecl(0, true)); } // Infer a submodule with the same name as this header file. StringRef Name = llvm::sys::path::stem(File->getName()); Result = findOrCreateModule(Name, Result, /*IsFramework=*/false, UmbrellaModule->InferExplicitSubmodules).first; // If inferred submodules export everything they import, add a // wildcard to the set of exports. if (UmbrellaModule->InferExportWildcard && Result->Exports.empty()) Result->Exports.push_back(Module::ExportDecl(0, true)); } else { // Record each of the directories we stepped through as being part of // the module we found, since the umbrella header covers them all. for (unsigned I = 0, N = SkippedDirs.size(); I != N; ++I) UmbrellaDirs[SkippedDirs[I]] = Result; } Headers[File] = Result; return Result; } SkippedDirs.push_back(Dir); // Retrieve our parent path. DirName = llvm::sys::path::parent_path(DirName); if (DirName.empty()) break; // Resolve the parent path to a directory entry. Dir = SourceMgr->getFileManager().getDirectory(DirName); } while (Dir); return 0; } Module *ModuleMap::findModule(StringRef Name) { llvm::StringMap::iterator Known = Modules.find(Name); if (Known != Modules.end()) return Known->getValue(); return 0; } Module *ModuleMap::lookupModuleUnqualified(StringRef Name, Module *Context) { for(; Context; Context = Context->Parent) { if (Module *Sub = lookupModuleQualified(Name, Context)) return Sub; } return findModule(Name); } Module *ModuleMap::lookupModuleQualified(StringRef Name, Module *Context) { if (!Context) return findModule(Name); llvm::StringMap::iterator Sub = Context->SubModules.find(Name); if (Sub != Context->SubModules.end()) return Sub->getValue(); return 0; } std::pair ModuleMap::findOrCreateModule(StringRef Name, Module *Parent, bool IsFramework, bool IsExplicit) { // Try to find an existing module with this name. if (Module *Found = Parent? Parent->SubModules[Name] : Modules[Name]) return std::make_pair(Found, false); // Create a new module with this name. Module *Result = new Module(Name, SourceLocation(), Parent, IsFramework, IsExplicit); if (Parent) Parent->SubModules[Name] = Result; else Modules[Name] = Result; return std::make_pair(Result, true); } Module * ModuleMap::inferFrameworkModule(StringRef ModuleName, const DirectoryEntry *FrameworkDir) { // Check whether we've already found this module. if (Module *Module = findModule(ModuleName)) return Module; // Look for an umbrella header. llvm::SmallString<128> UmbrellaName = StringRef(FrameworkDir->getName()); llvm::sys::path::append(UmbrellaName, "Headers"); llvm::sys::path::append(UmbrellaName, ModuleName + ".h"); const FileEntry *UmbrellaHeader = SourceMgr->getFileManager().getFile(UmbrellaName); // FIXME: If there's no umbrella header, we could probably scan the // framework to load *everything*. But, it's not clear that this is a good // idea. if (!UmbrellaHeader) return 0; Module *Result = new Module(ModuleName, SourceLocation(), /*IsFramework=*/true); // umbrella "umbrella-header-name" Result->UmbrellaHeader = UmbrellaHeader; Headers[UmbrellaHeader] = Result; UmbrellaDirs[FrameworkDir] = Result; // export * Result->Exports.push_back(Module::ExportDecl(0, true)); // module * { export * } Result->InferSubmodules = true; Result->InferExportWildcard = true; Modules[ModuleName] = Result; return Result; } void ModuleMap::setUmbrellaHeader(Module *Mod, const FileEntry *UmbrellaHeader){ Headers[UmbrellaHeader] = Mod; Mod->UmbrellaHeader = UmbrellaHeader; const DirectoryEntry *UmbrellaDir = UmbrellaHeader->getDir(); if (Mod->IsFramework) UmbrellaDir = SourceMgr->getFileManager().getDirectory( llvm::sys::path::parent_path(UmbrellaDir->getName())); UmbrellaDirs[UmbrellaDir] = Mod; } void ModuleMap::addHeader(Module *Mod, const FileEntry *Header) { Mod->Headers.push_back(Header); Headers[Header] = Mod; } const FileEntry * ModuleMap::getContainingModuleMapFile(Module *Module) { if (Module->DefinitionLoc.isInvalid() || !SourceMgr) return 0; return SourceMgr->getFileEntryForID( SourceMgr->getFileID(Module->DefinitionLoc)); } void ModuleMap::dump() { llvm::errs() << "Modules:"; for (llvm::StringMap::iterator M = Modules.begin(), MEnd = Modules.end(); M != MEnd; ++M) M->getValue()->print(llvm::errs(), 2); llvm::errs() << "Headers:"; for (llvm::DenseMap::iterator H = Headers.begin(), HEnd = Headers.end(); H != HEnd; ++H) { llvm::errs() << " \"" << H->first->getName() << "\" -> " << H->second->getFullModuleName() << "\n"; } } bool ModuleMap::resolveExports(Module *Mod, bool Complain) { bool HadError = false; for (unsigned I = 0, N = Mod->UnresolvedExports.size(); I != N; ++I) { Module::ExportDecl Export = resolveExport(Mod, Mod->UnresolvedExports[I], Complain); if (Export.getPointer() || Export.getInt()) Mod->Exports.push_back(Export); else HadError = true; } Mod->UnresolvedExports.clear(); return HadError; } Module *ModuleMap::inferModuleFromLocation(FullSourceLoc Loc) { if (Loc.isInvalid()) return 0; // Use the expansion location to determine which module we're in. FullSourceLoc ExpansionLoc = Loc.getExpansionLoc(); if (!ExpansionLoc.isFileID()) return 0; const SourceManager &SrcMgr = Loc.getManager(); FileID ExpansionFileID = ExpansionLoc.getFileID(); const FileEntry *ExpansionFile = SrcMgr.getFileEntryForID(ExpansionFileID); if (!ExpansionFile) return 0; // Find the module that owns this header. return findModuleForHeader(ExpansionFile); } //----------------------------------------------------------------------------// // Module map file parser //----------------------------------------------------------------------------// namespace clang { /// \brief A token in a module map file. struct MMToken { enum TokenKind { EndOfFile, HeaderKeyword, Identifier, ExplicitKeyword, ExportKeyword, FrameworkKeyword, ModuleKeyword, Period, UmbrellaKeyword, Star, StringLiteral, LBrace, RBrace } Kind; unsigned Location; unsigned StringLength; const char *StringData; void clear() { Kind = EndOfFile; Location = 0; StringLength = 0; StringData = 0; } bool is(TokenKind K) const { return Kind == K; } SourceLocation getLocation() const { return SourceLocation::getFromRawEncoding(Location); } StringRef getString() const { return StringRef(StringData, StringLength); } }; class ModuleMapParser { Lexer &L; SourceManager &SourceMgr; DiagnosticsEngine &Diags; ModuleMap ⤅ /// \brief The directory that this module map resides in. const DirectoryEntry *Directory; /// \brief Whether an error occurred. bool HadError; /// \brief Default target information, used only for string literal /// parsing. TargetInfo *Target; /// \brief Stores string data for the various string literals referenced /// during parsing. llvm::BumpPtrAllocator StringData; /// \brief The current token. MMToken Tok; /// \brief The active module. Module *ActiveModule; /// \brief Consume the current token and return its location. SourceLocation consumeToken(); /// \brief Skip tokens until we reach the a token with the given kind /// (or the end of the file). void skipUntil(MMToken::TokenKind K); void parseModuleDecl(); void parseUmbrellaDecl(); void parseHeaderDecl(); void parseExportDecl(); void parseInferredSubmoduleDecl(bool Explicit); public: explicit ModuleMapParser(Lexer &L, SourceManager &SourceMgr, DiagnosticsEngine &Diags, ModuleMap &Map, const DirectoryEntry *Directory) : L(L), SourceMgr(SourceMgr), Diags(Diags), Map(Map), Directory(Directory), HadError(false), ActiveModule(0) { TargetOptions TargetOpts; TargetOpts.Triple = llvm::sys::getDefaultTargetTriple(); Target = TargetInfo::CreateTargetInfo(Diags, TargetOpts); Tok.clear(); consumeToken(); } bool parseModuleMapFile(); }; } SourceLocation ModuleMapParser::consumeToken() { retry: SourceLocation Result = Tok.getLocation(); Tok.clear(); Token LToken; L.LexFromRawLexer(LToken); Tok.Location = LToken.getLocation().getRawEncoding(); switch (LToken.getKind()) { case tok::raw_identifier: Tok.StringData = LToken.getRawIdentifierData(); Tok.StringLength = LToken.getLength(); Tok.Kind = llvm::StringSwitch(Tok.getString()) .Case("header", MMToken::HeaderKeyword) .Case("explicit", MMToken::ExplicitKeyword) .Case("export", MMToken::ExportKeyword) .Case("framework", MMToken::FrameworkKeyword) .Case("module", MMToken::ModuleKeyword) .Case("umbrella", MMToken::UmbrellaKeyword) .Default(MMToken::Identifier); break; case tok::eof: Tok.Kind = MMToken::EndOfFile; break; case tok::l_brace: Tok.Kind = MMToken::LBrace; break; case tok::period: Tok.Kind = MMToken::Period; break; case tok::r_brace: Tok.Kind = MMToken::RBrace; break; case tok::star: Tok.Kind = MMToken::Star; break; case tok::string_literal: { // Parse the string literal. LangOptions LangOpts; StringLiteralParser StringLiteral(<oken, 1, SourceMgr, LangOpts, *Target); if (StringLiteral.hadError) goto retry; // Copy the string literal into our string data allocator. unsigned Length = StringLiteral.GetStringLength(); char *Saved = StringData.Allocate(Length + 1); memcpy(Saved, StringLiteral.GetString().data(), Length); Saved[Length] = 0; // Form the token. Tok.Kind = MMToken::StringLiteral; Tok.StringData = Saved; Tok.StringLength = Length; break; } case tok::comment: goto retry; default: Diags.Report(LToken.getLocation(), diag::err_mmap_unknown_token); HadError = true; goto retry; } return Result; } void ModuleMapParser::skipUntil(MMToken::TokenKind K) { unsigned braceDepth = 0; do { switch (Tok.Kind) { case MMToken::EndOfFile: return; case MMToken::LBrace: if (Tok.is(K) && braceDepth == 0) return; ++braceDepth; break; case MMToken::RBrace: if (braceDepth > 0) --braceDepth; else if (Tok.is(K)) return; break; default: if (braceDepth == 0 && Tok.is(K)) return; break; } consumeToken(); } while (true); } /// \brief Parse a module declaration. /// /// module-declaration: /// 'framework'[opt] 'module' identifier { module-member* } /// /// module-member: /// umbrella-declaration /// header-declaration /// 'explicit'[opt] submodule-declaration /// export-declaration /// /// submodule-declaration: /// module-declaration /// inferred-submodule-declaration void ModuleMapParser::parseModuleDecl() { assert(Tok.is(MMToken::ExplicitKeyword) || Tok.is(MMToken::ModuleKeyword) || Tok.is(MMToken::FrameworkKeyword)); // Parse 'framework' or 'explicit' keyword, if present. bool Framework = false; bool Explicit = false; if (Tok.is(MMToken::FrameworkKeyword)) { consumeToken(); Framework = true; } // Parse 'explicit' keyword, if present. else if (Tok.is(MMToken::ExplicitKeyword)) { consumeToken(); Explicit = true; } // Parse 'module' keyword. if (!Tok.is(MMToken::ModuleKeyword)) { Diags.Report(Tok.getLocation(), diag::err_mmap_expected_module_after_explicit); consumeToken(); HadError = true; return; } consumeToken(); // 'module' keyword // If we have a wildcard for the module name, this is an inferred submodule. // Parse it. if (Tok.is(MMToken::Star)) return parseInferredSubmoduleDecl(Explicit); // Parse the module name. if (!Tok.is(MMToken::Identifier)) { Diags.Report(Tok.getLocation(), diag::err_mmap_expected_module_name); HadError = true; return; } StringRef ModuleName = Tok.getString(); SourceLocation ModuleNameLoc = consumeToken(); // Parse the opening brace. if (!Tok.is(MMToken::LBrace)) { Diags.Report(Tok.getLocation(), diag::err_mmap_expected_lbrace) << ModuleName; HadError = true; return; } SourceLocation LBraceLoc = consumeToken(); // Determine whether this (sub)module has already been defined. llvm::StringMap &ModuleSpace = ActiveModule? ActiveModule->SubModules : Map.Modules; llvm::StringMap::iterator ExistingModule = ModuleSpace.find(ModuleName); if (ExistingModule != ModuleSpace.end()) { Diags.Report(ModuleNameLoc, diag::err_mmap_module_redefinition) << ModuleName; Diags.Report(ExistingModule->getValue()->DefinitionLoc, diag::note_mmap_prev_definition); // Skip the module definition. skipUntil(MMToken::RBrace); if (Tok.is(MMToken::RBrace)) consumeToken(); HadError = true; return; } // Start defining this module. ActiveModule = new Module(ModuleName, ModuleNameLoc, ActiveModule, Framework, Explicit); ModuleSpace[ModuleName] = ActiveModule; bool Done = false; do { switch (Tok.Kind) { case MMToken::EndOfFile: case MMToken::RBrace: Done = true; break; case MMToken::ExplicitKeyword: case MMToken::ModuleKeyword: parseModuleDecl(); break; case MMToken::ExportKeyword: parseExportDecl(); break; case MMToken::HeaderKeyword: parseHeaderDecl(); break; case MMToken::UmbrellaKeyword: parseUmbrellaDecl(); break; default: Diags.Report(Tok.getLocation(), diag::err_mmap_expected_member); consumeToken(); break; } } while (!Done); if (Tok.is(MMToken::RBrace)) consumeToken(); else { Diags.Report(Tok.getLocation(), diag::err_mmap_expected_rbrace); Diags.Report(LBraceLoc, diag::note_mmap_lbrace_match); HadError = true; } // We're done parsing this module. Pop back to our parent scope. ActiveModule = ActiveModule->Parent; } /// \brief Parse an umbrella header declaration. /// /// umbrella-declaration: /// 'umbrella' string-literal void ModuleMapParser::parseUmbrellaDecl() { assert(Tok.is(MMToken::UmbrellaKeyword)); SourceLocation UmbrellaLoc = consumeToken(); // Parse the header name. if (!Tok.is(MMToken::StringLiteral)) { Diags.Report(Tok.getLocation(), diag::err_mmap_expected_header) << "umbrella"; HadError = true; return; } StringRef FileName = Tok.getString(); SourceLocation FileNameLoc = consumeToken(); // Check whether we already have an umbrella header. if (ActiveModule->UmbrellaHeader) { Diags.Report(FileNameLoc, diag::err_mmap_umbrella_header_conflict) << ActiveModule->getFullModuleName() << ActiveModule->UmbrellaHeader->getName(); HadError = true; return; } // Only top-level modules can have umbrella headers. if (ActiveModule->Parent) { Diags.Report(UmbrellaLoc, diag::err_mmap_umbrella_header_submodule) << ActiveModule->getFullModuleName(); HadError = true; return; } // Look for this file. llvm::SmallString<128> PathName; const FileEntry *File = 0; if (llvm::sys::path::is_absolute(FileName)) { PathName = FileName; File = SourceMgr.getFileManager().getFile(PathName); } else { // Search for the header file within the search directory. PathName += Directory->getName(); unsigned PathLength = PathName.size(); if (ActiveModule->isPartOfFramework()) { // Check whether this file is in the public headers. llvm::sys::path::append(PathName, "Headers"); llvm::sys::path::append(PathName, FileName); File = SourceMgr.getFileManager().getFile(PathName); if (!File) { // Check whether this file is in the private headers. PathName.resize(PathLength); llvm::sys::path::append(PathName, "PrivateHeaders"); llvm::sys::path::append(PathName, FileName); File = SourceMgr.getFileManager().getFile(PathName); } // FIXME: Deal with subframeworks. } else { // Lookup for normal headers. llvm::sys::path::append(PathName, FileName); File = SourceMgr.getFileManager().getFile(PathName); } } // FIXME: We shouldn't be eagerly stat'ing every file named in a module map. // Come up with a lazy way to do this. if (File) { const DirectoryEntry *UmbrellaDir = File->getDir(); if (ActiveModule->IsFramework) { // For framework modules, use the framework directory as the umbrella // directory. UmbrellaDir = SourceMgr.getFileManager().getDirectory( llvm::sys::path::parent_path(UmbrellaDir->getName())); } if (const Module *OwningModule = Map.Headers[File]) { Diags.Report(FileNameLoc, diag::err_mmap_header_conflict) << FileName << OwningModule->getFullModuleName(); HadError = true; } else if ((OwningModule = Map.UmbrellaDirs[UmbrellaDir])) { Diags.Report(UmbrellaLoc, diag::err_mmap_umbrella_clash) << OwningModule->getFullModuleName(); HadError = true; } else { // Record this umbrella header. Map.setUmbrellaHeader(ActiveModule, File); } } else { Diags.Report(FileNameLoc, diag::err_mmap_header_not_found) << true << FileName; HadError = true; } } /// \brief Parse a header declaration. /// /// header-declaration: /// 'header' string-literal void ModuleMapParser::parseHeaderDecl() { assert(Tok.is(MMToken::HeaderKeyword)); consumeToken(); // Parse the header name. if (!Tok.is(MMToken::StringLiteral)) { Diags.Report(Tok.getLocation(), diag::err_mmap_expected_header) << "header"; HadError = true; return; } StringRef FileName = Tok.getString(); SourceLocation FileNameLoc = consumeToken(); // Look for this file. llvm::SmallString<128> PathName; if (llvm::sys::path::is_relative(FileName)) { // FIXME: Change this search to also look for private headers! PathName += Directory->getName(); if (ActiveModule->isPartOfFramework()) llvm::sys::path::append(PathName, "Headers"); } llvm::sys::path::append(PathName, FileName); // FIXME: We shouldn't be eagerly stat'ing every file named in a module map. // Come up with a lazy way to do this. if (const FileEntry *File = SourceMgr.getFileManager().getFile(PathName)) { if (const Module *OwningModule = Map.Headers[File]) { Diags.Report(FileNameLoc, diag::err_mmap_header_conflict) << FileName << OwningModule->getFullModuleName(); HadError = true; } else { // Record this file. Map.addHeader(ActiveModule, File); } } else { Diags.Report(FileNameLoc, diag::err_mmap_header_not_found) << false << FileName; HadError = true; } } /// \brief Parse a module export declaration. /// /// export-declaration: /// 'export' wildcard-module-id /// /// wildcard-module-id: /// identifier /// '*' /// identifier '.' wildcard-module-id void ModuleMapParser::parseExportDecl() { assert(Tok.is(MMToken::ExportKeyword)); SourceLocation ExportLoc = consumeToken(); // Parse the module-id with an optional wildcard at the end. ModuleId ParsedModuleId; bool Wildcard = false; do { if (Tok.is(MMToken::Identifier)) { ParsedModuleId.push_back(std::make_pair(Tok.getString(), Tok.getLocation())); consumeToken(); if (Tok.is(MMToken::Period)) { consumeToken(); continue; } break; } if(Tok.is(MMToken::Star)) { Wildcard = true; consumeToken(); break; } Diags.Report(Tok.getLocation(), diag::err_mmap_export_module_id); HadError = true; return; } while (true); Module::UnresolvedExportDecl Unresolved = { ExportLoc, ParsedModuleId, Wildcard }; ActiveModule->UnresolvedExports.push_back(Unresolved); } void ModuleMapParser::parseInferredSubmoduleDecl(bool Explicit) { assert(Tok.is(MMToken::Star)); SourceLocation StarLoc = consumeToken(); bool Failed = false; // Inferred modules must be submodules. if (!ActiveModule) { Diags.Report(StarLoc, diag::err_mmap_top_level_inferred_submodule); Failed = true; } // Inferred modules must have umbrella headers. if (!Failed && !ActiveModule->getTopLevelModule()->UmbrellaHeader) { Diags.Report(StarLoc, diag::err_mmap_inferred_no_umbrella); Failed = true; } // Check for redefinition of an inferred module. if (!Failed && ActiveModule->getTopLevelModule()->InferSubmodules) { Diags.Report(StarLoc, diag::err_mmap_inferred_redef); if (ActiveModule->getTopLevelModule()->InferredSubmoduleLoc.isValid()) Diags.Report(ActiveModule->getTopLevelModule()->InferredSubmoduleLoc, diag::note_mmap_prev_definition); Failed = true; } // If there were any problems with this inferred submodule, skip its body. if (Failed) { if (Tok.is(MMToken::LBrace)) { consumeToken(); skipUntil(MMToken::RBrace); if (Tok.is(MMToken::RBrace)) consumeToken(); } HadError = true; return; } // Note that we have an inferred submodule. Module *TopModule = ActiveModule->getTopLevelModule(); TopModule->InferSubmodules = true; TopModule->InferredSubmoduleLoc = StarLoc; TopModule->InferExplicitSubmodules = Explicit; // Parse the opening brace. if (!Tok.is(MMToken::LBrace)) { Diags.Report(Tok.getLocation(), diag::err_mmap_expected_lbrace_wildcard); HadError = true; return; } SourceLocation LBraceLoc = consumeToken(); // Parse the body of the inferred submodule. bool Done = false; do { switch (Tok.Kind) { case MMToken::EndOfFile: case MMToken::RBrace: Done = true; break; case MMToken::ExportKeyword: { consumeToken(); if (Tok.is(MMToken::Star)) TopModule->InferExportWildcard = true; else Diags.Report(Tok.getLocation(), diag::err_mmap_expected_export_wildcard); consumeToken(); break; } case MMToken::ExplicitKeyword: case MMToken::ModuleKeyword: case MMToken::HeaderKeyword: case MMToken::UmbrellaKeyword: default: Diags.Report(Tok.getLocation(), diag::err_mmap_expected_wildcard_member); consumeToken(); break; } } while (!Done); if (Tok.is(MMToken::RBrace)) consumeToken(); else { Diags.Report(Tok.getLocation(), diag::err_mmap_expected_rbrace); Diags.Report(LBraceLoc, diag::note_mmap_lbrace_match); HadError = true; } } /// \brief Parse a module map file. /// /// module-map-file: /// module-declaration* bool ModuleMapParser::parseModuleMapFile() { do { switch (Tok.Kind) { case MMToken::EndOfFile: return HadError; case MMToken::ModuleKeyword: case MMToken::FrameworkKeyword: parseModuleDecl(); break; case MMToken::ExplicitKeyword: case MMToken::ExportKeyword: case MMToken::HeaderKeyword: case MMToken::Identifier: case MMToken::LBrace: case MMToken::Period: case MMToken::RBrace: case MMToken::Star: case MMToken::StringLiteral: case MMToken::UmbrellaKeyword: Diags.Report(Tok.getLocation(), diag::err_mmap_expected_module); HadError = true; consumeToken(); break; } } while (true); return HadError; } bool ModuleMap::parseModuleMapFile(const FileEntry *File) { FileID ID = SourceMgr->createFileID(File, SourceLocation(), SrcMgr::C_User); const llvm::MemoryBuffer *Buffer = SourceMgr->getBuffer(ID); if (!Buffer) return true; // Parse this module map file. Lexer L(ID, SourceMgr->getBuffer(ID), *SourceMgr, LangOpts); Diags->getClient()->BeginSourceFile(LangOpts); ModuleMapParser Parser(L, *SourceMgr, *Diags, *this, File->getDir()); bool Result = Parser.parseModuleMapFile(); Diags->getClient()->EndSourceFile(); return Result; }