2007-10-07 14:04:32 +08:00
|
|
|
//===--- ASTConsumers.cpp - ASTConsumer implementations -------------------===//
|
2007-06-23 08:39:57 +08:00
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
2007-10-07 14:04:32 +08:00
|
|
|
// This file was developed by Chris Lattner and is distributed under the
|
2007-06-23 08:39:57 +08:00
|
|
|
// University of Illinois Open Source License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
2007-10-07 14:04:32 +08:00
|
|
|
// AST Consumer Implementations.
|
2007-06-23 08:39:57 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
2007-10-07 14:04:32 +08:00
|
|
|
#include "ASTConsumers.h"
|
2007-06-23 08:39:57 +08:00
|
|
|
#include "clang/AST/AST.h"
|
2007-09-16 07:02:28 +08:00
|
|
|
#include "clang/AST/ASTConsumer.h"
|
2007-08-22 05:42:03 +08:00
|
|
|
#include "clang/AST/CFG.h"
|
2007-09-06 08:17:54 +08:00
|
|
|
#include "clang/Analysis/LiveVariables.h"
|
2007-09-07 07:00:42 +08:00
|
|
|
#include "clang/Analysis/LocalCheckers.h"
|
2007-08-09 06:51:59 +08:00
|
|
|
using namespace clang;
|
2007-06-23 08:39:57 +08:00
|
|
|
|
2007-08-09 06:51:59 +08:00
|
|
|
|
|
|
|
static void PrintFunctionDeclStart(FunctionDecl *FD) {
|
2007-06-23 08:39:57 +08:00
|
|
|
bool HasBody = FD->getBody();
|
|
|
|
|
2007-08-26 12:02:13 +08: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-06-23 08:39:57 +08: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 12:02:13 +08:00
|
|
|
fprintf(stderr, "%s", Proto.c_str());
|
2007-06-23 08:39:57 +08:00
|
|
|
|
2007-08-09 06:51:59 +08:00
|
|
|
if (!FD->getBody())
|
2007-06-23 08:39:57 +08:00
|
|
|
fprintf(stderr, ";\n");
|
2007-08-09 06:51:59 +08:00
|
|
|
// Doesn't print the body.
|
2007-06-23 08:39:57 +08:00
|
|
|
}
|
|
|
|
|
2007-08-09 06:51:59 +08:00
|
|
|
static void PrintTypeDefDecl(TypedefDecl *TD) {
|
2007-06-23 08:39:57 +08:00
|
|
|
std::string S = TD->getName();
|
|
|
|
TD->getUnderlyingType().getAsStringInternal(S);
|
|
|
|
fprintf(stderr, "typedef %s;\n", S.c_str());
|
|
|
|
}
|
|
|
|
|
2007-11-11 04:59:13 +08:00
|
|
|
static void PrintObjcMethodDecl(ObjcMethodDecl *OMD) {
|
|
|
|
if (OMD->isInstance())
|
|
|
|
fprintf(stderr, "\n- ");
|
|
|
|
else
|
|
|
|
fprintf(stderr, "\n+ ");
|
|
|
|
if (!OMD->getResultType().isNull())
|
|
|
|
fprintf(stderr, "(%s) ", OMD->getResultType().getAsString().c_str());
|
|
|
|
// FIXME: just print original selector name!
|
|
|
|
fprintf(stderr, "%s ", OMD->getSelector().getName().c_str());
|
|
|
|
|
|
|
|
for (int i = 0; i < OMD->getNumParams(); i++) {
|
|
|
|
ParmVarDecl *PDecl = OMD->getParamDecl(i);
|
|
|
|
// FIXME: selector is missing here!
|
|
|
|
fprintf(stderr, " :(%s) %s", PDecl->getType().getAsString().c_str(),
|
|
|
|
PDecl->getName());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
static void PrintObjcImplementationDecl(ObjcImplementationDecl *OID) {
|
|
|
|
std::string I = OID->getName();
|
|
|
|
ObjcInterfaceDecl *SID = OID->getSuperClass();
|
|
|
|
if (SID) {
|
|
|
|
std::string S = SID->getName();
|
|
|
|
fprintf(stderr, "@implementation %s : %s", I.c_str(), S.c_str());
|
|
|
|
}
|
|
|
|
else
|
|
|
|
fprintf(stderr, "@implementation %s", I.c_str());
|
|
|
|
|
|
|
|
for (int i = 0; i < OID->getNumInstanceMethods(); i++) {
|
|
|
|
PrintObjcMethodDecl(OID->getInstanceMethods()[i]);
|
|
|
|
ObjcMethodDecl *OMD = OID->getInstanceMethods()[i];
|
|
|
|
if (OMD->getBody()) {
|
|
|
|
fprintf(stderr, " ");
|
|
|
|
OMD->getBody()->dumpPretty();
|
|
|
|
fprintf(stderr, "\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for (int i = 0; i < OID->getNumClassMethods(); i++) {
|
|
|
|
PrintObjcMethodDecl(OID->getClassMethods()[i]);
|
|
|
|
ObjcMethodDecl *OMD = OID->getClassMethods()[i];
|
|
|
|
if (OMD->getBody()) {
|
|
|
|
fprintf(stderr, " ");
|
|
|
|
OMD->getBody()->dumpPretty();
|
|
|
|
fprintf(stderr, "\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fprintf(stderr,"@end\n");
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2007-09-11 04:51:04 +08:00
|
|
|
static void PrintObjcInterfaceDecl(ObjcInterfaceDecl *OID) {
|
2007-10-09 07:06:41 +08: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)
|
2007-10-27 00:29:12 +08:00
|
|
|
fprintf(stderr, ">\n");
|
2007-10-09 07:06:41 +08:00
|
|
|
else
|
2007-10-27 00:29:12 +08:00
|
|
|
fprintf(stderr, "\n");
|
|
|
|
|
2007-11-12 21:56:41 +08:00
|
|
|
int NumIvars = OID->getNumInstanceVariables();
|
2007-10-27 00:29:12 +08:00
|
|
|
if (NumIvars > 0) {
|
2007-11-12 21:56:41 +08:00
|
|
|
ObjcIvarDecl **Ivars = OID->getInstanceVariables();
|
2007-10-27 00:29:12 +08:00
|
|
|
fprintf(stderr,"{");
|
|
|
|
for (int i = 0; i < NumIvars; i++) {
|
|
|
|
fprintf(stderr, "\t%s %s;\n", Ivars[i]->getType().getAsString().c_str(),
|
|
|
|
Ivars[i]->getName());
|
|
|
|
}
|
2007-11-07 06:01:00 +08:00
|
|
|
fprintf(stderr, "}\n");
|
2007-10-27 00:29:12 +08:00
|
|
|
}
|
2007-11-07 06:01:00 +08:00
|
|
|
|
|
|
|
int NumProperties = OID->getNumPropertyDecl();
|
|
|
|
if (NumProperties > 0) {
|
|
|
|
for (int i = 0; i < NumProperties; i++) {
|
|
|
|
ObjcPropertyDecl *PDecl = OID->getPropertyDecl()[i];
|
|
|
|
fprintf(stderr, "@property");
|
|
|
|
if (PDecl->getPropertyAttributes() != ObjcPropertyDecl::OBJC_PR_noattr) {
|
|
|
|
bool first = true;
|
|
|
|
fprintf(stderr, " (");
|
|
|
|
if (PDecl->getPropertyAttributes() & ObjcPropertyDecl::OBJC_PR_readonly)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "%creadonly", first ? ' ' : ',');
|
|
|
|
first = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (PDecl->getPropertyAttributes() & ObjcPropertyDecl::OBJC_PR_getter)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "%cgetter = %s", first ? ' ' : ','
|
|
|
|
, PDecl->getGetterName()->getName());
|
|
|
|
first = false;
|
|
|
|
}
|
|
|
|
if (PDecl->getPropertyAttributes() & ObjcPropertyDecl::OBJC_PR_setter)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "%csetter = %s:", first ? ' ' : ','
|
|
|
|
, PDecl->getSetterName()->getName());
|
|
|
|
first = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (PDecl->getPropertyAttributes() & ObjcPropertyDecl::OBJC_PR_assign)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "%cassign", first ? ' ' : ',');
|
|
|
|
first = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (PDecl->getPropertyAttributes() & ObjcPropertyDecl::OBJC_PR_readwrite)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "%creadwrite", first ? ' ' : ',');
|
|
|
|
first = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (PDecl->getPropertyAttributes() & ObjcPropertyDecl::OBJC_PR_retain)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "%cretain", first ? ' ' : ',');
|
|
|
|
first = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (PDecl->getPropertyAttributes() & ObjcPropertyDecl::OBJC_PR_copy)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "%ccopy", first ? ' ' : ',');
|
|
|
|
first = false;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (PDecl->getPropertyAttributes() & ObjcPropertyDecl::OBJC_PR_nonatomic)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "%cnonatomic", first ? ' ' : ',');
|
|
|
|
first = false;
|
|
|
|
}
|
|
|
|
fprintf(stderr, " )");
|
|
|
|
}
|
|
|
|
ObjcIvarDecl **IDecl = PDecl->getPropertyDecls();
|
|
|
|
fprintf(stderr, " %s %s", IDecl[0]->getType().getAsString().c_str(),
|
|
|
|
IDecl[0]->getName());
|
|
|
|
|
|
|
|
for (int j = 1; j < PDecl->getNumPropertyDecls(); j++) {
|
|
|
|
fprintf(stderr, ", %s", IDecl[j]->getName());
|
|
|
|
}
|
|
|
|
fprintf(stderr, ";\n");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
fprintf(stderr,"@end\n");
|
2007-09-11 04:51:04 +08:00
|
|
|
// FIXME: implement the rest...
|
|
|
|
}
|
|
|
|
|
2007-10-09 02:53:38 +08: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 07:42:27 +08: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 07:02:28 +08: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");
|
|
|
|
}
|
2007-11-14 07:48:03 +08:00
|
|
|
} else if (isa<ObjcMethodDecl>(D)) {
|
|
|
|
// Do nothing, methods definitions are printed in
|
|
|
|
// PrintObjcImplementationDecl.
|
2007-09-16 07:02:28 +08:00
|
|
|
} else if (TypedefDecl *TD = dyn_cast<TypedefDecl>(D)) {
|
|
|
|
PrintTypeDefDecl(TD);
|
2007-10-07 02:52:10 +08:00
|
|
|
} else if (ObjcInterfaceDecl *OID = dyn_cast<ObjcInterfaceDecl>(D)) {
|
|
|
|
PrintObjcInterfaceDecl(OID);
|
2007-10-09 02:53:38 +08:00
|
|
|
} else if (ObjcProtocolDecl *PID = dyn_cast<ObjcProtocolDecl>(D)) {
|
|
|
|
PrintObjcProtocolDecl(PID);
|
2007-10-07 02:52:10 +08: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-07 03:08:22 +08:00
|
|
|
fprintf(stderr, ";\n");
|
2007-10-07 02:52:10 +08:00
|
|
|
} else if (ObjcImplementationDecl *OID =
|
|
|
|
dyn_cast<ObjcImplementationDecl>(D)) {
|
2007-11-11 04:59:13 +08:00
|
|
|
PrintObjcImplementationDecl(OID);
|
2007-10-09 02:53:38 +08:00
|
|
|
} else if (ObjcCategoryImplDecl *OID =
|
|
|
|
dyn_cast<ObjcCategoryImplDecl>(D)) {
|
|
|
|
PrintObjcCategoryImplDecl(OID);
|
|
|
|
} else if (ObjcCategoryDecl *OID =
|
|
|
|
dyn_cast<ObjcCategoryDecl>(D)) {
|
|
|
|
PrintObjcCategoryDecl(OID);
|
2007-10-12 07:42:27 +08:00
|
|
|
} else if (ObjcCompatibleAliasDecl *OID =
|
|
|
|
dyn_cast<ObjcCompatibleAliasDecl>(D)) {
|
|
|
|
PrintObjcCompatibleAliasDecl(OID);
|
2007-10-07 02:52:10 +08:00
|
|
|
} else if (isa<ObjcClassDecl>(D)) {
|
|
|
|
fprintf(stderr, "@class [printing todo]\n");
|
2007-10-09 02:53:38 +08:00
|
|
|
} else if (ScopedDecl *SD = dyn_cast<ScopedDecl>(D)) {
|
|
|
|
fprintf(stderr, "Read top-level variable decl: '%s'\n", SD->getName());
|
2007-10-07 02:52:10 +08:00
|
|
|
} else {
|
|
|
|
assert(0 && "Unknown decl type!");
|
2007-08-09 06:51:59 +08:00
|
|
|
}
|
2007-06-23 08:39:57 +08:00
|
|
|
}
|
2007-09-16 07:02:28 +08:00
|
|
|
};
|
2007-06-23 08:39:57 +08:00
|
|
|
}
|
2007-08-09 06:51:59 +08:00
|
|
|
|
2007-09-16 07:02:28 +08: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-07 02:52:10 +08:00
|
|
|
} else if (ObjcInterfaceDecl *OID = dyn_cast<ObjcInterfaceDecl>(D)) {
|
|
|
|
fprintf(stderr, "Read objc interface '%s'\n", OID->getName());
|
2007-10-15 01:03:01 +08: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-07 02:52:10 +08:00
|
|
|
} else if (isa<ObjcForwardProtocolDecl>(D)) {
|
|
|
|
fprintf(stderr, "Read objc fwd protocol decl\n");
|
2007-10-15 01:03:01 +08:00
|
|
|
} else if (isa<ObjcClassDecl>(D)) {
|
|
|
|
fprintf(stderr, "Read objc fwd class decl\n");
|
2007-10-07 02:52:10 +08:00
|
|
|
} else {
|
|
|
|
assert(0 && "Unknown decl type!");
|
2007-08-09 06:51:59 +08:00
|
|
|
}
|
|
|
|
}
|
2007-09-16 07:02:28 +08:00
|
|
|
};
|
2007-08-09 06:51:59 +08:00
|
|
|
}
|
|
|
|
|
2007-09-16 07:02:28 +08:00
|
|
|
ASTConsumer *clang::CreateASTDumper() { return new ASTDumper(); }
|
|
|
|
|
2007-09-20 05:29:43 +08: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 07:47:56 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// CFGVisitor & VisitCFGs - Boilerplate interface and logic to visit
|
|
|
|
// the CFGs for all function definitions.
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
2007-09-16 07:21:08 +08:00
|
|
|
class CFGVisitor : public ASTConsumer {
|
2007-09-08 07:47:56 +08:00
|
|
|
public:
|
2007-09-16 07:21:08 +08:00
|
|
|
// CFG Visitor interface to be implemented by subclass.
|
2007-09-08 07:47:56 +08:00
|
|
|
virtual void VisitCFG(CFG& C) = 0;
|
|
|
|
virtual bool printFuncDeclStart() { return true; }
|
2007-09-16 07:21:08 +08:00
|
|
|
|
|
|
|
virtual void HandleTopLevelDecl(Decl *D);
|
2007-09-08 07:47:56 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
} // end anonymous namespace
|
|
|
|
|
2007-09-16 07:21:08 +08: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 05:42:03 +08:00
|
|
|
}
|
2007-09-16 07:21:08 +08:00
|
|
|
|
2007-09-18 01:10:02 +08:00
|
|
|
CFG *C = CFG::buildCFG(FD->getBody());
|
|
|
|
VisitCFG(*C);
|
|
|
|
delete C;
|
2007-08-22 05:42:03 +08:00
|
|
|
}
|
|
|
|
|
2007-09-08 07:47:56 +08: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 07:21:08 +08:00
|
|
|
virtual void VisitCFG(CFG &C) {
|
|
|
|
if (UseGraphviz)
|
|
|
|
C.viewCFG();
|
|
|
|
else
|
|
|
|
C.dump();
|
|
|
|
}
|
2007-09-08 07:47:56 +08:00
|
|
|
};
|
|
|
|
} // end anonymous namespace
|
2007-09-06 08:17:54 +08:00
|
|
|
|
2007-09-16 07:21:08 +08:00
|
|
|
ASTConsumer *clang::CreateCFGDumper(bool ViewGraphs) {
|
|
|
|
return new CFGDumper(ViewGraphs);
|
2007-09-06 08:17:54 +08:00
|
|
|
}
|
|
|
|
|
2007-09-08 07:47:56 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// AnalyzeLiveVariables - perform live variable analysis and dump results
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
class LivenessVisitor : public CFGVisitor {
|
2007-09-16 07:21:08 +08:00
|
|
|
SourceManager *SM;
|
2007-09-08 07:47:56 +08:00
|
|
|
public:
|
2007-09-16 07:21:08 +08:00
|
|
|
virtual void Initialize(ASTContext &Context, unsigned MainFileID) {
|
|
|
|
SM = &Context.SourceMgr;
|
|
|
|
}
|
|
|
|
|
2007-09-08 07:47:56 +08:00
|
|
|
virtual void VisitCFG(CFG& C) {
|
2007-10-02 04:33:52 +08:00
|
|
|
LiveVariables L(C);
|
2007-09-08 07:47:56 +08:00
|
|
|
L.runOnCFG(C);
|
2007-09-25 12:31:27 +08:00
|
|
|
L.dumpBlockLiveness(*SM);
|
2007-09-07 07:00:42 +08:00
|
|
|
}
|
2007-09-08 07:47:56 +08:00
|
|
|
};
|
|
|
|
} // end anonymous namespace
|
2007-09-07 07:00:42 +08:00
|
|
|
|
2007-09-16 07:21:08 +08:00
|
|
|
ASTConsumer *clang::CreateLiveVarAnalyzer() {
|
|
|
|
return new LivenessVisitor();
|
2007-09-07 07:00:42 +08:00
|
|
|
}
|
|
|
|
|
2007-09-08 07:47:56 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
2007-09-18 04:49:30 +08:00
|
|
|
// DeadStores - run checker to locate dead stores in a function
|
2007-09-08 07:47:56 +08:00
|
|
|
|
|
|
|
namespace {
|
|
|
|
class DeadStoreVisitor : public CFGVisitor {
|
2007-09-16 07:21:08 +08:00
|
|
|
Diagnostic &Diags;
|
|
|
|
ASTContext *Ctx;
|
2007-09-08 07:47:56 +08:00
|
|
|
public:
|
2007-09-16 07:21:08 +08: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 07:54:15 +08:00
|
|
|
virtual bool printFuncDeclStart() { return false; }
|
2007-09-08 07:47:56 +08:00
|
|
|
};
|
|
|
|
} // end anonymous namespace
|
2007-08-09 06:51:59 +08:00
|
|
|
|
2007-09-16 07:21:08 +08:00
|
|
|
ASTConsumer *clang::CreateDeadStoreChecker(Diagnostic &Diags) {
|
|
|
|
return new DeadStoreVisitor(Diags);
|
2007-09-08 07:47:56 +08:00
|
|
|
}
|
2007-09-17 03:46:59 +08:00
|
|
|
|
2007-09-18 04:49:30 +08: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-17 03:46:59 +08:00
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// LLVM Emitter
|
|
|
|
|
|
|
|
#include "clang/Basic/Diagnostic.h"
|
2007-11-01 04:01:01 +08:00
|
|
|
#include "clang/Basic/TargetInfo.h"
|
2007-09-17 03:46:59 +08:00
|
|
|
#include "clang/CodeGen/ModuleBuilder.h"
|
|
|
|
#include "llvm/Module.h"
|
2007-11-01 04:01:01 +08:00
|
|
|
#include "llvm/Target/TargetData.h"
|
|
|
|
#include "llvm/Target/TargetMachine.h"
|
2007-09-17 03:46:59 +08:00
|
|
|
#include <iostream>
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
class LLVMEmitter : public ASTConsumer {
|
|
|
|
Diagnostic &Diags;
|
|
|
|
llvm::Module *M;
|
2007-11-01 04:01:01 +08:00
|
|
|
const llvm::TargetData *TD;
|
2007-09-17 03:46:59 +08:00
|
|
|
ASTContext *Ctx;
|
2007-11-14 02:16:41 +08:00
|
|
|
CodeGen::CodeGenModule *Builder;
|
2007-09-17 03:46:59 +08:00
|
|
|
public:
|
|
|
|
LLVMEmitter(Diagnostic &diags) : Diags(diags) {}
|
|
|
|
virtual void Initialize(ASTContext &Context, unsigned MainFileID) {
|
|
|
|
Ctx = &Context;
|
|
|
|
M = new llvm::Module("foo");
|
2007-11-01 04:01:01 +08:00
|
|
|
M->setTargetTriple(Ctx->Target.getTargetTriple());
|
|
|
|
TD = new llvm::TargetData(Ctx->Target.getTargetDescription());
|
|
|
|
Builder = CodeGen::Init(Context, *M, *TD);
|
2007-09-17 03:46:59 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
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 {
|
2007-11-18 05:21:01 +08:00
|
|
|
assert(isa<TypeDecl>(D) && "Only expected type decls here");
|
2007-09-17 03:46:59 +08:00
|
|
|
// 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);
|
|
|
|
}
|
|
|
|
|