#include "Compiler.h"
#include <cassert>
#include <cstdlib>
#include <string>
#include <vector>
#include "clang/AST/ASTConsumer.h"
#include "clang/AST/ASTContext.h"
#include "clang/Basic/DiagnosticIDs.h"
#include "clang/Basic/FileManager.h"
#include "clang/Basic/FileSystemOptions.h"
#include "clang/Basic/LangOptions.h"
#include "clang/Basic/SourceManager.h"
#include "clang/Basic/TargetInfo.h"
#include "clang/Basic/TargetOptions.h"
#include "clang/Frontend/CodeGenOptions.h"
#include "clang/Frontend/DiagnosticOptions.h"
#include "clang/Frontend/DependencyOutputOptions.h"
#include "clang/Frontend/FrontendDiagnostic.h"
#include "clang/Frontend/TextDiagnosticPrinter.h"
#include "clang/Frontend/Utils.h"
#include "clang/Lex/Preprocessor.h"
#include "clang/Lex/HeaderSearch.h"
#include "clang/Parse/ParseAST.h"
#include "llvm/LLVMContext.h"
#include "llvm/ADT/IntrusiveRefCntPtr.h"
#include "llvm/Bitcode/ReaderWriter.h"
#include "llvm/Support/raw_ostream.h"
#include "llvm/Support/MemoryBuffer.h"
#include "llvm/Support/ErrorHandling.h"
#include "llvm/Support/ManagedStatic.h"
#include "llvm/Support/ToolOutputFile.h"
#include "llvm/Support/Path.h"
#include "llvm/Target/TargetSelect.h"
#include "Backend.h"
namespace ndkpc {
static inline llvm::tool_output_file *openOutputFile(const char *OutputFile,
unsigned Flags,
std::string* Error,
clang::Diagnostic* Diag) {
assert((OutputFile != NULL) && (Error != NULL) && (Diag != NULL) &&
"Invalid parameter!");
llvm::tool_output_file *F =
new llvm::tool_output_file(OutputFile, *Error, Flags);
if (F != NULL)
return F;
// Report error here.
Diag->Report(clang::diag::err_fe_error_opening) << OutputFile << *Error;
return NULL;
}
void Compiler::LLVMErrorHandler(void *UserData, const std::string &Message) {
clang::Diagnostic* Diags = static_cast<clang::Diagnostic*>(UserData);
Diags->Report(clang::diag::err_fe_error_backend) << Message;
exit(1);
}
void Compiler::createDiagnostic() {
mpDiagClient = new clang::TextDiagnosticPrinter(llvm::errs(),
clang::DiagnosticOptions());
mDiagIDs = new clang::DiagnosticIDs();
mDiagnostics = new clang::Diagnostic(mDiagIDs, mpDiagClient, true);
initDiagnostic();
return;
}
void Compiler::createTarget(const std::string &Triple, const std::string &CPU,
const std::vector<std::string> &Features) {
if (!Triple.empty())
mTargetOpts.Triple = Triple;
else
mTargetOpts.Triple = llvm::Triple::normalize(DEFAULT_TARGET_TRIPLE_STRING);
if (!CPU.empty())
mTargetOpts.CPU = CPU;
if (!Features.empty())
mTargetOpts.Features = Features;
mTarget.reset(clang::TargetInfo::CreateTargetInfo(*mDiagnostics,
mTargetOpts));
return;
}
void Compiler::createFileManager() {
mFileSysOpt.reset(new clang::FileSystemOptions());
mFileMgr.reset(new clang::FileManager(*mFileSysOpt));
}
void Compiler::createSourceManager() {
mSourceMgr.reset(new clang::SourceManager(*mDiagnostics, *mFileMgr));
return;
}
void Compiler::createPreprocessor() {
clang::HeaderSearch *HS = new clang::HeaderSearch(*mFileMgr);
mPP.reset(new clang::Preprocessor(*mDiagnostics,
mLangOpts,
*mTarget,
*mSourceMgr,
*HS,
NULL, /* IILookup */
/* OwnsHeaderSearch = */true));
std::vector<clang::DirectoryLookup> SearchList;
for (unsigned i = 0, e = mIncludePaths.size(); i != e; i++) {
if (const clang::DirectoryEntry *DE =
mFileMgr->getDirectory(mIncludePaths[i])) {
SearchList.push_back(clang::DirectoryLookup(DE,
clang::SrcMgr::C_System,
false, /* isUser */
false /* isFramework */));
}
}
HS->SetSearchPaths(SearchList, 0/* angledDirIdx FIXME CHECK */, 0/* systemDirIdx */, false/* noCurDirSearch */);
initPreprocessor();
return;
}
void Compiler::createASTContext() {
mASTContext.reset(new clang::ASTContext(mLangOpts,
*mSourceMgr,
*mTarget,
mPP->getIdentifierTable(),
mPP->getSelectorTable(),
mPP->getBuiltinInfo(),
/* size_reserve = */0));
initASTContext();
return;
}
clang::ASTConsumer
*Compiler::createBackend(const clang::CodeGenOptions& CodeGenOpts,
llvm::raw_ostream *OS,
OutputType OT) {
return new Backend(CodeGenOpts,
mTargetOpts,
mDiagnostics.getPtr(),
OS,
OT);
}
Compiler::Compiler() : mInitialized(false), mpDiagClient(NULL), mOT(OT_Default) {
}
void Compiler::injectPreDefined() {
typedef std::map<std::string, std::string> SymbolMapTy;
for (SymbolMapTy::iterator
it = mPreDefinedSymbolMap.begin(), et = mPreDefinedSymbolMap.end();
it != et; ++it) {
std::string Str = "#define "+it->first+" "+it->second+"\n";
mPP->setPredefines(Str);
}
}
void Compiler::init(const std::string &Triple, const std::string &CPU,
const std::vector<std::string> &Features, bool isCXX) {
mLangOpts.RTTI = 0; // Turn off the RTTI information support
mLangOpts.NeXTRuntime = 0; // Turn off the NeXT runtime uses
mLangOpts.C99 = 1;
if (isCXX) {
mLangOpts.CPlusPlus = 1;
}
mCodeGenOpts.OptimizationLevel = 3; /* -O3 */
createDiagnostic();
llvm::install_fatal_error_handler(LLVMErrorHandler, mDiagnostics.getPtr());
createTarget(Triple, CPU, Features);
createFileManager();
createSourceManager();
mInitialized = true;
return;
}
bool Compiler::setInputSource(llvm::StringRef InputFile,
const char *Text,
size_t TextLength) {
mInputFileName = InputFile.str();
// Reset the ID tables if we are reusing the SourceManager
mSourceMgr->clearIDTables();
// Load the source
llvm::MemoryBuffer *SB =
llvm::MemoryBuffer::getMemBuffer(Text, Text + TextLength);
mSourceMgr->createMainFileIDForMemBuffer(SB);
if (mSourceMgr->getMainFileID().isInvalid()) {
mDiagnostics->Report(clang::diag::err_fe_error_reading) << InputFile;
return false;
}
return true;
}
bool Compiler::setInputSource(llvm::StringRef InputFile) {
mInputFileName = InputFile.str();
mSourceMgr->clearIDTables();
const clang::FileEntry *File = mFileMgr->getFile(InputFile);
if (File)
mSourceMgr->createMainFileID(File);
if (mSourceMgr->getMainFileID().isInvalid()) {
mDiagnostics->Report(clang::diag::err_fe_error_reading) << InputFile;
return false;
}
return true;
}
bool Compiler::setOutput(const char *OutputFile) {
llvm::sys::Path OutputFilePath(OutputFile);
std::string Error;
llvm::tool_output_file *OS = NULL;
switch (mOT) {
case OT_Dependency:
case OT_Assembly:
case OT_LLVMAssembly: {
OS = openOutputFile(OutputFile, 0, &Error, mDiagnostics.getPtr());
break;
}
case OT_Nothing: {
break;
}
case OT_Object:
case OT_Bitcode: {
OS = openOutputFile(OutputFile,
llvm::raw_fd_ostream::F_Binary,
&Error,
mDiagnostics.getPtr());
break;
}
default: {
llvm_unreachable("Unknown compiler output type");
}
}
if (!Error.empty())
return false;
mOS.reset(OS);
mOutputFileName = OutputFile;
return true;
}
int Compiler::compile() {
if (mDiagnostics->hasErrorOccurred())
return 1;
if (mOS.get() == NULL)
return 1;
// Here is per-compilation needed initialization
createPreprocessor();
createASTContext();
mBackend.reset(createBackend(mCodeGenOpts, &mOS->os(), mOT));
// Inform the diagnostic client we are processing a source file
mpDiagClient->BeginSourceFile(mLangOpts, mPP.get());
if (mLangOpts.CPlusPlus == 1) {
mPP->setPredefines("#define __cplusplus\n");
}
this->injectPreDefined();
// The core of the slang compiler
ParseAST(*mPP, mBackend.get(), *mASTContext);
// Inform the diagnostic client we are done with previous source file
mpDiagClient->EndSourceFile();
// Declare success if no error
if (!mDiagnostics->hasErrorOccurred())
mOS->keep();
// The compilation ended, clear
mBackend.reset();
mASTContext.reset();
mPP.reset();
mOS.reset();
return mDiagnostics->hasErrorOccurred() ? 1 : 0;
}
void Compiler::reset() {
mDiagnostics->Reset();
return;
}
Compiler::~Compiler() {
llvm::llvm_shutdown();
return;
}
}