blob: b30f1e80ef2f53f63e7c529bb89439505744355e [file] [log] [blame]
//===--- RewriteTest.cpp - Playground for the code rewriter ---------------===//
//
// The LLVM Compiler Infrastructure
//
// This file is distributed under the University of Illinois Open Source
// License. See LICENSE.TXT for details.
//
//===----------------------------------------------------------------------===//
//
// Hacks and fun related to the code rewriter.
//
//===----------------------------------------------------------------------===//
#include "ASTConsumers.h"
#include "clang/AST/ASTConsumer.h"
#include "clang/Rewrite/Rewriter.h"
#include "clang/Rewrite/HTMLRewrite.h"
#include "clang/Basic/SourceManager.h"
#include "llvm/Support/MemoryBuffer.h"
#include "clang/AST/ASTContext.h"
#include "clang/Basic/Diagnostic.h"
#include "clang/Analysis/LocalCheckers.h"
#include "clang/AST/CFG.h"
#include <sstream>
using namespace clang;
//===----------------------------------------------------------------------===//
// Functional HTML pretty-printing.
//===----------------------------------------------------------------------===//
namespace {
class HTMLPrinter : public ASTConsumer {
Rewriter R;
public:
HTMLPrinter() {}
virtual ~HTMLPrinter();
void Initialize(ASTContext &context);
};
}
ASTConsumer* clang::CreateHTMLPrinter() { return new HTMLPrinter(); }
void HTMLPrinter::Initialize(ASTContext &context) {
R.setSourceMgr(context.getSourceManager());
}
HTMLPrinter::~HTMLPrinter() {
unsigned FileID = R.getSourceMgr().getMainFileID();
html::EscapeText(R, FileID);
html::AddLineNumbers(R, FileID);
html::AddHeaderFooterInternalBuiltinCSS(R, FileID);
// Emit the HTML.
if (const RewriteBuffer *RewriteBuf = R.getRewriteBufferFor(FileID)) {
std::string S(RewriteBuf->begin(), RewriteBuf->end());
printf("%s\n", S.c_str());
}
}
//===----------------------------------------------------------------------===//
// Other HTML pretty-printing code used to test new features.
//===----------------------------------------------------------------------===//
namespace {
class HTMLTest : public ASTConsumer {
Rewriter R;
ASTContext* Ctx;
public:
HTMLTest() : Ctx(NULL) {}
virtual ~HTMLTest();
virtual void HandleTopLevelDecl(Decl* D);
void Initialize(ASTContext &context);
void ProcessBody(Stmt* S);
};
}
ASTConsumer* clang::CreateHTMLTest() { return new HTMLTest(); }
void HTMLTest::Initialize(ASTContext &context) {
Ctx = &context;
R.setSourceMgr(context.getSourceManager());
}
void HTMLTest::HandleTopLevelDecl(Decl* D) {
if (FunctionDecl *FD = dyn_cast<FunctionDecl>(D))
if (Stmt* B = FD->getBody()) {
SourceLocation L = B->getLocStart();
if (L.isFileID() && L.getFileID() == R.getSourceMgr().getMainFileID())
ProcessBody(B);
}
}
HTMLTest::~HTMLTest() {
unsigned FileID = R.getSourceMgr().getMainFileID();
html::EscapeText(R, FileID);
html::AddLineNumbers(R, FileID);
html::AddHeaderFooterInternalBuiltinCSS(R, FileID);
// Emit the HTML.
if (const RewriteBuffer *RewriteBuf = R.getRewriteBufferFor(FileID)) {
std::string S(RewriteBuf->begin(), RewriteBuf->end());
printf("%s\n", S.c_str());
}
}
namespace {
class HTMLDiagnostic : public DiagnosticClient {
Rewriter& R;
public:
HTMLDiagnostic(Rewriter& r) : R(r) {}
virtual void HandleDiagnostic(Diagnostic &Diags,
Diagnostic::Level DiagLevel,
FullSourceLoc Pos,
diag::kind ID,
const std::string *Strs,
unsigned NumStrs,
const SourceRange *Ranges,
unsigned NumRanges);
};
}
void HTMLTest::ProcessBody(Stmt* S) {
CFG* cfg = CFG::buildCFG(S);
if (!cfg)
return;
HTMLDiagnostic HD(R);
Diagnostic D(HD);
CheckDeadStores(*cfg, *Ctx, D);
}
void HTMLDiagnostic::HandleDiagnostic(Diagnostic &Diags,
Diagnostic::Level DiagLevel,
FullSourceLoc Pos,
diag::kind ID,
const std::string *Strs,
unsigned NumStrs,
const SourceRange *Ranges,
unsigned NumRanges) {
// For now, just draw a box above the line in question, and emit the
// warning.
if (!Pos.isValid())
return;
SourceManager& SM = R.getSourceMgr();
FullSourceLoc LPos = Pos.getLogicalLoc();
unsigned FileID = LPos.getLocation().getFileID();
assert (&LPos.getManager() == &SM && "SourceManagers are different!");
unsigned MainFileID = SM.getMainFileID();
if (FileID != MainFileID)
return;
// Compute the column number. Rewind from the current position to the start
// of the line.
unsigned ColNo = LPos.getColumnNumber();
const char *TokLogicalPtr = LPos.getCharacterData();
const char *LineStart = TokLogicalPtr-ColNo;
// Ripped from TextDiagnostics::FormatDiagnostic:
std::string Msg = Diags.getDescription(ID);
for (unsigned i = 0; i < Msg.size() - 1; ++i) {
if (Msg[i] == '%' && isdigit(Msg[i + 1])) {
unsigned StrNo = Msg[i + 1] - '0';
Msg = std::string(Msg.begin(), Msg.begin() + i) +
(StrNo < NumStrs ? Strs[StrNo] : "<<<INTERNAL ERROR>>>") +
std::string(Msg.begin() + i + 2, Msg.end());
}
}
// Create the html for the message.
std::ostringstream os;
os << "\n<tr><td class=\"num\"></td><td class=\"line\">"
<< "<div class=\"msg\" style=\"margin-left:"
<< ColNo << "ex\">";
switch (DiagLevel) {
default: assert(0 && "Unknown diagnostic type!");
case Diagnostic::Note: os << "note: "; break;
case Diagnostic::Warning: os << "warning: "; break;
case Diagnostic::Error: os << "error: "; break;
case Diagnostic::Fatal: os << "fatal error: "; break;
break;
}
os << Msg << "</div></td></tr>";
// Insert the new html.
const llvm::MemoryBuffer *Buf = SM.getBuffer(FileID);
const char* FileStart = Buf->getBufferStart();
R.InsertStrBefore(SourceLocation::getFileLoc(FileID, LineStart - FileStart),
os.str());
// Now highlight the ranges.
for (unsigned i = 0; i < NumRanges; ++i) {
SourceLocation B = SM.getLogicalLoc(Ranges->getBegin());
SourceLocation E = SM.getLogicalLoc(Ranges->getEnd());
// We do this because the position seems to point to the beginning of
// the last character. FIXME: Is this what is suppose to happen?
std::pair<unsigned,unsigned> X = SM.getDecomposedFileLoc(E);
E = SourceLocation::getFileLoc(X.first, X.second+1);
++Ranges;
if (B.getFileID() != MainFileID || E.getFileID() != MainFileID)
continue;
// Highlight the range. Make the span tag the outermost tag for the
// selected range.
R.InsertCStrBefore(B, "<span class=\"mrange\">");
R.InsertCStrAfter(E, "</span>");
}
}