2011-12-01 07:21:26 +08:00
|
|
|
//===--- Module.h - Describe a module ---------------------------*- 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 Module class, which describes a module in the source
|
|
|
|
// code.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#include "clang/Basic/Module.h"
|
|
|
|
#include "clang/Basic/FileManager.h"
|
|
|
|
#include "llvm/Support/raw_ostream.h"
|
|
|
|
using namespace clang;
|
|
|
|
|
|
|
|
Module::~Module() {
|
|
|
|
for (llvm::StringMap<Module *>::iterator I = SubModules.begin(),
|
|
|
|
IEnd = SubModules.end();
|
|
|
|
I != IEnd; ++I) {
|
|
|
|
delete I->getValue();
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2011-12-06 01:28:06 +08:00
|
|
|
bool Module::isSubModuleOf(Module *Other) const {
|
|
|
|
const Module *This = this;
|
|
|
|
do {
|
|
|
|
if (This == Other)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
This = This->Parent;
|
|
|
|
} while (This);
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2011-12-06 06:27:44 +08:00
|
|
|
const Module *Module::getTopLevelModule() const {
|
|
|
|
const Module *Result = this;
|
|
|
|
while (Result->Parent)
|
|
|
|
Result = Result->Parent;
|
|
|
|
|
|
|
|
return Result;
|
|
|
|
}
|
|
|
|
|
2011-12-01 07:21:26 +08:00
|
|
|
std::string Module::getFullModuleName() const {
|
|
|
|
llvm::SmallVector<StringRef, 2> Names;
|
|
|
|
|
|
|
|
// Build up the set of module names (from innermost to outermost).
|
|
|
|
for (const Module *M = this; M; M = M->Parent)
|
|
|
|
Names.push_back(M->Name);
|
|
|
|
|
|
|
|
std::string Result;
|
|
|
|
for (llvm::SmallVector<StringRef, 2>::reverse_iterator I = Names.rbegin(),
|
|
|
|
IEnd = Names.rend();
|
|
|
|
I != IEnd; ++I) {
|
|
|
|
if (!Result.empty())
|
|
|
|
Result += '.';
|
|
|
|
|
|
|
|
Result += *I;
|
|
|
|
}
|
|
|
|
|
|
|
|
return Result;
|
|
|
|
}
|
|
|
|
|
2011-12-09 01:39:04 +08:00
|
|
|
const DirectoryEntry *Module::getUmbrellaDir() const {
|
|
|
|
if (const FileEntry *Header = getUmbrellaHeader())
|
|
|
|
return Header->getDir();
|
|
|
|
|
|
|
|
return Umbrella.dyn_cast<const DirectoryEntry *>();
|
|
|
|
}
|
|
|
|
|
2011-12-03 02:58:38 +08:00
|
|
|
static void printModuleId(llvm::raw_ostream &OS, const ModuleId &Id) {
|
|
|
|
for (unsigned I = 0, N = Id.size(); I != N; ++I) {
|
|
|
|
if (I)
|
|
|
|
OS << ".";
|
|
|
|
OS << Id[I].first;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2011-12-01 07:21:26 +08:00
|
|
|
void Module::print(llvm::raw_ostream &OS, unsigned Indent) const {
|
|
|
|
OS.indent(Indent);
|
|
|
|
if (IsFramework)
|
|
|
|
OS << "framework ";
|
|
|
|
if (IsExplicit)
|
|
|
|
OS << "explicit ";
|
|
|
|
OS << "module " << Name << " {\n";
|
|
|
|
|
2011-12-09 01:39:04 +08:00
|
|
|
if (const FileEntry *UmbrellaHeader = getUmbrellaHeader()) {
|
2011-12-01 07:21:26 +08:00
|
|
|
OS.indent(Indent + 2);
|
2011-12-09 02:00:48 +08:00
|
|
|
OS << "umbrella header \"";
|
2011-12-01 07:21:26 +08:00
|
|
|
OS.write_escaped(UmbrellaHeader->getName());
|
|
|
|
OS << "\"\n";
|
2011-12-09 02:00:48 +08:00
|
|
|
} else if (const DirectoryEntry *UmbrellaDir = getUmbrellaDir()) {
|
|
|
|
OS.indent(Indent + 2);
|
|
|
|
OS << "umbrella \"";
|
|
|
|
OS.write_escaped(UmbrellaDir->getName());
|
|
|
|
OS << "\"\n";
|
2011-12-01 07:21:26 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
for (unsigned I = 0, N = Headers.size(); I != N; ++I) {
|
|
|
|
OS.indent(Indent + 2);
|
|
|
|
OS << "header \"";
|
|
|
|
OS.write_escaped(Headers[I]->getName());
|
|
|
|
OS << "\"\n";
|
|
|
|
}
|
|
|
|
|
|
|
|
for (llvm::StringMap<Module *>::const_iterator MI = SubModules.begin(),
|
2011-12-06 06:27:44 +08:00
|
|
|
MIEnd = SubModules.end();
|
2011-12-01 07:21:26 +08:00
|
|
|
MI != MIEnd; ++MI)
|
|
|
|
MI->getValue()->print(OS, Indent + 2);
|
|
|
|
|
2011-12-03 02:58:38 +08:00
|
|
|
for (unsigned I = 0, N = Exports.size(); I != N; ++I) {
|
|
|
|
OS.indent(Indent + 2);
|
2011-12-06 01:34:59 +08:00
|
|
|
OS << "export ";
|
|
|
|
if (Module *Restriction = Exports[I].getPointer()) {
|
|
|
|
OS << Restriction->getFullModuleName();
|
|
|
|
if (Exports[I].getInt())
|
|
|
|
OS << ".*";
|
|
|
|
} else {
|
|
|
|
OS << "*";
|
|
|
|
}
|
2011-12-03 02:58:38 +08:00
|
|
|
OS << "\n";
|
|
|
|
}
|
|
|
|
|
|
|
|
for (unsigned I = 0, N = UnresolvedExports.size(); I != N; ++I) {
|
|
|
|
OS.indent(Indent + 2);
|
|
|
|
OS << "export ";
|
|
|
|
printModuleId(OS, UnresolvedExports[I].Id);
|
2011-12-06 01:34:59 +08:00
|
|
|
if (UnresolvedExports[I].Wildcard) {
|
|
|
|
if (UnresolvedExports[I].Id.empty())
|
|
|
|
OS << "*";
|
|
|
|
else
|
|
|
|
OS << ".*";
|
|
|
|
}
|
2011-12-03 02:58:38 +08:00
|
|
|
OS << "\n";
|
|
|
|
}
|
|
|
|
|
2011-12-06 06:27:44 +08:00
|
|
|
if (InferSubmodules) {
|
|
|
|
OS.indent(Indent + 2);
|
|
|
|
if (InferExplicitSubmodules)
|
|
|
|
OS << "explicit ";
|
|
|
|
OS << "module * {\n";
|
|
|
|
if (InferExportWildcard) {
|
|
|
|
OS.indent(Indent + 4);
|
|
|
|
OS << "export *\n";
|
|
|
|
}
|
|
|
|
OS.indent(Indent + 2);
|
|
|
|
OS << "}\n";
|
|
|
|
}
|
|
|
|
|
2011-12-01 07:21:26 +08:00
|
|
|
OS.indent(Indent);
|
|
|
|
OS << "}\n";
|
|
|
|
}
|
|
|
|
|
|
|
|
void Module::dump() const {
|
|
|
|
print(llvm::errs());
|
|
|
|
}
|
|
|
|
|
|
|
|
|