mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2025-01-01 00:33:09 +00:00
c10fa6c801
http://lists.cs.uiuc.edu/pipermail/llvm-commits/Week-of-Mon-20120130/136146.html Implemented CaseIterator and it solves almost all described issues: we don't need to mix operand/case/successor indexing anymore. Base iterator class is implemented as a template since it may be initialized either from "const SwitchInst*" or from "SwitchInst*". ConstCaseIt is just a read-only iterator. CaseIt is read-write iterator; it allows to change case successor and case value. Usage of iterator allows totally remove resolveXXXX methods. All indexing convertions done automatically inside the iterator's getters. Main way of iterator usage looks like this: SwitchInst *SI = ... // intialize it somehow for (SwitchInst::CaseIt i = SI->caseBegin(), e = SI->caseEnd(); i != e; ++i) { BasicBlock *BB = i.getCaseSuccessor(); ConstantInt *V = i.getCaseValue(); // Do something. } If you want to convert case number to TerminatorInst successor index, just use getSuccessorIndex iterator's method. If you want initialize iterator from TerminatorInst successor index, use CaseIt::fromSuccessorIndex(...) method. There are also related changes in llvm-clients: klee and clang. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@152297 91177308-0d34-0410-b5e6-96231b3b80d8
115 lines
3.5 KiB
C++
115 lines
3.5 KiB
C++
//===-- CFGPrinter.h - CFG printer external interface -----------*- 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 external functions that can be called to explicitly
|
|
// instantiate the CFG printer.
|
|
//
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
#ifndef LLVM_ANALYSIS_CFGPRINTER_H
|
|
#define LLVM_ANALYSIS_CFGPRINTER_H
|
|
|
|
#include "llvm/Constants.h"
|
|
#include "llvm/Function.h"
|
|
#include "llvm/Instructions.h"
|
|
#include "llvm/Assembly/Writer.h"
|
|
#include "llvm/Support/CFG.h"
|
|
#include "llvm/Support/GraphWriter.h"
|
|
|
|
namespace llvm {
|
|
template<>
|
|
struct DOTGraphTraits<const Function*> : public DefaultDOTGraphTraits {
|
|
|
|
DOTGraphTraits (bool isSimple=false) : DefaultDOTGraphTraits(isSimple) {}
|
|
|
|
static std::string getGraphName(const Function *F) {
|
|
return "CFG for '" + F->getName().str() + "' function";
|
|
}
|
|
|
|
static std::string getSimpleNodeLabel(const BasicBlock *Node,
|
|
const Function *) {
|
|
if (!Node->getName().empty())
|
|
return Node->getName().str();
|
|
|
|
std::string Str;
|
|
raw_string_ostream OS(Str);
|
|
|
|
WriteAsOperand(OS, Node, false);
|
|
return OS.str();
|
|
}
|
|
|
|
static std::string getCompleteNodeLabel(const BasicBlock *Node,
|
|
const Function *) {
|
|
std::string Str;
|
|
raw_string_ostream OS(Str);
|
|
|
|
if (Node->getName().empty()) {
|
|
WriteAsOperand(OS, Node, false);
|
|
OS << ":";
|
|
}
|
|
|
|
OS << *Node;
|
|
std::string OutStr = OS.str();
|
|
if (OutStr[0] == '\n') OutStr.erase(OutStr.begin());
|
|
|
|
// Process string output to make it nicer...
|
|
for (unsigned i = 0; i != OutStr.length(); ++i)
|
|
if (OutStr[i] == '\n') { // Left justify
|
|
OutStr[i] = '\\';
|
|
OutStr.insert(OutStr.begin()+i+1, 'l');
|
|
} else if (OutStr[i] == ';') { // Delete comments!
|
|
unsigned Idx = OutStr.find('\n', i+1); // Find end of line
|
|
OutStr.erase(OutStr.begin()+i, OutStr.begin()+Idx);
|
|
--i;
|
|
}
|
|
|
|
return OutStr;
|
|
}
|
|
|
|
std::string getNodeLabel(const BasicBlock *Node,
|
|
const Function *Graph) {
|
|
if (isSimple())
|
|
return getSimpleNodeLabel(Node, Graph);
|
|
else
|
|
return getCompleteNodeLabel(Node, Graph);
|
|
}
|
|
|
|
static std::string getEdgeSourceLabel(const BasicBlock *Node,
|
|
succ_const_iterator I) {
|
|
// Label source of conditional branches with "T" or "F"
|
|
if (const BranchInst *BI = dyn_cast<BranchInst>(Node->getTerminator()))
|
|
if (BI->isConditional())
|
|
return (I == succ_begin(Node)) ? "T" : "F";
|
|
|
|
// Label source of switch edges with the associated value.
|
|
if (const SwitchInst *SI = dyn_cast<SwitchInst>(Node->getTerminator())) {
|
|
unsigned SuccNo = I.getSuccessorIndex();
|
|
|
|
if (SuccNo == 0) return "def";
|
|
|
|
std::string Str;
|
|
raw_string_ostream OS(Str);
|
|
SwitchInst::ConstCaseIt Case =
|
|
SwitchInst::ConstCaseIt::fromSuccessorIndex(SI, SuccNo);
|
|
OS << Case.getCaseValue()->getValue();
|
|
return OS.str();
|
|
}
|
|
return "";
|
|
}
|
|
};
|
|
} // End llvm namespace
|
|
|
|
namespace llvm {
|
|
class FunctionPass;
|
|
FunctionPass *createCFGPrinterPass ();
|
|
FunctionPass *createCFGOnlyPrinterPass ();
|
|
} // End llvm namespace
|
|
|
|
#endif
|