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" 15718292f2SDouglas Gregor #include "clang/Lex/Lexer.h" 16718292f2SDouglas Gregor #include "clang/Lex/LiteralSupport.h" 17718292f2SDouglas Gregor #include "clang/Lex/LexDiagnostic.h" 18718292f2SDouglas Gregor #include "clang/Basic/Diagnostic.h" 19718292f2SDouglas Gregor #include "clang/Basic/FileManager.h" 20718292f2SDouglas Gregor #include "clang/Basic/TargetInfo.h" 21718292f2SDouglas Gregor #include "clang/Basic/TargetOptions.h" 22718292f2SDouglas Gregor #include "llvm/Support/Allocator.h" 23718292f2SDouglas Gregor #include "llvm/Support/Host.h" 245257fc63SDouglas Gregor #include "llvm/Support/PathV2.h" 25718292f2SDouglas Gregor #include "llvm/Support/raw_ostream.h" 26718292f2SDouglas Gregor #include "llvm/ADT/StringRef.h" 27718292f2SDouglas Gregor #include "llvm/ADT/StringSwitch.h" 28718292f2SDouglas Gregor using namespace clang; 29718292f2SDouglas Gregor 30718292f2SDouglas Gregor //----------------------------------------------------------------------------// 31718292f2SDouglas Gregor // Module 32718292f2SDouglas Gregor //----------------------------------------------------------------------------// 33718292f2SDouglas Gregor 34718292f2SDouglas Gregor std::string ModuleMap::Module::getFullModuleName() const { 35718292f2SDouglas Gregor llvm::SmallVector<StringRef, 2> Names; 36718292f2SDouglas Gregor 37718292f2SDouglas Gregor // Build up the set of module names (from innermost to outermost). 38718292f2SDouglas Gregor for (const Module *M = this; M; M = M->Parent) 39718292f2SDouglas Gregor Names.push_back(M->Name); 40718292f2SDouglas Gregor 41718292f2SDouglas Gregor std::string Result; 42718292f2SDouglas Gregor for (llvm::SmallVector<StringRef, 2>::reverse_iterator I = Names.rbegin(), 43718292f2SDouglas Gregor IEnd = Names.rend(); 44718292f2SDouglas Gregor I != IEnd; ++I) { 45718292f2SDouglas Gregor if (!Result.empty()) 46718292f2SDouglas Gregor Result += '.'; 47718292f2SDouglas Gregor 48718292f2SDouglas Gregor Result += *I; 49718292f2SDouglas Gregor } 50718292f2SDouglas Gregor 51718292f2SDouglas Gregor return Result; 52718292f2SDouglas Gregor } 53718292f2SDouglas Gregor 54ab0c8a84SDouglas Gregor StringRef ModuleMap::Module::getTopLevelModuleName() const { 55ab0c8a84SDouglas Gregor const Module *Top = this; 56ab0c8a84SDouglas Gregor while (Top->Parent) 57ab0c8a84SDouglas Gregor Top = Top->Parent; 58ab0c8a84SDouglas Gregor 59ab0c8a84SDouglas Gregor return Top->Name; 60ab0c8a84SDouglas Gregor } 61ab0c8a84SDouglas Gregor 62718292f2SDouglas Gregor //----------------------------------------------------------------------------// 63718292f2SDouglas Gregor // Module map 64718292f2SDouglas Gregor //----------------------------------------------------------------------------// 65718292f2SDouglas Gregor 66718292f2SDouglas Gregor ModuleMap::ModuleMap(FileManager &FileMgr, const DiagnosticConsumer &DC) { 67718292f2SDouglas Gregor llvm::IntrusiveRefCntPtr<DiagnosticIDs> DiagIDs(new DiagnosticIDs); 68718292f2SDouglas Gregor Diags = llvm::IntrusiveRefCntPtr<DiagnosticsEngine>( 69718292f2SDouglas Gregor new DiagnosticsEngine(DiagIDs)); 70718292f2SDouglas Gregor Diags->setClient(DC.clone(*Diags), /*ShouldOwnClient=*/true); 71718292f2SDouglas Gregor SourceMgr = new SourceManager(*Diags, FileMgr); 72718292f2SDouglas Gregor } 73718292f2SDouglas Gregor 74718292f2SDouglas Gregor ModuleMap::~ModuleMap() { 75718292f2SDouglas Gregor delete SourceMgr; 76718292f2SDouglas Gregor } 77718292f2SDouglas Gregor 78ab0c8a84SDouglas Gregor ModuleMap::Module *ModuleMap::findModuleForHeader(const FileEntry *File) { 79ab0c8a84SDouglas Gregor llvm::DenseMap<const FileEntry *, Module *>::iterator Known 80ab0c8a84SDouglas Gregor = Headers.find(File); 81ab0c8a84SDouglas Gregor if (Known != Headers.end()) 82ab0c8a84SDouglas Gregor return Known->second; 83ab0c8a84SDouglas Gregor 84ab0c8a84SDouglas Gregor return 0; 85ab0c8a84SDouglas Gregor } 86ab0c8a84SDouglas Gregor 87*88bdfb0eSDouglas Gregor ModuleMap::Module *ModuleMap::findModule(StringRef Name) { 88*88bdfb0eSDouglas Gregor llvm::StringMap<Module *>::iterator Known = Modules.find(Name); 89*88bdfb0eSDouglas Gregor if (Known != Modules.end()) 90*88bdfb0eSDouglas Gregor return Known->getValue(); 91*88bdfb0eSDouglas Gregor 92*88bdfb0eSDouglas Gregor return 0; 93*88bdfb0eSDouglas Gregor } 94*88bdfb0eSDouglas Gregor 95718292f2SDouglas Gregor static void indent(llvm::raw_ostream &OS, unsigned Spaces) { 96718292f2SDouglas Gregor OS << std::string(' ', Spaces); 97718292f2SDouglas Gregor } 98718292f2SDouglas Gregor 99718292f2SDouglas Gregor static void dumpModule(llvm::raw_ostream &OS, ModuleMap::Module *M, 100718292f2SDouglas Gregor unsigned Indent) { 101718292f2SDouglas Gregor indent(OS, Indent); 102718292f2SDouglas Gregor if (M->IsExplicit) 103718292f2SDouglas Gregor OS << "explicit "; 104718292f2SDouglas Gregor OS << M->Name << " {\n"; 105718292f2SDouglas Gregor 106718292f2SDouglas Gregor if (M->UmbrellaHeader) { 107718292f2SDouglas Gregor indent(OS, Indent + 2); 108718292f2SDouglas Gregor OS << "umbrella \"" << M->UmbrellaHeader->getName() << "\"\n"; 109718292f2SDouglas Gregor } 110718292f2SDouglas Gregor 111718292f2SDouglas Gregor for (unsigned I = 0, N = M->Headers.size(); I != N; ++I) { 112718292f2SDouglas Gregor indent(OS, Indent + 2); 113718292f2SDouglas Gregor OS << "header \"" << M->Headers[I]->getName() << "\"\n"; 114718292f2SDouglas Gregor } 115718292f2SDouglas Gregor 116718292f2SDouglas Gregor for (llvm::StringMap<ModuleMap::Module *>::iterator 117718292f2SDouglas Gregor MI = M->SubModules.begin(), 118718292f2SDouglas Gregor MIEnd = M->SubModules.end(); 119718292f2SDouglas Gregor MI != MIEnd; ++MI) 120718292f2SDouglas Gregor dumpModule(llvm::errs(), MI->getValue(), Indent + 2); 121718292f2SDouglas Gregor 122718292f2SDouglas Gregor indent(OS, Indent); 123718292f2SDouglas Gregor OS << "}\n"; 124718292f2SDouglas Gregor } 125718292f2SDouglas Gregor 126718292f2SDouglas Gregor void ModuleMap::dump() { 127718292f2SDouglas Gregor llvm::errs() << "Modules:"; 128718292f2SDouglas Gregor for (llvm::StringMap<Module *>::iterator M = Modules.begin(), 129718292f2SDouglas Gregor MEnd = Modules.end(); 130718292f2SDouglas Gregor M != MEnd; ++M) 131718292f2SDouglas Gregor dumpModule(llvm::errs(), M->getValue(), 2); 132718292f2SDouglas Gregor 133718292f2SDouglas Gregor llvm::errs() << "Headers:"; 134718292f2SDouglas Gregor for (llvm::DenseMap<const FileEntry *, Module *>::iterator 135718292f2SDouglas Gregor H = Headers.begin(), 136718292f2SDouglas Gregor HEnd = Headers.end(); 137718292f2SDouglas Gregor H != HEnd; ++H) { 138718292f2SDouglas Gregor llvm::errs() << " \"" << H->first->getName() << "\" -> " 139718292f2SDouglas Gregor << H->second->getFullModuleName() << "\n"; 140718292f2SDouglas Gregor } 141718292f2SDouglas Gregor } 142718292f2SDouglas Gregor 143718292f2SDouglas Gregor //----------------------------------------------------------------------------// 144718292f2SDouglas Gregor // Module map file parser 145718292f2SDouglas Gregor //----------------------------------------------------------------------------// 146718292f2SDouglas Gregor 147718292f2SDouglas Gregor namespace clang { 148718292f2SDouglas Gregor /// \brief A token in a module map file. 149718292f2SDouglas Gregor struct MMToken { 150718292f2SDouglas Gregor enum TokenKind { 151718292f2SDouglas Gregor EndOfFile, 152718292f2SDouglas Gregor HeaderKeyword, 153718292f2SDouglas Gregor Identifier, 154718292f2SDouglas Gregor ExplicitKeyword, 155718292f2SDouglas Gregor ModuleKeyword, 156718292f2SDouglas Gregor UmbrellaKeyword, 157718292f2SDouglas Gregor StringLiteral, 158718292f2SDouglas Gregor LBrace, 159718292f2SDouglas Gregor RBrace 160718292f2SDouglas Gregor } Kind; 161718292f2SDouglas Gregor 162718292f2SDouglas Gregor unsigned Location; 163718292f2SDouglas Gregor unsigned StringLength; 164718292f2SDouglas Gregor const char *StringData; 165718292f2SDouglas Gregor 166718292f2SDouglas Gregor void clear() { 167718292f2SDouglas Gregor Kind = EndOfFile; 168718292f2SDouglas Gregor Location = 0; 169718292f2SDouglas Gregor StringLength = 0; 170718292f2SDouglas Gregor StringData = 0; 171718292f2SDouglas Gregor } 172718292f2SDouglas Gregor 173718292f2SDouglas Gregor bool is(TokenKind K) const { return Kind == K; } 174718292f2SDouglas Gregor 175718292f2SDouglas Gregor SourceLocation getLocation() const { 176718292f2SDouglas Gregor return SourceLocation::getFromRawEncoding(Location); 177718292f2SDouglas Gregor } 178718292f2SDouglas Gregor 179718292f2SDouglas Gregor StringRef getString() const { 180718292f2SDouglas Gregor return StringRef(StringData, StringLength); 181718292f2SDouglas Gregor } 182718292f2SDouglas Gregor }; 183718292f2SDouglas Gregor 184718292f2SDouglas Gregor class ModuleMapParser { 185718292f2SDouglas Gregor Lexer &L; 186718292f2SDouglas Gregor SourceManager &SourceMgr; 187718292f2SDouglas Gregor DiagnosticsEngine &Diags; 188718292f2SDouglas Gregor ModuleMap ⤅ 189718292f2SDouglas Gregor 1905257fc63SDouglas Gregor /// \brief The directory that this module map resides in. 1915257fc63SDouglas Gregor const DirectoryEntry *Directory; 1925257fc63SDouglas Gregor 193718292f2SDouglas Gregor /// \brief Whether an error occurred. 194718292f2SDouglas Gregor bool HadError; 195718292f2SDouglas Gregor 196718292f2SDouglas Gregor /// \brief Default target information, used only for string literal 197718292f2SDouglas Gregor /// parsing. 198718292f2SDouglas Gregor TargetInfo *Target; 199718292f2SDouglas Gregor 200718292f2SDouglas Gregor /// \brief Stores string data for the various string literals referenced 201718292f2SDouglas Gregor /// during parsing. 202718292f2SDouglas Gregor llvm::BumpPtrAllocator StringData; 203718292f2SDouglas Gregor 204718292f2SDouglas Gregor /// \brief The current token. 205718292f2SDouglas Gregor MMToken Tok; 206718292f2SDouglas Gregor 207718292f2SDouglas Gregor /// \brief The active module. 208718292f2SDouglas Gregor ModuleMap::Module *ActiveModule; 209718292f2SDouglas Gregor 210718292f2SDouglas Gregor /// \brief Consume the current token and return its location. 211718292f2SDouglas Gregor SourceLocation consumeToken(); 212718292f2SDouglas Gregor 213718292f2SDouglas Gregor /// \brief Skip tokens until we reach the a token with the given kind 214718292f2SDouglas Gregor /// (or the end of the file). 215718292f2SDouglas Gregor void skipUntil(MMToken::TokenKind K); 216718292f2SDouglas Gregor 217718292f2SDouglas Gregor void parseModuleDecl(); 218718292f2SDouglas Gregor void parseUmbrellaDecl(); 219718292f2SDouglas Gregor void parseHeaderDecl(); 220718292f2SDouglas Gregor 221718292f2SDouglas Gregor public: 222718292f2SDouglas Gregor typedef ModuleMap::Module Module; 223718292f2SDouglas Gregor 224718292f2SDouglas Gregor explicit ModuleMapParser(Lexer &L, SourceManager &SourceMgr, 225718292f2SDouglas Gregor DiagnosticsEngine &Diags, 2265257fc63SDouglas Gregor ModuleMap &Map, 2275257fc63SDouglas Gregor const DirectoryEntry *Directory) 2285257fc63SDouglas Gregor : L(L), SourceMgr(SourceMgr), Diags(Diags), Map(Map), 2295257fc63SDouglas Gregor Directory(Directory), HadError(false), ActiveModule(0) 230718292f2SDouglas Gregor { 231718292f2SDouglas Gregor TargetOptions TargetOpts; 232718292f2SDouglas Gregor TargetOpts.Triple = llvm::sys::getDefaultTargetTriple(); 233718292f2SDouglas Gregor Target = TargetInfo::CreateTargetInfo(Diags, TargetOpts); 234718292f2SDouglas Gregor 235718292f2SDouglas Gregor Tok.clear(); 236718292f2SDouglas Gregor consumeToken(); 237718292f2SDouglas Gregor } 238718292f2SDouglas Gregor 239718292f2SDouglas Gregor bool parseModuleMapFile(); 240718292f2SDouglas Gregor }; 241718292f2SDouglas Gregor } 242718292f2SDouglas Gregor 243718292f2SDouglas Gregor SourceLocation ModuleMapParser::consumeToken() { 244718292f2SDouglas Gregor retry: 245718292f2SDouglas Gregor SourceLocation Result = Tok.getLocation(); 246718292f2SDouglas Gregor Tok.clear(); 247718292f2SDouglas Gregor 248718292f2SDouglas Gregor Token LToken; 249718292f2SDouglas Gregor L.LexFromRawLexer(LToken); 250718292f2SDouglas Gregor Tok.Location = LToken.getLocation().getRawEncoding(); 251718292f2SDouglas Gregor switch (LToken.getKind()) { 252718292f2SDouglas Gregor case tok::raw_identifier: 253718292f2SDouglas Gregor Tok.StringData = LToken.getRawIdentifierData(); 254718292f2SDouglas Gregor Tok.StringLength = LToken.getLength(); 255718292f2SDouglas Gregor Tok.Kind = llvm::StringSwitch<MMToken::TokenKind>(Tok.getString()) 256718292f2SDouglas Gregor .Case("header", MMToken::HeaderKeyword) 257718292f2SDouglas Gregor .Case("explicit", MMToken::ExplicitKeyword) 258718292f2SDouglas Gregor .Case("module", MMToken::ModuleKeyword) 259718292f2SDouglas Gregor .Case("umbrella", MMToken::UmbrellaKeyword) 260718292f2SDouglas Gregor .Default(MMToken::Identifier); 261718292f2SDouglas Gregor break; 262718292f2SDouglas Gregor 263718292f2SDouglas Gregor case tok::eof: 264718292f2SDouglas Gregor Tok.Kind = MMToken::EndOfFile; 265718292f2SDouglas Gregor break; 266718292f2SDouglas Gregor 267718292f2SDouglas Gregor case tok::l_brace: 268718292f2SDouglas Gregor Tok.Kind = MMToken::LBrace; 269718292f2SDouglas Gregor break; 270718292f2SDouglas Gregor 271718292f2SDouglas Gregor case tok::r_brace: 272718292f2SDouglas Gregor Tok.Kind = MMToken::RBrace; 273718292f2SDouglas Gregor break; 274718292f2SDouglas Gregor 275718292f2SDouglas Gregor case tok::string_literal: { 276718292f2SDouglas Gregor // Parse the string literal. 277718292f2SDouglas Gregor LangOptions LangOpts; 278718292f2SDouglas Gregor StringLiteralParser StringLiteral(<oken, 1, SourceMgr, LangOpts, *Target); 279718292f2SDouglas Gregor if (StringLiteral.hadError) 280718292f2SDouglas Gregor goto retry; 281718292f2SDouglas Gregor 282718292f2SDouglas Gregor // Copy the string literal into our string data allocator. 283718292f2SDouglas Gregor unsigned Length = StringLiteral.GetStringLength(); 284718292f2SDouglas Gregor char *Saved = StringData.Allocate<char>(Length + 1); 285718292f2SDouglas Gregor memcpy(Saved, StringLiteral.GetString().data(), Length); 286718292f2SDouglas Gregor Saved[Length] = 0; 287718292f2SDouglas Gregor 288718292f2SDouglas Gregor // Form the token. 289718292f2SDouglas Gregor Tok.Kind = MMToken::StringLiteral; 290718292f2SDouglas Gregor Tok.StringData = Saved; 291718292f2SDouglas Gregor Tok.StringLength = Length; 292718292f2SDouglas Gregor break; 293718292f2SDouglas Gregor } 294718292f2SDouglas Gregor 295718292f2SDouglas Gregor case tok::comment: 296718292f2SDouglas Gregor goto retry; 297718292f2SDouglas Gregor 298718292f2SDouglas Gregor default: 299718292f2SDouglas Gregor Diags.Report(LToken.getLocation(), diag::err_mmap_unknown_token); 300718292f2SDouglas Gregor HadError = true; 301718292f2SDouglas Gregor goto retry; 302718292f2SDouglas Gregor } 303718292f2SDouglas Gregor 304718292f2SDouglas Gregor return Result; 305718292f2SDouglas Gregor } 306718292f2SDouglas Gregor 307718292f2SDouglas Gregor void ModuleMapParser::skipUntil(MMToken::TokenKind K) { 308718292f2SDouglas Gregor unsigned braceDepth = 0; 309718292f2SDouglas Gregor do { 310718292f2SDouglas Gregor switch (Tok.Kind) { 311718292f2SDouglas Gregor case MMToken::EndOfFile: 312718292f2SDouglas Gregor return; 313718292f2SDouglas Gregor 314718292f2SDouglas Gregor case MMToken::LBrace: 315718292f2SDouglas Gregor if (Tok.is(K) && braceDepth == 0) 316718292f2SDouglas Gregor return; 317718292f2SDouglas Gregor 318718292f2SDouglas Gregor ++braceDepth; 319718292f2SDouglas Gregor break; 320718292f2SDouglas Gregor 321718292f2SDouglas Gregor case MMToken::RBrace: 322718292f2SDouglas Gregor if (braceDepth > 0) 323718292f2SDouglas Gregor --braceDepth; 324718292f2SDouglas Gregor else if (Tok.is(K)) 325718292f2SDouglas Gregor return; 326718292f2SDouglas Gregor break; 327718292f2SDouglas Gregor 328718292f2SDouglas Gregor default: 329718292f2SDouglas Gregor if (braceDepth == 0 && Tok.is(K)) 330718292f2SDouglas Gregor return; 331718292f2SDouglas Gregor break; 332718292f2SDouglas Gregor } 333718292f2SDouglas Gregor 334718292f2SDouglas Gregor consumeToken(); 335718292f2SDouglas Gregor } while (true); 336718292f2SDouglas Gregor } 337718292f2SDouglas Gregor 338718292f2SDouglas Gregor /// \brief Parse a module declaration. 339718292f2SDouglas Gregor /// 340718292f2SDouglas Gregor /// module-declaration: 341718292f2SDouglas Gregor /// 'module' identifier { module-member* } 342718292f2SDouglas Gregor /// 343718292f2SDouglas Gregor /// module-member: 344718292f2SDouglas Gregor /// umbrella-declaration 345718292f2SDouglas Gregor /// header-declaration 346718292f2SDouglas Gregor /// 'explicit'[opt] module-declaration 347718292f2SDouglas Gregor void ModuleMapParser::parseModuleDecl() { 348718292f2SDouglas Gregor assert(Tok.is(MMToken::ExplicitKeyword) || Tok.is(MMToken::ModuleKeyword)); 349718292f2SDouglas Gregor 350718292f2SDouglas Gregor // Parse 'explicit' keyword, if present. 351718292f2SDouglas Gregor bool Explicit = false; 352718292f2SDouglas Gregor if (Tok.is(MMToken::ExplicitKeyword)) { 353718292f2SDouglas Gregor consumeToken(); 354718292f2SDouglas Gregor Explicit = true; 355718292f2SDouglas Gregor } 356718292f2SDouglas Gregor 357718292f2SDouglas Gregor // Parse 'module' keyword. 358718292f2SDouglas Gregor if (!Tok.is(MMToken::ModuleKeyword)) { 359718292f2SDouglas Gregor Diags.Report(Tok.getLocation(), 360718292f2SDouglas Gregor diag::err_mmap_expected_module_after_explicit); 361718292f2SDouglas Gregor consumeToken(); 362718292f2SDouglas Gregor HadError = true; 363718292f2SDouglas Gregor return; 364718292f2SDouglas Gregor } 365718292f2SDouglas Gregor consumeToken(); // 'module' keyword 366718292f2SDouglas Gregor 367718292f2SDouglas Gregor // Parse the module name. 368718292f2SDouglas Gregor if (!Tok.is(MMToken::Identifier)) { 369718292f2SDouglas Gregor Diags.Report(Tok.getLocation(), diag::err_mmap_expected_module_name); 370718292f2SDouglas Gregor HadError = true; 371718292f2SDouglas Gregor return; 372718292f2SDouglas Gregor } 373718292f2SDouglas Gregor StringRef ModuleName = Tok.getString(); 374718292f2SDouglas Gregor SourceLocation ModuleNameLoc = consumeToken(); 375718292f2SDouglas Gregor 376718292f2SDouglas Gregor // Parse the opening brace. 377718292f2SDouglas Gregor if (!Tok.is(MMToken::LBrace)) { 378718292f2SDouglas Gregor Diags.Report(Tok.getLocation(), diag::err_mmap_expected_lbrace) 379718292f2SDouglas Gregor << ModuleName; 380718292f2SDouglas Gregor HadError = true; 381718292f2SDouglas Gregor return; 382718292f2SDouglas Gregor } 383718292f2SDouglas Gregor SourceLocation LBraceLoc = consumeToken(); 384718292f2SDouglas Gregor 385718292f2SDouglas Gregor // Determine whether this (sub)module has already been defined. 386718292f2SDouglas Gregor llvm::StringMap<Module *> &ModuleSpace 387718292f2SDouglas Gregor = ActiveModule? ActiveModule->SubModules : Map.Modules; 388718292f2SDouglas Gregor llvm::StringMap<Module *>::iterator ExistingModule 389718292f2SDouglas Gregor = ModuleSpace.find(ModuleName); 390718292f2SDouglas Gregor if (ExistingModule != ModuleSpace.end()) { 391718292f2SDouglas Gregor Diags.Report(ModuleNameLoc, diag::err_mmap_module_redefinition) 392718292f2SDouglas Gregor << ModuleName; 393718292f2SDouglas Gregor Diags.Report(ExistingModule->getValue()->DefinitionLoc, 394718292f2SDouglas Gregor diag::note_mmap_prev_definition); 395718292f2SDouglas Gregor 396718292f2SDouglas Gregor // Skip the module definition. 397718292f2SDouglas Gregor skipUntil(MMToken::RBrace); 398718292f2SDouglas Gregor if (Tok.is(MMToken::RBrace)) 399718292f2SDouglas Gregor consumeToken(); 400718292f2SDouglas Gregor 401718292f2SDouglas Gregor HadError = true; 402718292f2SDouglas Gregor return; 403718292f2SDouglas Gregor } 404718292f2SDouglas Gregor 405718292f2SDouglas Gregor // Start defining this module. 406718292f2SDouglas Gregor ActiveModule = new Module(ModuleName, ModuleNameLoc, ActiveModule, Explicit); 407718292f2SDouglas Gregor ModuleSpace[ModuleName] = ActiveModule; 408718292f2SDouglas Gregor 409718292f2SDouglas Gregor bool Done = false; 410718292f2SDouglas Gregor do { 411718292f2SDouglas Gregor switch (Tok.Kind) { 412718292f2SDouglas Gregor case MMToken::EndOfFile: 413718292f2SDouglas Gregor case MMToken::RBrace: 414718292f2SDouglas Gregor Done = true; 415718292f2SDouglas Gregor break; 416718292f2SDouglas Gregor 417718292f2SDouglas Gregor case MMToken::ExplicitKeyword: 418718292f2SDouglas Gregor case MMToken::ModuleKeyword: 419718292f2SDouglas Gregor parseModuleDecl(); 420718292f2SDouglas Gregor break; 421718292f2SDouglas Gregor 422718292f2SDouglas Gregor case MMToken::HeaderKeyword: 423718292f2SDouglas Gregor parseHeaderDecl(); 424718292f2SDouglas Gregor break; 425718292f2SDouglas Gregor 426718292f2SDouglas Gregor case MMToken::UmbrellaKeyword: 427718292f2SDouglas Gregor parseUmbrellaDecl(); 428718292f2SDouglas Gregor break; 429718292f2SDouglas Gregor 430718292f2SDouglas Gregor default: 431718292f2SDouglas Gregor Diags.Report(Tok.getLocation(), diag::err_mmap_expected_member); 432718292f2SDouglas Gregor consumeToken(); 433718292f2SDouglas Gregor break; 434718292f2SDouglas Gregor } 435718292f2SDouglas Gregor } while (!Done); 436718292f2SDouglas Gregor 437718292f2SDouglas Gregor if (Tok.is(MMToken::RBrace)) 438718292f2SDouglas Gregor consumeToken(); 439718292f2SDouglas Gregor else { 440718292f2SDouglas Gregor Diags.Report(Tok.getLocation(), diag::err_mmap_expected_rbrace); 441718292f2SDouglas Gregor Diags.Report(LBraceLoc, diag::note_mmap_lbrace_match); 442718292f2SDouglas Gregor HadError = true; 443718292f2SDouglas Gregor } 444718292f2SDouglas Gregor 445718292f2SDouglas Gregor // We're done parsing this module. Pop back to our parent scope. 446718292f2SDouglas Gregor ActiveModule = ActiveModule->Parent; 447718292f2SDouglas Gregor } 448718292f2SDouglas Gregor 449718292f2SDouglas Gregor /// \brief Parse an umbrella header declaration. 450718292f2SDouglas Gregor /// 451718292f2SDouglas Gregor /// umbrella-declaration: 452718292f2SDouglas Gregor /// 'umbrella' string-literal 453718292f2SDouglas Gregor void ModuleMapParser::parseUmbrellaDecl() { 454718292f2SDouglas Gregor assert(Tok.is(MMToken::UmbrellaKeyword)); 455718292f2SDouglas Gregor SourceLocation UmbrellaLoc = consumeToken(); 456718292f2SDouglas Gregor 457718292f2SDouglas Gregor // Parse the header name. 458718292f2SDouglas Gregor if (!Tok.is(MMToken::StringLiteral)) { 459718292f2SDouglas Gregor Diags.Report(Tok.getLocation(), diag::err_mmap_expected_header) 460718292f2SDouglas Gregor << "umbrella"; 461718292f2SDouglas Gregor HadError = true; 462718292f2SDouglas Gregor return; 463718292f2SDouglas Gregor } 464718292f2SDouglas Gregor StringRef FileName = Tok.getString(); 465718292f2SDouglas Gregor SourceLocation FileNameLoc = consumeToken(); 466718292f2SDouglas Gregor 4675257fc63SDouglas Gregor // Check whether we already have an umbrella header. 4685257fc63SDouglas Gregor if (ActiveModule->UmbrellaHeader) { 4695257fc63SDouglas Gregor Diags.Report(FileNameLoc, diag::err_mmap_umbrella_header_conflict) 4705257fc63SDouglas Gregor << ActiveModule->getFullModuleName() 4715257fc63SDouglas Gregor << ActiveModule->UmbrellaHeader->getName(); 4725257fc63SDouglas Gregor HadError = true; 4735257fc63SDouglas Gregor return; 4745257fc63SDouglas Gregor } 4755257fc63SDouglas Gregor 4765257fc63SDouglas Gregor // Only top-level modules can have umbrella headers. 4775257fc63SDouglas Gregor if (ActiveModule->Parent) { 4785257fc63SDouglas Gregor Diags.Report(UmbrellaLoc, diag::err_mmap_umbrella_header_submodule) 4795257fc63SDouglas Gregor << ActiveModule->getFullModuleName(); 4805257fc63SDouglas Gregor HadError = true; 4815257fc63SDouglas Gregor return; 4825257fc63SDouglas Gregor } 4835257fc63SDouglas Gregor 4845257fc63SDouglas Gregor // Look for this file. 4855257fc63SDouglas Gregor llvm::SmallString<128> PathName; 4865257fc63SDouglas Gregor PathName += Directory->getName(); 4875257fc63SDouglas Gregor llvm::sys::path::append(PathName, FileName); 4885257fc63SDouglas Gregor 4895257fc63SDouglas Gregor // FIXME: We shouldn't be eagerly stat'ing every file named in a module map. 4905257fc63SDouglas Gregor // Come up with a lazy way to do this. 4915257fc63SDouglas Gregor if (const FileEntry *File = SourceMgr.getFileManager().getFile(PathName)) { 4925257fc63SDouglas Gregor if (const Module *OwningModule = Map.Headers[File]) { 4935257fc63SDouglas Gregor Diags.Report(FileNameLoc, diag::err_mmap_header_conflict) 4945257fc63SDouglas Gregor << FileName << OwningModule->getFullModuleName(); 4955257fc63SDouglas Gregor HadError = true; 4965257fc63SDouglas Gregor } else { 4975257fc63SDouglas Gregor // Record this umbrella header. 4985257fc63SDouglas Gregor ActiveModule->UmbrellaHeader = File; 4995257fc63SDouglas Gregor Map.Headers[File] = ActiveModule; 5005257fc63SDouglas Gregor } 5015257fc63SDouglas Gregor } else { 5025257fc63SDouglas Gregor Diags.Report(FileNameLoc, diag::err_mmap_header_not_found) 5035257fc63SDouglas Gregor << true << FileName; 5045257fc63SDouglas Gregor HadError = true; 5055257fc63SDouglas Gregor } 506718292f2SDouglas Gregor } 507718292f2SDouglas Gregor 508718292f2SDouglas Gregor /// \brief Parse a header declaration. 509718292f2SDouglas Gregor /// 510718292f2SDouglas Gregor /// header-declaration: 511718292f2SDouglas Gregor /// 'header' string-literal 512718292f2SDouglas Gregor void ModuleMapParser::parseHeaderDecl() { 513718292f2SDouglas Gregor assert(Tok.is(MMToken::HeaderKeyword)); 514718292f2SDouglas Gregor SourceLocation HeaderLoc = consumeToken(); 515718292f2SDouglas Gregor 516718292f2SDouglas Gregor // Parse the header name. 517718292f2SDouglas Gregor if (!Tok.is(MMToken::StringLiteral)) { 518718292f2SDouglas Gregor Diags.Report(Tok.getLocation(), diag::err_mmap_expected_header) 519718292f2SDouglas Gregor << "header"; 520718292f2SDouglas Gregor HadError = true; 521718292f2SDouglas Gregor return; 522718292f2SDouglas Gregor } 523718292f2SDouglas Gregor StringRef FileName = Tok.getString(); 524718292f2SDouglas Gregor SourceLocation FileNameLoc = consumeToken(); 525718292f2SDouglas Gregor 5265257fc63SDouglas Gregor // Look for this file. 5275257fc63SDouglas Gregor llvm::SmallString<128> PathName; 5285257fc63SDouglas Gregor PathName += Directory->getName(); 5295257fc63SDouglas Gregor llvm::sys::path::append(PathName, FileName); 5305257fc63SDouglas Gregor 5315257fc63SDouglas Gregor // FIXME: We shouldn't be eagerly stat'ing every file named in a module map. 5325257fc63SDouglas Gregor // Come up with a lazy way to do this. 5335257fc63SDouglas Gregor if (const FileEntry *File = SourceMgr.getFileManager().getFile(PathName)) { 5345257fc63SDouglas Gregor if (const Module *OwningModule = Map.Headers[File]) { 5355257fc63SDouglas Gregor Diags.Report(FileNameLoc, diag::err_mmap_header_conflict) 5365257fc63SDouglas Gregor << FileName << OwningModule->getFullModuleName(); 5375257fc63SDouglas Gregor HadError = true; 5385257fc63SDouglas Gregor } else { 5395257fc63SDouglas Gregor // Record this file. 5405257fc63SDouglas Gregor ActiveModule->Headers.push_back(File); 5415257fc63SDouglas Gregor Map.Headers[File] = ActiveModule; 5425257fc63SDouglas Gregor } 5435257fc63SDouglas Gregor } else { 5445257fc63SDouglas Gregor Diags.Report(FileNameLoc, diag::err_mmap_header_not_found) 5455257fc63SDouglas Gregor << false << FileName; 5465257fc63SDouglas Gregor HadError = true; 5475257fc63SDouglas Gregor } 548718292f2SDouglas Gregor } 549718292f2SDouglas Gregor 550718292f2SDouglas Gregor /// \brief Parse a module map file. 551718292f2SDouglas Gregor /// 552718292f2SDouglas Gregor /// module-map-file: 553718292f2SDouglas Gregor /// module-declaration* 554718292f2SDouglas Gregor bool ModuleMapParser::parseModuleMapFile() { 555718292f2SDouglas Gregor do { 556718292f2SDouglas Gregor switch (Tok.Kind) { 557718292f2SDouglas Gregor case MMToken::EndOfFile: 558718292f2SDouglas Gregor return HadError; 559718292f2SDouglas Gregor 560718292f2SDouglas Gregor case MMToken::ModuleKeyword: 561718292f2SDouglas Gregor parseModuleDecl(); 562718292f2SDouglas Gregor break; 563718292f2SDouglas Gregor 564718292f2SDouglas Gregor case MMToken::ExplicitKeyword: 565718292f2SDouglas Gregor case MMToken::HeaderKeyword: 566718292f2SDouglas Gregor case MMToken::Identifier: 567718292f2SDouglas Gregor case MMToken::LBrace: 568718292f2SDouglas Gregor case MMToken::RBrace: 569718292f2SDouglas Gregor case MMToken::StringLiteral: 570718292f2SDouglas Gregor case MMToken::UmbrellaKeyword: 571718292f2SDouglas Gregor Diags.Report(Tok.getLocation(), diag::err_mmap_expected_module); 572718292f2SDouglas Gregor HadError = true; 573718292f2SDouglas Gregor consumeToken(); 574718292f2SDouglas Gregor break; 575718292f2SDouglas Gregor } 576718292f2SDouglas Gregor } while (true); 577718292f2SDouglas Gregor 578718292f2SDouglas Gregor return HadError; 579718292f2SDouglas Gregor } 580718292f2SDouglas Gregor 581718292f2SDouglas Gregor bool ModuleMap::parseModuleMapFile(const FileEntry *File) { 582718292f2SDouglas Gregor FileID ID = SourceMgr->createFileID(File, SourceLocation(), SrcMgr::C_User); 583718292f2SDouglas Gregor const llvm::MemoryBuffer *Buffer = SourceMgr->getBuffer(ID); 584718292f2SDouglas Gregor if (!Buffer) 585718292f2SDouglas Gregor return true; 586718292f2SDouglas Gregor 587718292f2SDouglas Gregor // Parse this module map file. 588718292f2SDouglas Gregor Lexer L(ID, SourceMgr->getBuffer(ID), *SourceMgr, LangOpts); 589718292f2SDouglas Gregor Diags->getClient()->BeginSourceFile(LangOpts); 5905257fc63SDouglas Gregor ModuleMapParser Parser(L, *SourceMgr, *Diags, *this, File->getDir()); 591718292f2SDouglas Gregor bool Result = Parser.parseModuleMapFile(); 592718292f2SDouglas Gregor Diags->getClient()->EndSourceFile(); 593718292f2SDouglas Gregor 594718292f2SDouglas Gregor return Result; 595718292f2SDouglas Gregor } 596