//===-- 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" #include "llvm/Pass.h" #include "llvm/Support/CFG.h" enum Color{ WHITE, GREY, BLACK }; namespace { struct EmitFunctionTable : public Pass { bool run(Module &M); }; RegisterOpt X("emitfuncs", "Emit a Function Table"); } char doDFS(BasicBlock * node,std::map &color){ color[node] = GREY; for(succ_iterator vl = succ_begin(node), ve = succ_end(node); vl != ve; ++vl){ 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 color; return doDFS(F->begin(), color); } // Per Module pass for inserting function table bool EmitFunctionTable::run(Module &M){ std::vector vType; std::vector vConsts; std::vector sBCons; unsigned int counter = 0; for(Module::iterator MI = M.begin(), ME = M.end(); MI != ME; ++MI) if (!MI->isExternal()) { vType.push_back(MI->getType()); //std::cerr<getType(), true, GlobalValue::ExternalLinkage, cstruct, "llvmFunctionTable"); M.getGlobalList().push_back(gb); 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); ConstantInt *cnst = ConstantSInt::get(Type::IntTy, counter); GlobalVariable *fnCount = new GlobalVariable(Type::IntTy, true, GlobalValue::ExternalLinkage, cnst, "llvmFunctionCount"); M.getGlobalList().push_back(fnCount); return true; // Always modifies program }