2002-07-16 18:58:08 +00:00
|
|
|
//===-- EmitFunctions.cpp - interface to insert instrumentation --*- C++ -*--=//
|
|
|
|
//
|
|
|
|
// This inserts a global constant table with function pointers all along
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "llvm/Constants.h"
|
|
|
|
#include "llvm/DerivedTypes.h"
|
|
|
|
#include "llvm/Module.h"
|
2003-01-14 22:39:29 +00:00
|
|
|
#include "llvm/Pass.h"
|
2003-07-18 20:55:26 +00:00
|
|
|
#include "llvm/Support/CFG.h"
|
|
|
|
|
|
|
|
enum Color{
|
|
|
|
WHITE,
|
|
|
|
GREY,
|
|
|
|
BLACK
|
|
|
|
};
|
2002-07-16 18:58:08 +00:00
|
|
|
|
2002-07-23 18:06:35 +00:00
|
|
|
namespace {
|
|
|
|
struct EmitFunctionTable : public Pass {
|
|
|
|
bool run(Module &M);
|
|
|
|
};
|
|
|
|
|
2002-07-26 21:12:46 +00:00
|
|
|
RegisterOpt<EmitFunctionTable> X("emitfuncs", "Emit a Function Table");
|
2002-07-23 18:06:35 +00:00
|
|
|
}
|
2002-07-16 18:58:08 +00:00
|
|
|
|
2003-07-18 20:55:26 +00:00
|
|
|
char doDFS(BasicBlock * node,std::map<BasicBlock *, Color > &color){
|
|
|
|
color[node] = GREY;
|
|
|
|
|
2003-09-24 22:07:33 +00:00
|
|
|
for(succ_iterator vl = succ_begin(node), ve = succ_end(node); vl != ve; ++vl){
|
2003-07-18 20:55:26 +00:00
|
|
|
|
|
|
|
BasicBlock *BB = *vl;
|
|
|
|
|
|
|
|
if(color[BB]!=GREY && color[BB]!=BLACK){
|
|
|
|
if(!doDFS(BB, color)){
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
//if has backedge
|
|
|
|
else if(color[BB]==GREY)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
color[node] = BLACK;
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
char hasBackEdge(Function *F){
|
|
|
|
std::map<BasicBlock *, Color > color;
|
|
|
|
return doDFS(F->begin(), color);
|
|
|
|
}
|
|
|
|
|
2002-07-16 18:58:08 +00:00
|
|
|
// Per Module pass for inserting function table
|
|
|
|
bool EmitFunctionTable::run(Module &M){
|
2003-05-22 22:00:07 +00:00
|
|
|
std::vector<const Type*> vType;
|
2003-07-18 20:55:26 +00:00
|
|
|
|
2003-05-22 22:00:07 +00:00
|
|
|
std::vector<Constant *> vConsts;
|
2003-07-18 20:55:26 +00:00
|
|
|
std::vector<Constant *> sBCons;
|
|
|
|
|
|
|
|
unsigned int counter = 0;
|
2003-05-22 22:00:07 +00:00
|
|
|
for(Module::iterator MI = M.begin(), ME = M.end(); MI != ME; ++MI)
|
2002-07-16 18:58:08 +00:00
|
|
|
if (!MI->isExternal()) {
|
|
|
|
vType.push_back(MI->getType());
|
2003-07-18 20:55:26 +00:00
|
|
|
|
|
|
|
//std::cerr<<MI;
|
|
|
|
|
2003-05-22 22:00:07 +00:00
|
|
|
vConsts.push_back(ConstantPointerRef::get(MI));
|
2003-07-18 20:55:26 +00:00
|
|
|
sBCons.push_back(ConstantInt::get(Type::SByteTy, hasBackEdge(MI)));
|
|
|
|
|
2003-06-01 02:40:49 +00:00
|
|
|
counter++;
|
2002-07-16 18:58:08 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
StructType *sttype = StructType::get(vType);
|
|
|
|
ConstantStruct *cstruct = ConstantStruct::get(sttype, vConsts);
|
|
|
|
|
2003-04-16 20:28:45 +00:00
|
|
|
GlobalVariable *gb = new GlobalVariable(cstruct->getType(), true,
|
|
|
|
GlobalValue::ExternalLinkage,
|
2002-07-16 18:58:08 +00:00
|
|
|
cstruct, "llvmFunctionTable");
|
|
|
|
M.getGlobalList().push_back(gb);
|
2003-06-01 02:40:49 +00:00
|
|
|
|
2003-07-18 20:55:26 +00:00
|
|
|
ConstantArray *constArray = ConstantArray::get(ArrayType::get(Type::SByteTy,
|
|
|
|
sBCons.size()),
|
|
|
|
sBCons);
|
|
|
|
|
|
|
|
GlobalVariable *funcArray = new GlobalVariable(constArray->getType(), true,
|
|
|
|
GlobalValue::ExternalLinkage,
|
|
|
|
constArray, "llvmSimpleFunction");
|
|
|
|
|
|
|
|
M.getGlobalList().push_back(funcArray);
|
|
|
|
|
2003-06-04 20:08:47 +00:00
|
|
|
ConstantInt *cnst = ConstantSInt::get(Type::IntTy, counter);
|
2003-06-01 02:40:49 +00:00
|
|
|
GlobalVariable *fnCount = new GlobalVariable(Type::IntTy, true,
|
|
|
|
GlobalValue::ExternalLinkage,
|
|
|
|
cnst, "llvmFunctionCount");
|
|
|
|
M.getGlobalList().push_back(fnCount);
|
2002-07-16 18:58:08 +00:00
|
|
|
return true; // Always modifies program
|
|
|
|
}
|