//===- ProfileInfoLoaderPass.cpp - LLVM Pass to load profile info ---------===// // // The LLVM Compiler Infrastructure // // This file was developed by the LLVM research group and is distributed under // the University of Illinois Open Source License. See LICENSE.TXT for details. // //===----------------------------------------------------------------------===// // // This file implements a concrete implementation of profiling information that // loads the information from a profile dump file. // //===----------------------------------------------------------------------===// #include "llvm/BasicBlock.h" #include "llvm/InstrTypes.h" #include "llvm/Pass.h" #include "llvm/Analysis/ProfileInfo.h" #include "llvm/Analysis/ProfileInfoLoader.h" #include "Support/CommandLine.h" #include using namespace llvm; namespace { cl::opt ProfileInfoFilename("profile-info-file", cl::init("llvmprof.out"), cl::value_desc("filename"), cl::desc("Profile file loaded by -profile-loader")); class LoaderPass : public Pass, public ProfileInfo { std::string Filename; public: LoaderPass(const std::string &filename = "") : Filename(filename) { if (filename.empty()) Filename = ProfileInfoFilename; } virtual void getAnalysisUsage(AnalysisUsage &AU) const { AU.setPreservesAll(); } virtual const char *getPassName() const { return "Profiling information loader"; } /// run - Load the profile information from the specified file. virtual bool run(Module &M); }; RegisterOpt X("profile-loader", "Load profile information from llvmprof.out"); RegisterAnalysisGroup Y; } // End of anonymous namespace /// createProfileLoaderPass - This function returns a Pass that loads the /// profiling information for the module from the specified filename, making it /// available to the optimizers. Pass *llvm::createProfileLoaderPass(const std::string &Filename) { return new LoaderPass(Filename); } bool LoaderPass::run(Module &M) { ProfileInfoLoader PIL("profile-loader", Filename, M); EdgeCounts.clear(); bool PrintedWarning = false; std::vector > ECs; PIL.getEdgeCounts(ECs); for (unsigned i = 0, e = ECs.size(); i != e; ++i) { BasicBlock *BB = ECs[i].first.first; unsigned SuccNum = ECs[i].first.second; TerminatorInst *TI = BB->getTerminator(); if (SuccNum >= TI->getNumSuccessors()) { if (!PrintedWarning) { std::cerr << "WARNING: profile information is inconsistent with " << "the current program!\n"; PrintedWarning = true; } } else { EdgeCounts[std::make_pair(BB, TI->getSuccessor(SuccNum))]+= ECs[i].second; } } return false; }