2007-10-07 10:04:32 +04:00
|
|
|
//===--- ASTConsumers.cpp - ASTConsumer implementations -------------------===//
|
2007-07-11 21:01:13 +04:00
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
2007-10-07 10:04:32 +04:00
|
|
|
// This file was developed by Chris Lattner and is distributed under the
|
2007-07-11 21:01:13 +04:00
|
|
|
// University of Illinois Open Source License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
2007-10-07 10:04:32 +04:00
|
|
|
// AST Consumer Implementations.
|
2007-07-11 21:01:13 +04:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2007-10-07 10:04:32 +04:00
|
|
|
#include "ASTConsumers.h"
|
2007-07-11 21:01:13 +04:00
|
|
|
#include "clang/AST/AST.h"
|
2007-09-16 03:02:28 +04:00
|
|
|
#include "clang/AST/ASTConsumer.h"
|
2007-08-22 01:42:03 +04:00
|
|
|
#include "clang/AST/CFG.h"
|
2007-09-06 04:17:54 +04:00
|
|
|
#include "clang/Analysis/LiveVariables.h"
|
2007-09-07 03:00:42 +04:00
|
|
|
#include "clang/Analysis/LocalCheckers.h"
|
2007-08-09 02:51:59 +04:00
|
|
|
using namespace clang;
|
2007-07-11 21:01:13 +04:00
|
|
|
|
2007-08-09 02:51:59 +04:00
|
|
|
|
|
|
|
static void PrintFunctionDeclStart(FunctionDecl *FD) {
|
2007-07-11 21:01:13 +04:00
|
|
|
bool HasBody = FD->getBody();
|
|
|
|
|
2007-08-26 08:02:13 +04:00
|
|
|
fprintf(stderr, "\n");
|
|
|
|
|
|
|
|
switch (FD->getStorageClass()) {
|
|
|
|
default: assert(0 && "Unknown storage class");
|
|
|
|
case FunctionDecl::None: break;
|
|
|
|
case FunctionDecl::Extern: fprintf(stderr, "extern "); break;
|
|
|
|
case FunctionDecl::Static: fprintf(stderr, "static "); break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (FD->isInline())
|
|
|
|
fprintf(stderr, "inline ");
|
|
|
|
|
2007-07-11 21:01:13 +04:00
|
|
|
std::string Proto = FD->getName();
|
|
|
|
FunctionType *AFT = cast<FunctionType>(FD->getType());
|
|
|
|
|
|
|
|
if (FunctionTypeProto *FT = dyn_cast<FunctionTypeProto>(AFT)) {
|
|
|
|
Proto += "(";
|
|
|
|
for (unsigned i = 0, e = FD->getNumParams(); i != e; ++i) {
|
|
|
|
if (i) Proto += ", ";
|
|
|
|
std::string ParamStr;
|
|
|
|
if (HasBody) ParamStr = FD->getParamDecl(i)->getName();
|
|
|
|
|
|
|
|
FT->getArgType(i).getAsStringInternal(ParamStr);
|
|
|
|
Proto += ParamStr;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (FT->isVariadic()) {
|
|
|
|
if (FD->getNumParams()) Proto += ", ";
|
|
|
|
Proto += "...";
|
|
|
|
}
|
|
|
|
Proto += ")";
|
|
|
|
} else {
|
|
|
|
assert(isa<FunctionTypeNoProto>(AFT));
|
|
|
|
Proto += "()";
|
|
|
|
}
|
|
|
|
|
|
|
|
AFT->getResultType().getAsStringInternal(Proto);
|
2007-08-26 08:02:13 +04:00
|
|
|
fprintf(stderr, "%s", Proto.c_str());
|
2007-07-11 21:01:13 +04:00
|
|
|
|
2007-08-09 02:51:59 +04:00
|
|
|
if (!FD->getBody())
|
2007-07-11 21:01:13 +04:00
|
|
|
fprintf(stderr, ";\n");
|
2007-08-09 02:51:59 +04:00
|
|
|
// Doesn't print the body.
|
2007-07-11 21:01:13 +04:00
|
|
|
}
|
|
|
|
|
2007-08-09 02:51:59 +04:00
|
|
|
static void PrintTypeDefDecl(TypedefDecl *TD) {
|
2007-07-11 21:01:13 +04:00
|
|
|
std::string S = TD->getName();
|
|
|
|
TD->getUnderlyingType().getAsStringInternal(S);
|
|
|
|
fprintf(stderr, "typedef %s;\n", S.c_str());
|
|
|
|
}
|
|
|
|
|
2007-09-11 00:51:04 +04:00
|
|
|
static void PrintObjcInterfaceDecl(ObjcInterfaceDecl *OID) {
|
2007-10-09 03:06:41 +04:00
|
|
|
std::string I = OID->getName();
|
|
|
|
ObjcInterfaceDecl *SID = OID->getSuperClass();
|
|
|
|
if (SID) {
|
|
|
|
std::string S = SID->getName();
|
|
|
|
fprintf(stderr, "@interface %s : %s", I.c_str(), S.c_str());
|
|
|
|
}
|
|
|
|
else
|
|
|
|
fprintf(stderr, "@interface %s", I.c_str());
|
|
|
|
// Protocols?
|
|
|
|
int count = OID->getNumIntfRefProtocols();
|
|
|
|
if (count > 0) {
|
|
|
|
ObjcProtocolDecl **refProtocols = OID->getReferencedProtocols();
|
|
|
|
for (int i = 0; i < count; i++)
|
|
|
|
fprintf(stderr, "%c%s", (i == 0 ? '<' : ','),
|
|
|
|
refProtocols[i]->getName());
|
|
|
|
}
|
|
|
|
if (count > 0)
|
|
|
|
fprintf(stderr, ">;\n");
|
|
|
|
else
|
|
|
|
fprintf(stderr, ";\n");
|
2007-09-11 00:51:04 +04:00
|
|
|
// FIXME: implement the rest...
|
|
|
|
}
|
|
|
|
|
2007-10-08 22:53:38 +04:00
|
|
|
static void PrintObjcProtocolDecl(ObjcProtocolDecl *PID) {
|
|
|
|
std::string S = PID->getName();
|
|
|
|
fprintf(stderr, "@protocol %s;\n", S.c_str());
|
|
|
|
// FIXME: implement the rest...
|
|
|
|
}
|
|
|
|
|
|
|
|
static void PrintObjcCategoryImplDecl(ObjcCategoryImplDecl *PID) {
|
|
|
|
std::string S = PID->getName();
|
|
|
|
std::string I = PID->getClassInterface()->getName();
|
|
|
|
fprintf(stderr, "@implementation %s(%s);\n", I.c_str(), S.c_str());
|
|
|
|
// FIXME: implement the rest...
|
|
|
|
}
|
|
|
|
|
|
|
|
static void PrintObjcCategoryDecl(ObjcCategoryDecl *PID) {
|
|
|
|
std::string S = PID->getName();
|
|
|
|
std::string I = PID->getClassInterface()->getName();
|
|
|
|
fprintf(stderr, "@interface %s(%s);\n", I.c_str(), S.c_str());
|
|
|
|
// FIXME: implement the rest...
|
|
|
|
}
|
|
|
|
|
2007-10-12 03:42:27 +04:00
|
|
|
static void PrintObjcCompatibleAliasDecl(ObjcCompatibleAliasDecl *AID) {
|
|
|
|
std::string A = AID->getName();
|
|
|
|
std::string I = AID->getClassInterface()->getName();
|
|
|
|
fprintf(stderr, "@compatibility_alias %s %s;\n", A.c_str(), I.c_str());
|
|
|
|
}
|
|
|
|
|
2007-09-16 03:02:28 +04:00
|
|
|
namespace {
|
|
|
|
class ASTPrinter : public ASTConsumer {
|
|
|
|
virtual void HandleTopLevelDecl(Decl *D) {
|
|
|
|
if (FunctionDecl *FD = dyn_cast<FunctionDecl>(D)) {
|
|
|
|
PrintFunctionDeclStart(FD);
|
|
|
|
|
|
|
|
if (FD->getBody()) {
|
|
|
|
fprintf(stderr, " ");
|
|
|
|
FD->getBody()->dumpPretty();
|
|
|
|
fprintf(stderr, "\n");
|
|
|
|
}
|
|
|
|
} else if (TypedefDecl *TD = dyn_cast<TypedefDecl>(D)) {
|
|
|
|
PrintTypeDefDecl(TD);
|
2007-10-06 22:52:10 +04:00
|
|
|
} else if (ObjcInterfaceDecl *OID = dyn_cast<ObjcInterfaceDecl>(D)) {
|
|
|
|
PrintObjcInterfaceDecl(OID);
|
2007-10-08 22:53:38 +04:00
|
|
|
} else if (ObjcProtocolDecl *PID = dyn_cast<ObjcProtocolDecl>(D)) {
|
|
|
|
PrintObjcProtocolDecl(PID);
|
2007-10-06 22:52:10 +04:00
|
|
|
} else if (ObjcForwardProtocolDecl *OFPD =
|
|
|
|
dyn_cast<ObjcForwardProtocolDecl>(D)) {
|
|
|
|
fprintf(stderr, "@protocol ");
|
|
|
|
for (unsigned i = 0, e = OFPD->getNumForwardDecls(); i != e; ++i) {
|
|
|
|
const ObjcProtocolDecl *D = OFPD->getForwardProtocolDecl(i);
|
|
|
|
if (i) fprintf(stderr, ", ");
|
|
|
|
fprintf(stderr, "%s", D->getName());
|
|
|
|
}
|
2007-10-06 23:08:22 +04:00
|
|
|
fprintf(stderr, ";\n");
|
2007-10-06 22:52:10 +04:00
|
|
|
} else if (ObjcImplementationDecl *OID =
|
|
|
|
dyn_cast<ObjcImplementationDecl>(D)) {
|
|
|
|
fprintf(stderr, "@implementation %s [printing todo]\n",
|
|
|
|
OID->getName());
|
2007-10-08 22:53:38 +04:00
|
|
|
} else if (ObjcCategoryImplDecl *OID =
|
|
|
|
dyn_cast<ObjcCategoryImplDecl>(D)) {
|
|
|
|
PrintObjcCategoryImplDecl(OID);
|
|
|
|
} else if (ObjcCategoryDecl *OID =
|
|
|
|
dyn_cast<ObjcCategoryDecl>(D)) {
|
|
|
|
PrintObjcCategoryDecl(OID);
|
2007-10-12 03:42:27 +04:00
|
|
|
} else if (ObjcCompatibleAliasDecl *OID =
|
|
|
|
dyn_cast<ObjcCompatibleAliasDecl>(D)) {
|
|
|
|
PrintObjcCompatibleAliasDecl(OID);
|
2007-10-06 22:52:10 +04:00
|
|
|
} else if (isa<ObjcClassDecl>(D)) {
|
|
|
|
fprintf(stderr, "@class [printing todo]\n");
|
2007-10-08 22:53:38 +04:00
|
|
|
} else if (ScopedDecl *SD = dyn_cast<ScopedDecl>(D)) {
|
|
|
|
fprintf(stderr, "Read top-level variable decl: '%s'\n", SD->getName());
|
2007-10-06 22:52:10 +04:00
|
|
|
} else {
|
|
|
|
assert(0 && "Unknown decl type!");
|
2007-08-09 02:51:59 +04:00
|
|
|
}
|
2007-07-11 21:01:13 +04:00
|
|
|
}
|
2007-09-16 03:02:28 +04:00
|
|
|
};
|
2007-07-11 21:01:13 +04:00
|
|
|
}
|
2007-08-09 02:51:59 +04:00
|
|
|
|
2007-09-16 03:02:28 +04:00
|
|
|
ASTConsumer *clang::CreateASTPrinter() { return new ASTPrinter(); }
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
class ASTDumper : public ASTConsumer {
|
|
|
|
SourceManager *SM;
|
|
|
|
public:
|
|
|
|
void Initialize(ASTContext &Context, unsigned MainFileID) {
|
|
|
|
SM = &Context.SourceMgr;
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void HandleTopLevelDecl(Decl *D) {
|
|
|
|
if (FunctionDecl *FD = dyn_cast<FunctionDecl>(D)) {
|
|
|
|
PrintFunctionDeclStart(FD);
|
|
|
|
|
|
|
|
if (FD->getBody()) {
|
|
|
|
fprintf(stderr, "\n");
|
|
|
|
FD->getBody()->dumpAll(*SM);
|
|
|
|
fprintf(stderr, "\n");
|
|
|
|
}
|
|
|
|
} else if (TypedefDecl *TD = dyn_cast<TypedefDecl>(D)) {
|
|
|
|
PrintTypeDefDecl(TD);
|
|
|
|
} else if (ScopedDecl *SD = dyn_cast<ScopedDecl>(D)) {
|
|
|
|
fprintf(stderr, "Read top-level variable decl: '%s'\n", SD->getName());
|
2007-10-06 22:52:10 +04:00
|
|
|
} else if (ObjcInterfaceDecl *OID = dyn_cast<ObjcInterfaceDecl>(D)) {
|
|
|
|
fprintf(stderr, "Read objc interface '%s'\n", OID->getName());
|
2007-10-14 21:03:01 +04:00
|
|
|
} else if (ObjcProtocolDecl *OPD = dyn_cast<ObjcProtocolDecl>(D)) {
|
|
|
|
fprintf(stderr, "Read objc protocol '%s'\n", OPD->getName());
|
|
|
|
} else if (ObjcCategoryDecl *OCD = dyn_cast<ObjcCategoryDecl>(D)) {
|
|
|
|
fprintf(stderr, "Read objc category '%s'\n", OCD->getName());
|
2007-10-06 22:52:10 +04:00
|
|
|
} else if (isa<ObjcForwardProtocolDecl>(D)) {
|
|
|
|
fprintf(stderr, "Read objc fwd protocol decl\n");
|
2007-10-14 21:03:01 +04:00
|
|
|
} else if (isa<ObjcClassDecl>(D)) {
|
|
|
|
fprintf(stderr, "Read objc fwd class decl\n");
|
2007-10-06 22:52:10 +04:00
|
|
|
} else {
|
|
|
|
assert(0 && "Unknown decl type!");
|
2007-08-09 02:51:59 +04:00
|
|
|
}
|
|
|
|
}
|
2007-09-16 03:02:28 +04:00
|
|
|
};
|
2007-08-09 02:51:59 +04:00
|
|
|
}
|
|
|
|
|
2007-09-16 03:02:28 +04:00
|
|
|
ASTConsumer *clang::CreateASTDumper() { return new ASTDumper(); }
|
|
|
|
|
2007-09-20 01:29:43 +04:00
|
|
|
namespace {
|
|
|
|
class ASTViewer : public ASTConsumer {
|
|
|
|
SourceManager *SM;
|
|
|
|
public:
|
|
|
|
void Initialize(ASTContext &Context, unsigned MainFileID) {
|
|
|
|
SM = &Context.SourceMgr;
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void HandleTopLevelDecl(Decl *D) {
|
|
|
|
if (FunctionDecl *FD = dyn_cast<FunctionDecl>(D)) {
|
|
|
|
PrintFunctionDeclStart(FD);
|
|
|
|
|
|
|
|
if (FD->getBody()) {
|
|
|
|
fprintf(stderr, "\n");
|
|
|
|
FD->getBody()->viewAST();
|
|
|
|
fprintf(stderr, "\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
ASTConsumer *clang::CreateASTViewer() { return new ASTViewer(); }
|
|
|
|
|
|
|
|
|
2007-09-08 03:47:56 +04:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// CFGVisitor & VisitCFGs - Boilerplate interface and logic to visit
|
|
|
|
// the CFGs for all function definitions.
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
2007-09-16 03:21:08 +04:00
|
|
|
class CFGVisitor : public ASTConsumer {
|
2007-09-08 03:47:56 +04:00
|
|
|
public:
|
2007-09-16 03:21:08 +04:00
|
|
|
// CFG Visitor interface to be implemented by subclass.
|
2007-09-08 03:47:56 +04:00
|
|
|
virtual void VisitCFG(CFG& C) = 0;
|
|
|
|
virtual bool printFuncDeclStart() { return true; }
|
2007-09-16 03:21:08 +04:00
|
|
|
|
|
|
|
virtual void HandleTopLevelDecl(Decl *D);
|
2007-09-08 03:47:56 +04:00
|
|
|
};
|
|
|
|
|
|
|
|
} // end anonymous namespace
|
|
|
|
|
2007-09-16 03:21:08 +04:00
|
|
|
void CFGVisitor::HandleTopLevelDecl(Decl *D) {
|
|
|
|
FunctionDecl *FD = dyn_cast<FunctionDecl>(D);
|
|
|
|
if (!FD || !FD->getBody())
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (printFuncDeclStart()) {
|
|
|
|
PrintFunctionDeclStart(FD);
|
|
|
|
fprintf(stderr,"\n");
|
2007-08-22 01:42:03 +04:00
|
|
|
}
|
2007-09-16 03:21:08 +04:00
|
|
|
|
2007-09-17 21:10:02 +04:00
|
|
|
CFG *C = CFG::buildCFG(FD->getBody());
|
|
|
|
VisitCFG(*C);
|
|
|
|
delete C;
|
2007-08-22 01:42:03 +04:00
|
|
|
}
|
|
|
|
|
2007-09-08 03:47:56 +04:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// DumpCFGs - Dump CFGs to stderr or visualize with Graphviz
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
class CFGDumper : public CFGVisitor {
|
|
|
|
const bool UseGraphviz;
|
|
|
|
public:
|
|
|
|
CFGDumper(bool use_graphviz) : UseGraphviz(use_graphviz) {}
|
|
|
|
|
2007-09-16 03:21:08 +04:00
|
|
|
virtual void VisitCFG(CFG &C) {
|
|
|
|
if (UseGraphviz)
|
|
|
|
C.viewCFG();
|
|
|
|
else
|
|
|
|
C.dump();
|
|
|
|
}
|
2007-09-08 03:47:56 +04:00
|
|
|
};
|
|
|
|
} // end anonymous namespace
|
2007-09-06 04:17:54 +04:00
|
|
|
|
2007-09-16 03:21:08 +04:00
|
|
|
ASTConsumer *clang::CreateCFGDumper(bool ViewGraphs) {
|
|
|
|
return new CFGDumper(ViewGraphs);
|
2007-09-06 04:17:54 +04:00
|
|
|
}
|
|
|
|
|
2007-09-08 03:47:56 +04:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// AnalyzeLiveVariables - perform live variable analysis and dump results
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
class LivenessVisitor : public CFGVisitor {
|
2007-09-16 03:21:08 +04:00
|
|
|
SourceManager *SM;
|
2007-09-08 03:47:56 +04:00
|
|
|
public:
|
2007-09-16 03:21:08 +04:00
|
|
|
virtual void Initialize(ASTContext &Context, unsigned MainFileID) {
|
|
|
|
SM = &Context.SourceMgr;
|
|
|
|
}
|
|
|
|
|
2007-09-08 03:47:56 +04:00
|
|
|
virtual void VisitCFG(CFG& C) {
|
2007-10-02 00:33:52 +04:00
|
|
|
LiveVariables L(C);
|
2007-09-08 03:47:56 +04:00
|
|
|
L.runOnCFG(C);
|
2007-09-25 08:31:27 +04:00
|
|
|
L.dumpBlockLiveness(*SM);
|
2007-09-07 03:00:42 +04:00
|
|
|
}
|
2007-09-08 03:47:56 +04:00
|
|
|
};
|
|
|
|
} // end anonymous namespace
|
2007-09-07 03:00:42 +04:00
|
|
|
|
2007-09-16 03:21:08 +04:00
|
|
|
ASTConsumer *clang::CreateLiveVarAnalyzer() {
|
|
|
|
return new LivenessVisitor();
|
2007-09-07 03:00:42 +04:00
|
|
|
}
|
|
|
|
|
2007-09-08 03:47:56 +04:00
|
|
|
//===----------------------------------------------------------------------===//
|
2007-09-18 00:49:30 +04:00
|
|
|
// DeadStores - run checker to locate dead stores in a function
|
2007-09-08 03:47:56 +04:00
|
|
|
|
|
|
|
namespace {
|
|
|
|
class DeadStoreVisitor : public CFGVisitor {
|
2007-09-16 03:21:08 +04:00
|
|
|
Diagnostic &Diags;
|
|
|
|
ASTContext *Ctx;
|
2007-09-08 03:47:56 +04:00
|
|
|
public:
|
2007-09-16 03:21:08 +04:00
|
|
|
DeadStoreVisitor(Diagnostic &diags) : Diags(diags) {}
|
|
|
|
virtual void Initialize(ASTContext &Context, unsigned MainFileID) {
|
|
|
|
Ctx = &Context;
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void VisitCFG(CFG& C) { CheckDeadStores(C, *Ctx, Diags); }
|
2007-09-08 03:54:15 +04:00
|
|
|
virtual bool printFuncDeclStart() { return false; }
|
2007-09-08 03:47:56 +04:00
|
|
|
};
|
|
|
|
} // end anonymous namespace
|
2007-08-09 02:51:59 +04:00
|
|
|
|
2007-09-16 03:21:08 +04:00
|
|
|
ASTConsumer *clang::CreateDeadStoreChecker(Diagnostic &Diags) {
|
|
|
|
return new DeadStoreVisitor(Diags);
|
2007-09-08 03:47:56 +04:00
|
|
|
}
|
2007-09-16 23:46:59 +04:00
|
|
|
|
2007-09-18 00:49:30 +04:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// Unitialized Values - run checker to flag potential uses of uninitalized
|
|
|
|
// variables.
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
class UninitValsVisitor : public CFGVisitor {
|
|
|
|
Diagnostic &Diags;
|
|
|
|
ASTContext *Ctx;
|
|
|
|
public:
|
|
|
|
UninitValsVisitor(Diagnostic &diags) : Diags(diags) {}
|
|
|
|
virtual void Initialize(ASTContext &Context, unsigned MainFileID) {
|
|
|
|
Ctx = &Context;
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void VisitCFG(CFG& C) { CheckUninitializedValues(C, *Ctx, Diags); }
|
|
|
|
virtual bool printFuncDeclStart() { return false; }
|
|
|
|
};
|
|
|
|
} // end anonymous namespace
|
|
|
|
|
|
|
|
ASTConsumer *clang::CreateUnitValsChecker(Diagnostic &Diags) {
|
|
|
|
return new UninitValsVisitor(Diags);
|
|
|
|
}
|
|
|
|
|
2007-09-16 23:46:59 +04:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// LLVM Emitter
|
|
|
|
|
|
|
|
#include "clang/Basic/Diagnostic.h"
|
|
|
|
#include "clang/CodeGen/ModuleBuilder.h"
|
|
|
|
#include "llvm/Module.h"
|
|
|
|
#include <iostream>
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
class LLVMEmitter : public ASTConsumer {
|
|
|
|
Diagnostic &Diags;
|
|
|
|
llvm::Module *M;
|
|
|
|
ASTContext *Ctx;
|
|
|
|
CodeGen::BuilderTy *Builder;
|
|
|
|
public:
|
|
|
|
LLVMEmitter(Diagnostic &diags) : Diags(diags) {}
|
|
|
|
virtual void Initialize(ASTContext &Context, unsigned MainFileID) {
|
|
|
|
Ctx = &Context;
|
|
|
|
M = new llvm::Module("foo");
|
|
|
|
Builder = CodeGen::Init(Context, *M);
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void HandleTopLevelDecl(Decl *D) {
|
|
|
|
// If an error occurred, stop code generation, but continue parsing and
|
|
|
|
// semantic analysis (to ensure all warnings and errors are emitted).
|
|
|
|
if (Diags.hasErrorOccurred())
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (FunctionDecl *FD = dyn_cast<FunctionDecl>(D)) {
|
|
|
|
CodeGen::CodeGenFunction(Builder, FD);
|
|
|
|
} else if (FileVarDecl *FVD = dyn_cast<FileVarDecl>(D)) {
|
|
|
|
CodeGen::CodeGenGlobalVar(Builder, FVD);
|
|
|
|
} else {
|
|
|
|
assert(isa<TypedefDecl>(D) && "Only expected typedefs here");
|
|
|
|
// don't codegen for now, eventually pass down for debug info.
|
|
|
|
//std::cerr << "Read top-level typedef decl: '" << D->getName() << "'\n";
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
~LLVMEmitter() {
|
|
|
|
CodeGen::Terminate(Builder);
|
|
|
|
|
|
|
|
// Print the generated code.
|
|
|
|
M->print(std::cout);
|
|
|
|
delete M;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
} // end anonymous namespace
|
|
|
|
|
|
|
|
ASTConsumer *clang::CreateLLVMEmitter(Diagnostic &Diags) {
|
|
|
|
return new LLVMEmitter(Diags);
|
|
|
|
}
|
|
|
|
|