forked from OSchip/llvm-project
1013 lines
30 KiB
C++
1013 lines
30 KiB
C++
//===--- 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<DiagnosticIDs> DiagIDs(new DiagnosticIDs);
|
|
Diags = llvm::IntrusiveRefCntPtr<DiagnosticsEngine>(
|
|
new DiagnosticsEngine(DiagIDs));
|
|
Diags->setClient(DC.clone(*Diags), /*ShouldOwnClient=*/true);
|
|
SourceMgr = new SourceManager(*Diags, FileMgr);
|
|
}
|
|
|
|
ModuleMap::~ModuleMap() {
|
|
for (llvm::StringMap<Module *>::iterator I = Modules.begin(),
|
|
IEnd = Modules.end();
|
|
I != IEnd; ++I) {
|
|
delete I->getValue();
|
|
}
|
|
|
|
delete SourceMgr;
|
|
}
|
|
|
|
Module *ModuleMap::findModuleForHeader(const FileEntry *File) {
|
|
llvm::DenseMap<const FileEntry *, Module *>::iterator Known
|
|
= Headers.find(File);
|
|
if (Known != Headers.end())
|
|
return Known->second;
|
|
|
|
const DirectoryEntry *Dir = File->getDir();
|
|
llvm::SmallVector<const DirectoryEntry *, 2> SkippedDirs;
|
|
StringRef DirName = Dir->getName();
|
|
|
|
// Keep walking up the directory hierarchy, looking for a directory with
|
|
// an umbrella header.
|
|
do {
|
|
llvm::DenseMap<const DirectoryEntry *, Module *>::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<Module *>::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<Module *>::iterator Sub = Context->SubModules.find(Name);
|
|
if (Sub != Context->SubModules.end())
|
|
return Sub->getValue();
|
|
|
|
return 0;
|
|
}
|
|
|
|
std::pair<Module *, bool>
|
|
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<Module *>::iterator M = Modules.begin(),
|
|
MEnd = Modules.end();
|
|
M != MEnd; ++M)
|
|
M->getValue()->print(llvm::errs(), 2);
|
|
|
|
llvm::errs() << "Headers:";
|
|
for (llvm::DenseMap<const FileEntry *, Module *>::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<MMToken::TokenKind>(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<char>(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<Module *> &ModuleSpace
|
|
= ActiveModule? ActiveModule->SubModules : Map.Modules;
|
|
llvm::StringMap<Module *>::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;
|
|
}
|