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 84*b65dbfffSDouglas Gregor const DirectoryEntry *Dir = File->getDir(); 85*b65dbfffSDouglas Gregor llvm::DenseMap<const DirectoryEntry *, Module *>::iterator KnownDir 86*b65dbfffSDouglas Gregor = UmbrellaDirs.find(Dir); 87*b65dbfffSDouglas Gregor if (KnownDir != UmbrellaDirs.end()) 88*b65dbfffSDouglas Gregor return KnownDir->second; 89*b65dbfffSDouglas Gregor 90*b65dbfffSDouglas Gregor // Walk up the directory hierarchy looking for umbrella headers. 91*b65dbfffSDouglas Gregor llvm::SmallVector<const DirectoryEntry *, 2> SkippedDirs; 92*b65dbfffSDouglas Gregor StringRef DirName = Dir->getName(); 93*b65dbfffSDouglas Gregor do { 94*b65dbfffSDouglas Gregor // Retrieve our parent path. 95*b65dbfffSDouglas Gregor DirName = llvm::sys::path::parent_path(DirName); 96*b65dbfffSDouglas Gregor if (DirName.empty()) 97*b65dbfffSDouglas Gregor break; 98*b65dbfffSDouglas Gregor 99*b65dbfffSDouglas Gregor // Resolve the parent path to a directory entry. 100*b65dbfffSDouglas Gregor Dir = SourceMgr->getFileManager().getDirectory(DirName); 101*b65dbfffSDouglas Gregor if (!Dir) 102*b65dbfffSDouglas Gregor break; 103*b65dbfffSDouglas Gregor 104*b65dbfffSDouglas Gregor KnownDir = UmbrellaDirs.find(Dir); 105*b65dbfffSDouglas Gregor if (KnownDir != UmbrellaDirs.end()) { 106*b65dbfffSDouglas Gregor Module *Result = KnownDir->second; 107*b65dbfffSDouglas Gregor 108*b65dbfffSDouglas Gregor // Record each of the directories we stepped through as being part of 109*b65dbfffSDouglas Gregor // the module we found, since the umbrella header covers them all. 110*b65dbfffSDouglas Gregor for (unsigned I = 0, N = SkippedDirs.size(); I != N; ++I) 111*b65dbfffSDouglas Gregor UmbrellaDirs[SkippedDirs[I]] = Result; 112*b65dbfffSDouglas Gregor 113*b65dbfffSDouglas Gregor return Result; 114*b65dbfffSDouglas Gregor } 115*b65dbfffSDouglas Gregor 116*b65dbfffSDouglas Gregor SkippedDirs.push_back(Dir); 117*b65dbfffSDouglas Gregor } while (true); 118*b65dbfffSDouglas Gregor 119ab0c8a84SDouglas Gregor return 0; 120ab0c8a84SDouglas Gregor } 121ab0c8a84SDouglas Gregor 12288bdfb0eSDouglas Gregor ModuleMap::Module *ModuleMap::findModule(StringRef Name) { 12388bdfb0eSDouglas Gregor llvm::StringMap<Module *>::iterator Known = Modules.find(Name); 12488bdfb0eSDouglas Gregor if (Known != Modules.end()) 12588bdfb0eSDouglas Gregor return Known->getValue(); 12688bdfb0eSDouglas Gregor 12788bdfb0eSDouglas Gregor return 0; 12888bdfb0eSDouglas Gregor } 12988bdfb0eSDouglas Gregor 130718292f2SDouglas Gregor static void indent(llvm::raw_ostream &OS, unsigned Spaces) { 131718292f2SDouglas Gregor OS << std::string(' ', Spaces); 132718292f2SDouglas Gregor } 133718292f2SDouglas Gregor 134718292f2SDouglas Gregor static void dumpModule(llvm::raw_ostream &OS, ModuleMap::Module *M, 135718292f2SDouglas Gregor unsigned Indent) { 136718292f2SDouglas Gregor indent(OS, Indent); 137718292f2SDouglas Gregor if (M->IsExplicit) 138718292f2SDouglas Gregor OS << "explicit "; 139718292f2SDouglas Gregor OS << M->Name << " {\n"; 140718292f2SDouglas Gregor 141718292f2SDouglas Gregor if (M->UmbrellaHeader) { 142718292f2SDouglas Gregor indent(OS, Indent + 2); 143718292f2SDouglas Gregor OS << "umbrella \"" << M->UmbrellaHeader->getName() << "\"\n"; 144718292f2SDouglas Gregor } 145718292f2SDouglas Gregor 146718292f2SDouglas Gregor for (unsigned I = 0, N = M->Headers.size(); I != N; ++I) { 147718292f2SDouglas Gregor indent(OS, Indent + 2); 148718292f2SDouglas Gregor OS << "header \"" << M->Headers[I]->getName() << "\"\n"; 149718292f2SDouglas Gregor } 150718292f2SDouglas Gregor 151718292f2SDouglas Gregor for (llvm::StringMap<ModuleMap::Module *>::iterator 152718292f2SDouglas Gregor MI = M->SubModules.begin(), 153718292f2SDouglas Gregor MIEnd = M->SubModules.end(); 154718292f2SDouglas Gregor MI != MIEnd; ++MI) 155718292f2SDouglas Gregor dumpModule(llvm::errs(), MI->getValue(), Indent + 2); 156718292f2SDouglas Gregor 157718292f2SDouglas Gregor indent(OS, Indent); 158718292f2SDouglas Gregor OS << "}\n"; 159718292f2SDouglas Gregor } 160718292f2SDouglas Gregor 161718292f2SDouglas Gregor void ModuleMap::dump() { 162718292f2SDouglas Gregor llvm::errs() << "Modules:"; 163718292f2SDouglas Gregor for (llvm::StringMap<Module *>::iterator M = Modules.begin(), 164718292f2SDouglas Gregor MEnd = Modules.end(); 165718292f2SDouglas Gregor M != MEnd; ++M) 166718292f2SDouglas Gregor dumpModule(llvm::errs(), M->getValue(), 2); 167718292f2SDouglas Gregor 168718292f2SDouglas Gregor llvm::errs() << "Headers:"; 169718292f2SDouglas Gregor for (llvm::DenseMap<const FileEntry *, Module *>::iterator 170718292f2SDouglas Gregor H = Headers.begin(), 171718292f2SDouglas Gregor HEnd = Headers.end(); 172718292f2SDouglas Gregor H != HEnd; ++H) { 173718292f2SDouglas Gregor llvm::errs() << " \"" << H->first->getName() << "\" -> " 174718292f2SDouglas Gregor << H->second->getFullModuleName() << "\n"; 175718292f2SDouglas Gregor } 176718292f2SDouglas Gregor } 177718292f2SDouglas Gregor 178718292f2SDouglas Gregor //----------------------------------------------------------------------------// 179718292f2SDouglas Gregor // Module map file parser 180718292f2SDouglas Gregor //----------------------------------------------------------------------------// 181718292f2SDouglas Gregor 182718292f2SDouglas Gregor namespace clang { 183718292f2SDouglas Gregor /// \brief A token in a module map file. 184718292f2SDouglas Gregor struct MMToken { 185718292f2SDouglas Gregor enum TokenKind { 186718292f2SDouglas Gregor EndOfFile, 187718292f2SDouglas Gregor HeaderKeyword, 188718292f2SDouglas Gregor Identifier, 189718292f2SDouglas Gregor ExplicitKeyword, 190718292f2SDouglas Gregor ModuleKeyword, 191718292f2SDouglas Gregor UmbrellaKeyword, 192718292f2SDouglas Gregor StringLiteral, 193718292f2SDouglas Gregor LBrace, 194718292f2SDouglas Gregor RBrace 195718292f2SDouglas Gregor } Kind; 196718292f2SDouglas Gregor 197718292f2SDouglas Gregor unsigned Location; 198718292f2SDouglas Gregor unsigned StringLength; 199718292f2SDouglas Gregor const char *StringData; 200718292f2SDouglas Gregor 201718292f2SDouglas Gregor void clear() { 202718292f2SDouglas Gregor Kind = EndOfFile; 203718292f2SDouglas Gregor Location = 0; 204718292f2SDouglas Gregor StringLength = 0; 205718292f2SDouglas Gregor StringData = 0; 206718292f2SDouglas Gregor } 207718292f2SDouglas Gregor 208718292f2SDouglas Gregor bool is(TokenKind K) const { return Kind == K; } 209718292f2SDouglas Gregor 210718292f2SDouglas Gregor SourceLocation getLocation() const { 211718292f2SDouglas Gregor return SourceLocation::getFromRawEncoding(Location); 212718292f2SDouglas Gregor } 213718292f2SDouglas Gregor 214718292f2SDouglas Gregor StringRef getString() const { 215718292f2SDouglas Gregor return StringRef(StringData, StringLength); 216718292f2SDouglas Gregor } 217718292f2SDouglas Gregor }; 218718292f2SDouglas Gregor 219718292f2SDouglas Gregor class ModuleMapParser { 220718292f2SDouglas Gregor Lexer &L; 221718292f2SDouglas Gregor SourceManager &SourceMgr; 222718292f2SDouglas Gregor DiagnosticsEngine &Diags; 223718292f2SDouglas Gregor ModuleMap ⤅ 224718292f2SDouglas Gregor 2255257fc63SDouglas Gregor /// \brief The directory that this module map resides in. 2265257fc63SDouglas Gregor const DirectoryEntry *Directory; 2275257fc63SDouglas Gregor 228718292f2SDouglas Gregor /// \brief Whether an error occurred. 229718292f2SDouglas Gregor bool HadError; 230718292f2SDouglas Gregor 231718292f2SDouglas Gregor /// \brief Default target information, used only for string literal 232718292f2SDouglas Gregor /// parsing. 233718292f2SDouglas Gregor TargetInfo *Target; 234718292f2SDouglas Gregor 235718292f2SDouglas Gregor /// \brief Stores string data for the various string literals referenced 236718292f2SDouglas Gregor /// during parsing. 237718292f2SDouglas Gregor llvm::BumpPtrAllocator StringData; 238718292f2SDouglas Gregor 239718292f2SDouglas Gregor /// \brief The current token. 240718292f2SDouglas Gregor MMToken Tok; 241718292f2SDouglas Gregor 242718292f2SDouglas Gregor /// \brief The active module. 243718292f2SDouglas Gregor ModuleMap::Module *ActiveModule; 244718292f2SDouglas Gregor 245718292f2SDouglas Gregor /// \brief Consume the current token and return its location. 246718292f2SDouglas Gregor SourceLocation consumeToken(); 247718292f2SDouglas Gregor 248718292f2SDouglas Gregor /// \brief Skip tokens until we reach the a token with the given kind 249718292f2SDouglas Gregor /// (or the end of the file). 250718292f2SDouglas Gregor void skipUntil(MMToken::TokenKind K); 251718292f2SDouglas Gregor 252718292f2SDouglas Gregor void parseModuleDecl(); 253718292f2SDouglas Gregor void parseUmbrellaDecl(); 254718292f2SDouglas Gregor void parseHeaderDecl(); 255718292f2SDouglas Gregor 256718292f2SDouglas Gregor public: 257718292f2SDouglas Gregor typedef ModuleMap::Module Module; 258718292f2SDouglas Gregor 259718292f2SDouglas Gregor explicit ModuleMapParser(Lexer &L, SourceManager &SourceMgr, 260718292f2SDouglas Gregor DiagnosticsEngine &Diags, 2615257fc63SDouglas Gregor ModuleMap &Map, 2625257fc63SDouglas Gregor const DirectoryEntry *Directory) 2635257fc63SDouglas Gregor : L(L), SourceMgr(SourceMgr), Diags(Diags), Map(Map), 2645257fc63SDouglas Gregor Directory(Directory), HadError(false), ActiveModule(0) 265718292f2SDouglas Gregor { 266718292f2SDouglas Gregor TargetOptions TargetOpts; 267718292f2SDouglas Gregor TargetOpts.Triple = llvm::sys::getDefaultTargetTriple(); 268718292f2SDouglas Gregor Target = TargetInfo::CreateTargetInfo(Diags, TargetOpts); 269718292f2SDouglas Gregor 270718292f2SDouglas Gregor Tok.clear(); 271718292f2SDouglas Gregor consumeToken(); 272718292f2SDouglas Gregor } 273718292f2SDouglas Gregor 274718292f2SDouglas Gregor bool parseModuleMapFile(); 275718292f2SDouglas Gregor }; 276718292f2SDouglas Gregor } 277718292f2SDouglas Gregor 278718292f2SDouglas Gregor SourceLocation ModuleMapParser::consumeToken() { 279718292f2SDouglas Gregor retry: 280718292f2SDouglas Gregor SourceLocation Result = Tok.getLocation(); 281718292f2SDouglas Gregor Tok.clear(); 282718292f2SDouglas Gregor 283718292f2SDouglas Gregor Token LToken; 284718292f2SDouglas Gregor L.LexFromRawLexer(LToken); 285718292f2SDouglas Gregor Tok.Location = LToken.getLocation().getRawEncoding(); 286718292f2SDouglas Gregor switch (LToken.getKind()) { 287718292f2SDouglas Gregor case tok::raw_identifier: 288718292f2SDouglas Gregor Tok.StringData = LToken.getRawIdentifierData(); 289718292f2SDouglas Gregor Tok.StringLength = LToken.getLength(); 290718292f2SDouglas Gregor Tok.Kind = llvm::StringSwitch<MMToken::TokenKind>(Tok.getString()) 291718292f2SDouglas Gregor .Case("header", MMToken::HeaderKeyword) 292718292f2SDouglas Gregor .Case("explicit", MMToken::ExplicitKeyword) 293718292f2SDouglas Gregor .Case("module", MMToken::ModuleKeyword) 294718292f2SDouglas Gregor .Case("umbrella", MMToken::UmbrellaKeyword) 295718292f2SDouglas Gregor .Default(MMToken::Identifier); 296718292f2SDouglas Gregor break; 297718292f2SDouglas Gregor 298718292f2SDouglas Gregor case tok::eof: 299718292f2SDouglas Gregor Tok.Kind = MMToken::EndOfFile; 300718292f2SDouglas Gregor break; 301718292f2SDouglas Gregor 302718292f2SDouglas Gregor case tok::l_brace: 303718292f2SDouglas Gregor Tok.Kind = MMToken::LBrace; 304718292f2SDouglas Gregor break; 305718292f2SDouglas Gregor 306718292f2SDouglas Gregor case tok::r_brace: 307718292f2SDouglas Gregor Tok.Kind = MMToken::RBrace; 308718292f2SDouglas Gregor break; 309718292f2SDouglas Gregor 310718292f2SDouglas Gregor case tok::string_literal: { 311718292f2SDouglas Gregor // Parse the string literal. 312718292f2SDouglas Gregor LangOptions LangOpts; 313718292f2SDouglas Gregor StringLiteralParser StringLiteral(<oken, 1, SourceMgr, LangOpts, *Target); 314718292f2SDouglas Gregor if (StringLiteral.hadError) 315718292f2SDouglas Gregor goto retry; 316718292f2SDouglas Gregor 317718292f2SDouglas Gregor // Copy the string literal into our string data allocator. 318718292f2SDouglas Gregor unsigned Length = StringLiteral.GetStringLength(); 319718292f2SDouglas Gregor char *Saved = StringData.Allocate<char>(Length + 1); 320718292f2SDouglas Gregor memcpy(Saved, StringLiteral.GetString().data(), Length); 321718292f2SDouglas Gregor Saved[Length] = 0; 322718292f2SDouglas Gregor 323718292f2SDouglas Gregor // Form the token. 324718292f2SDouglas Gregor Tok.Kind = MMToken::StringLiteral; 325718292f2SDouglas Gregor Tok.StringData = Saved; 326718292f2SDouglas Gregor Tok.StringLength = Length; 327718292f2SDouglas Gregor break; 328718292f2SDouglas Gregor } 329718292f2SDouglas Gregor 330718292f2SDouglas Gregor case tok::comment: 331718292f2SDouglas Gregor goto retry; 332718292f2SDouglas Gregor 333718292f2SDouglas Gregor default: 334718292f2SDouglas Gregor Diags.Report(LToken.getLocation(), diag::err_mmap_unknown_token); 335718292f2SDouglas Gregor HadError = true; 336718292f2SDouglas Gregor goto retry; 337718292f2SDouglas Gregor } 338718292f2SDouglas Gregor 339718292f2SDouglas Gregor return Result; 340718292f2SDouglas Gregor } 341718292f2SDouglas Gregor 342718292f2SDouglas Gregor void ModuleMapParser::skipUntil(MMToken::TokenKind K) { 343718292f2SDouglas Gregor unsigned braceDepth = 0; 344718292f2SDouglas Gregor do { 345718292f2SDouglas Gregor switch (Tok.Kind) { 346718292f2SDouglas Gregor case MMToken::EndOfFile: 347718292f2SDouglas Gregor return; 348718292f2SDouglas Gregor 349718292f2SDouglas Gregor case MMToken::LBrace: 350718292f2SDouglas Gregor if (Tok.is(K) && braceDepth == 0) 351718292f2SDouglas Gregor return; 352718292f2SDouglas Gregor 353718292f2SDouglas Gregor ++braceDepth; 354718292f2SDouglas Gregor break; 355718292f2SDouglas Gregor 356718292f2SDouglas Gregor case MMToken::RBrace: 357718292f2SDouglas Gregor if (braceDepth > 0) 358718292f2SDouglas Gregor --braceDepth; 359718292f2SDouglas Gregor else if (Tok.is(K)) 360718292f2SDouglas Gregor return; 361718292f2SDouglas Gregor break; 362718292f2SDouglas Gregor 363718292f2SDouglas Gregor default: 364718292f2SDouglas Gregor if (braceDepth == 0 && Tok.is(K)) 365718292f2SDouglas Gregor return; 366718292f2SDouglas Gregor break; 367718292f2SDouglas Gregor } 368718292f2SDouglas Gregor 369718292f2SDouglas Gregor consumeToken(); 370718292f2SDouglas Gregor } while (true); 371718292f2SDouglas Gregor } 372718292f2SDouglas Gregor 373718292f2SDouglas Gregor /// \brief Parse a module declaration. 374718292f2SDouglas Gregor /// 375718292f2SDouglas Gregor /// module-declaration: 376718292f2SDouglas Gregor /// 'module' identifier { module-member* } 377718292f2SDouglas Gregor /// 378718292f2SDouglas Gregor /// module-member: 379718292f2SDouglas Gregor /// umbrella-declaration 380718292f2SDouglas Gregor /// header-declaration 381718292f2SDouglas Gregor /// 'explicit'[opt] module-declaration 382718292f2SDouglas Gregor void ModuleMapParser::parseModuleDecl() { 383718292f2SDouglas Gregor assert(Tok.is(MMToken::ExplicitKeyword) || Tok.is(MMToken::ModuleKeyword)); 384718292f2SDouglas Gregor 385718292f2SDouglas Gregor // Parse 'explicit' keyword, if present. 386718292f2SDouglas Gregor bool Explicit = false; 387718292f2SDouglas Gregor if (Tok.is(MMToken::ExplicitKeyword)) { 388718292f2SDouglas Gregor consumeToken(); 389718292f2SDouglas Gregor Explicit = true; 390718292f2SDouglas Gregor } 391718292f2SDouglas Gregor 392718292f2SDouglas Gregor // Parse 'module' keyword. 393718292f2SDouglas Gregor if (!Tok.is(MMToken::ModuleKeyword)) { 394718292f2SDouglas Gregor Diags.Report(Tok.getLocation(), 395718292f2SDouglas Gregor diag::err_mmap_expected_module_after_explicit); 396718292f2SDouglas Gregor consumeToken(); 397718292f2SDouglas Gregor HadError = true; 398718292f2SDouglas Gregor return; 399718292f2SDouglas Gregor } 400718292f2SDouglas Gregor consumeToken(); // 'module' keyword 401718292f2SDouglas Gregor 402718292f2SDouglas Gregor // Parse the module name. 403718292f2SDouglas Gregor if (!Tok.is(MMToken::Identifier)) { 404718292f2SDouglas Gregor Diags.Report(Tok.getLocation(), diag::err_mmap_expected_module_name); 405718292f2SDouglas Gregor HadError = true; 406718292f2SDouglas Gregor return; 407718292f2SDouglas Gregor } 408718292f2SDouglas Gregor StringRef ModuleName = Tok.getString(); 409718292f2SDouglas Gregor SourceLocation ModuleNameLoc = consumeToken(); 410718292f2SDouglas Gregor 411718292f2SDouglas Gregor // Parse the opening brace. 412718292f2SDouglas Gregor if (!Tok.is(MMToken::LBrace)) { 413718292f2SDouglas Gregor Diags.Report(Tok.getLocation(), diag::err_mmap_expected_lbrace) 414718292f2SDouglas Gregor << ModuleName; 415718292f2SDouglas Gregor HadError = true; 416718292f2SDouglas Gregor return; 417718292f2SDouglas Gregor } 418718292f2SDouglas Gregor SourceLocation LBraceLoc = consumeToken(); 419718292f2SDouglas Gregor 420718292f2SDouglas Gregor // Determine whether this (sub)module has already been defined. 421718292f2SDouglas Gregor llvm::StringMap<Module *> &ModuleSpace 422718292f2SDouglas Gregor = ActiveModule? ActiveModule->SubModules : Map.Modules; 423718292f2SDouglas Gregor llvm::StringMap<Module *>::iterator ExistingModule 424718292f2SDouglas Gregor = ModuleSpace.find(ModuleName); 425718292f2SDouglas Gregor if (ExistingModule != ModuleSpace.end()) { 426718292f2SDouglas Gregor Diags.Report(ModuleNameLoc, diag::err_mmap_module_redefinition) 427718292f2SDouglas Gregor << ModuleName; 428718292f2SDouglas Gregor Diags.Report(ExistingModule->getValue()->DefinitionLoc, 429718292f2SDouglas Gregor diag::note_mmap_prev_definition); 430718292f2SDouglas Gregor 431718292f2SDouglas Gregor // Skip the module definition. 432718292f2SDouglas Gregor skipUntil(MMToken::RBrace); 433718292f2SDouglas Gregor if (Tok.is(MMToken::RBrace)) 434718292f2SDouglas Gregor consumeToken(); 435718292f2SDouglas Gregor 436718292f2SDouglas Gregor HadError = true; 437718292f2SDouglas Gregor return; 438718292f2SDouglas Gregor } 439718292f2SDouglas Gregor 440718292f2SDouglas Gregor // Start defining this module. 441718292f2SDouglas Gregor ActiveModule = new Module(ModuleName, ModuleNameLoc, ActiveModule, Explicit); 442718292f2SDouglas Gregor ModuleSpace[ModuleName] = ActiveModule; 443718292f2SDouglas Gregor 444718292f2SDouglas Gregor bool Done = false; 445718292f2SDouglas Gregor do { 446718292f2SDouglas Gregor switch (Tok.Kind) { 447718292f2SDouglas Gregor case MMToken::EndOfFile: 448718292f2SDouglas Gregor case MMToken::RBrace: 449718292f2SDouglas Gregor Done = true; 450718292f2SDouglas Gregor break; 451718292f2SDouglas Gregor 452718292f2SDouglas Gregor case MMToken::ExplicitKeyword: 453718292f2SDouglas Gregor case MMToken::ModuleKeyword: 454718292f2SDouglas Gregor parseModuleDecl(); 455718292f2SDouglas Gregor break; 456718292f2SDouglas Gregor 457718292f2SDouglas Gregor case MMToken::HeaderKeyword: 458718292f2SDouglas Gregor parseHeaderDecl(); 459718292f2SDouglas Gregor break; 460718292f2SDouglas Gregor 461718292f2SDouglas Gregor case MMToken::UmbrellaKeyword: 462718292f2SDouglas Gregor parseUmbrellaDecl(); 463718292f2SDouglas Gregor break; 464718292f2SDouglas Gregor 465718292f2SDouglas Gregor default: 466718292f2SDouglas Gregor Diags.Report(Tok.getLocation(), diag::err_mmap_expected_member); 467718292f2SDouglas Gregor consumeToken(); 468718292f2SDouglas Gregor break; 469718292f2SDouglas Gregor } 470718292f2SDouglas Gregor } while (!Done); 471718292f2SDouglas Gregor 472718292f2SDouglas Gregor if (Tok.is(MMToken::RBrace)) 473718292f2SDouglas Gregor consumeToken(); 474718292f2SDouglas Gregor else { 475718292f2SDouglas Gregor Diags.Report(Tok.getLocation(), diag::err_mmap_expected_rbrace); 476718292f2SDouglas Gregor Diags.Report(LBraceLoc, diag::note_mmap_lbrace_match); 477718292f2SDouglas Gregor HadError = true; 478718292f2SDouglas Gregor } 479718292f2SDouglas Gregor 480718292f2SDouglas Gregor // We're done parsing this module. Pop back to our parent scope. 481718292f2SDouglas Gregor ActiveModule = ActiveModule->Parent; 482718292f2SDouglas Gregor } 483718292f2SDouglas Gregor 484718292f2SDouglas Gregor /// \brief Parse an umbrella header declaration. 485718292f2SDouglas Gregor /// 486718292f2SDouglas Gregor /// umbrella-declaration: 487718292f2SDouglas Gregor /// 'umbrella' string-literal 488718292f2SDouglas Gregor void ModuleMapParser::parseUmbrellaDecl() { 489718292f2SDouglas Gregor assert(Tok.is(MMToken::UmbrellaKeyword)); 490718292f2SDouglas Gregor SourceLocation UmbrellaLoc = consumeToken(); 491718292f2SDouglas Gregor 492718292f2SDouglas Gregor // Parse the header name. 493718292f2SDouglas Gregor if (!Tok.is(MMToken::StringLiteral)) { 494718292f2SDouglas Gregor Diags.Report(Tok.getLocation(), diag::err_mmap_expected_header) 495718292f2SDouglas Gregor << "umbrella"; 496718292f2SDouglas Gregor HadError = true; 497718292f2SDouglas Gregor return; 498718292f2SDouglas Gregor } 499718292f2SDouglas Gregor StringRef FileName = Tok.getString(); 500718292f2SDouglas Gregor SourceLocation FileNameLoc = consumeToken(); 501718292f2SDouglas Gregor 5025257fc63SDouglas Gregor // Check whether we already have an umbrella header. 5035257fc63SDouglas Gregor if (ActiveModule->UmbrellaHeader) { 5045257fc63SDouglas Gregor Diags.Report(FileNameLoc, diag::err_mmap_umbrella_header_conflict) 5055257fc63SDouglas Gregor << ActiveModule->getFullModuleName() 5065257fc63SDouglas Gregor << ActiveModule->UmbrellaHeader->getName(); 5075257fc63SDouglas Gregor HadError = true; 5085257fc63SDouglas Gregor return; 5095257fc63SDouglas Gregor } 5105257fc63SDouglas Gregor 5115257fc63SDouglas Gregor // Only top-level modules can have umbrella headers. 5125257fc63SDouglas Gregor if (ActiveModule->Parent) { 5135257fc63SDouglas Gregor Diags.Report(UmbrellaLoc, diag::err_mmap_umbrella_header_submodule) 5145257fc63SDouglas Gregor << ActiveModule->getFullModuleName(); 5155257fc63SDouglas Gregor HadError = true; 5165257fc63SDouglas Gregor return; 5175257fc63SDouglas Gregor } 5185257fc63SDouglas Gregor 5195257fc63SDouglas Gregor // Look for this file. 5205257fc63SDouglas Gregor llvm::SmallString<128> PathName; 5215257fc63SDouglas Gregor PathName += Directory->getName(); 5225257fc63SDouglas Gregor llvm::sys::path::append(PathName, FileName); 5235257fc63SDouglas Gregor 5245257fc63SDouglas Gregor // FIXME: We shouldn't be eagerly stat'ing every file named in a module map. 5255257fc63SDouglas Gregor // Come up with a lazy way to do this. 5265257fc63SDouglas Gregor if (const FileEntry *File = SourceMgr.getFileManager().getFile(PathName)) { 5275257fc63SDouglas Gregor if (const Module *OwningModule = Map.Headers[File]) { 5285257fc63SDouglas Gregor Diags.Report(FileNameLoc, diag::err_mmap_header_conflict) 5295257fc63SDouglas Gregor << FileName << OwningModule->getFullModuleName(); 5305257fc63SDouglas Gregor HadError = true; 531*b65dbfffSDouglas Gregor } else if ((OwningModule = Map.UmbrellaDirs[Directory])) { 532*b65dbfffSDouglas Gregor Diags.Report(UmbrellaLoc, diag::err_mmap_umbrella_clash) 533*b65dbfffSDouglas Gregor << OwningModule->getFullModuleName(); 534*b65dbfffSDouglas Gregor HadError = true; 5355257fc63SDouglas Gregor } else { 5365257fc63SDouglas Gregor // Record this umbrella header. 5375257fc63SDouglas Gregor ActiveModule->UmbrellaHeader = File; 5385257fc63SDouglas Gregor Map.Headers[File] = ActiveModule; 539*b65dbfffSDouglas Gregor Map.UmbrellaDirs[Directory] = ActiveModule; 5405257fc63SDouglas Gregor } 5415257fc63SDouglas Gregor } else { 5425257fc63SDouglas Gregor Diags.Report(FileNameLoc, diag::err_mmap_header_not_found) 5435257fc63SDouglas Gregor << true << FileName; 5445257fc63SDouglas Gregor HadError = true; 5455257fc63SDouglas Gregor } 546718292f2SDouglas Gregor } 547718292f2SDouglas Gregor 548718292f2SDouglas Gregor /// \brief Parse a header declaration. 549718292f2SDouglas Gregor /// 550718292f2SDouglas Gregor /// header-declaration: 551718292f2SDouglas Gregor /// 'header' string-literal 552718292f2SDouglas Gregor void ModuleMapParser::parseHeaderDecl() { 553718292f2SDouglas Gregor assert(Tok.is(MMToken::HeaderKeyword)); 5541871ed3dSBenjamin Kramer consumeToken(); 555718292f2SDouglas Gregor 556718292f2SDouglas Gregor // Parse the header name. 557718292f2SDouglas Gregor if (!Tok.is(MMToken::StringLiteral)) { 558718292f2SDouglas Gregor Diags.Report(Tok.getLocation(), diag::err_mmap_expected_header) 559718292f2SDouglas Gregor << "header"; 560718292f2SDouglas Gregor HadError = true; 561718292f2SDouglas Gregor return; 562718292f2SDouglas Gregor } 563718292f2SDouglas Gregor StringRef FileName = Tok.getString(); 564718292f2SDouglas Gregor SourceLocation FileNameLoc = consumeToken(); 565718292f2SDouglas Gregor 5665257fc63SDouglas Gregor // Look for this file. 5675257fc63SDouglas Gregor llvm::SmallString<128> PathName; 5685257fc63SDouglas Gregor PathName += Directory->getName(); 5695257fc63SDouglas Gregor llvm::sys::path::append(PathName, FileName); 5705257fc63SDouglas Gregor 5715257fc63SDouglas Gregor // FIXME: We shouldn't be eagerly stat'ing every file named in a module map. 5725257fc63SDouglas Gregor // Come up with a lazy way to do this. 5735257fc63SDouglas Gregor if (const FileEntry *File = SourceMgr.getFileManager().getFile(PathName)) { 5745257fc63SDouglas Gregor if (const Module *OwningModule = Map.Headers[File]) { 5755257fc63SDouglas Gregor Diags.Report(FileNameLoc, diag::err_mmap_header_conflict) 5765257fc63SDouglas Gregor << FileName << OwningModule->getFullModuleName(); 5775257fc63SDouglas Gregor HadError = true; 5785257fc63SDouglas Gregor } else { 5795257fc63SDouglas Gregor // Record this file. 5805257fc63SDouglas Gregor ActiveModule->Headers.push_back(File); 5815257fc63SDouglas Gregor Map.Headers[File] = ActiveModule; 5825257fc63SDouglas Gregor } 5835257fc63SDouglas Gregor } else { 5845257fc63SDouglas Gregor Diags.Report(FileNameLoc, diag::err_mmap_header_not_found) 5855257fc63SDouglas Gregor << false << FileName; 5865257fc63SDouglas Gregor HadError = true; 5875257fc63SDouglas Gregor } 588718292f2SDouglas Gregor } 589718292f2SDouglas Gregor 590718292f2SDouglas Gregor /// \brief Parse a module map file. 591718292f2SDouglas Gregor /// 592718292f2SDouglas Gregor /// module-map-file: 593718292f2SDouglas Gregor /// module-declaration* 594718292f2SDouglas Gregor bool ModuleMapParser::parseModuleMapFile() { 595718292f2SDouglas Gregor do { 596718292f2SDouglas Gregor switch (Tok.Kind) { 597718292f2SDouglas Gregor case MMToken::EndOfFile: 598718292f2SDouglas Gregor return HadError; 599718292f2SDouglas Gregor 600718292f2SDouglas Gregor case MMToken::ModuleKeyword: 601718292f2SDouglas Gregor parseModuleDecl(); 602718292f2SDouglas Gregor break; 603718292f2SDouglas Gregor 604718292f2SDouglas Gregor case MMToken::ExplicitKeyword: 605718292f2SDouglas Gregor case MMToken::HeaderKeyword: 606718292f2SDouglas Gregor case MMToken::Identifier: 607718292f2SDouglas Gregor case MMToken::LBrace: 608718292f2SDouglas Gregor case MMToken::RBrace: 609718292f2SDouglas Gregor case MMToken::StringLiteral: 610718292f2SDouglas Gregor case MMToken::UmbrellaKeyword: 611718292f2SDouglas Gregor Diags.Report(Tok.getLocation(), diag::err_mmap_expected_module); 612718292f2SDouglas Gregor HadError = true; 613718292f2SDouglas Gregor consumeToken(); 614718292f2SDouglas Gregor break; 615718292f2SDouglas Gregor } 616718292f2SDouglas Gregor } while (true); 617718292f2SDouglas Gregor 618718292f2SDouglas Gregor return HadError; 619718292f2SDouglas Gregor } 620718292f2SDouglas Gregor 621718292f2SDouglas Gregor bool ModuleMap::parseModuleMapFile(const FileEntry *File) { 622718292f2SDouglas Gregor FileID ID = SourceMgr->createFileID(File, SourceLocation(), SrcMgr::C_User); 623718292f2SDouglas Gregor const llvm::MemoryBuffer *Buffer = SourceMgr->getBuffer(ID); 624718292f2SDouglas Gregor if (!Buffer) 625718292f2SDouglas Gregor return true; 626718292f2SDouglas Gregor 627718292f2SDouglas Gregor // Parse this module map file. 628718292f2SDouglas Gregor Lexer L(ID, SourceMgr->getBuffer(ID), *SourceMgr, LangOpts); 629718292f2SDouglas Gregor Diags->getClient()->BeginSourceFile(LangOpts); 6305257fc63SDouglas Gregor ModuleMapParser Parser(L, *SourceMgr, *Diags, *this, File->getDir()); 631718292f2SDouglas Gregor bool Result = Parser.parseModuleMapFile(); 632718292f2SDouglas Gregor Diags->getClient()->EndSourceFile(); 633718292f2SDouglas Gregor 634718292f2SDouglas Gregor return Result; 635718292f2SDouglas Gregor } 636