1 //===- PathProfileInfo.cpp ------------------------------------*- C++ -*---===//
2 //
3 // The LLVM Compiler Infrastructure
4 //
5 // This file is distributed under the University of Illinois Open Source
6 // License. See LICENSE.TXT for details.
7 //
8 //===----------------------------------------------------------------------===//
9 //
10 // This file defines the interface used by optimizers to load path profiles,
11 // and provides a loader pass which reads a path profile file.
12 //
13 //===----------------------------------------------------------------------===//
14 #define DEBUG_TYPE "path-profile-info"
15
16 #include "llvm/Module.h"
17 #include "llvm/Pass.h"
18 #include "llvm/Analysis/Passes.h"
19 #include "llvm/Analysis/ProfileInfoTypes.h"
20 #include "llvm/Analysis/PathProfileInfo.h"
21 #include "llvm/Support/CommandLine.h"
22 #include "llvm/Support/Debug.h"
23 #include "llvm/Support/raw_ostream.h"
24
25 #include <cstdio>
26
27 using namespace llvm;
28
29 // command line option for loading path profiles
30 static cl::opt<std::string>
31 PathProfileInfoFilename("path-profile-loader-file", cl::init("llvmprof.out"),
32 cl::value_desc("filename"),
33 cl::desc("Path profile file loaded by -path-profile-loader"), cl::Hidden);
34
35 namespace {
36 class PathProfileLoaderPass : public ModulePass, public PathProfileInfo {
37 public:
PathProfileLoaderPass()38 PathProfileLoaderPass() : ModulePass(ID) { }
39 ~PathProfileLoaderPass();
40
41 // this pass doesn't change anything (only loads information)
getAnalysisUsage(AnalysisUsage & AU) const42 virtual void getAnalysisUsage(AnalysisUsage &AU) const {
43 AU.setPreservesAll();
44 }
45
46 // the full name of the loader pass
getPassName() const47 virtual const char* getPassName() const {
48 return "Path Profiling Information Loader";
49 }
50
51 // required since this pass implements multiple inheritance
getAdjustedAnalysisPointer(AnalysisID PI)52 virtual void *getAdjustedAnalysisPointer(AnalysisID PI) {
53 if (PI == &PathProfileInfo::ID)
54 return (PathProfileInfo*)this;
55 return this;
56 }
57
58 // entry point to run the pass
59 bool runOnModule(Module &M);
60
61 // pass identification
62 static char ID;
63
64 private:
65 // make a reference table to refer to function by number
66 void buildFunctionRefs(Module &M);
67
68 // process argument info of a program from the input file
69 void handleArgumentInfo();
70
71 // process path number information from the input file
72 void handlePathInfo();
73
74 // array of references to the functions in the module
75 std::vector<Function*> _functions;
76
77 // path profile file handle
78 FILE* _file;
79
80 // path profile file name
81 std::string _filename;
82 };
83 }
84
85 // register PathLoader
86 char PathProfileLoaderPass::ID = 0;
87
88 INITIALIZE_ANALYSIS_GROUP(PathProfileInfo, "Path Profile Information",
89 NoPathProfileInfo)
90 INITIALIZE_AG_PASS(PathProfileLoaderPass, PathProfileInfo,
91 "path-profile-loader",
92 "Load path profile information from file",
93 false, true, false)
94
95 char &llvm::PathProfileLoaderPassID = PathProfileLoaderPass::ID;
96
97 // link PathLoader as a pass, and make it available as an optimisation
createPathProfileLoaderPass()98 ModulePass *llvm::createPathProfileLoaderPass() {
99 return new PathProfileLoaderPass;
100 }
101
102 // ----------------------------------------------------------------------------
103 // PathEdge implementation
104 //
ProfilePathEdge(BasicBlock * source,BasicBlock * target,unsigned duplicateNumber)105 ProfilePathEdge::ProfilePathEdge (BasicBlock* source, BasicBlock* target,
106 unsigned duplicateNumber)
107 : _source(source), _target(target), _duplicateNumber(duplicateNumber) {}
108
109 // ----------------------------------------------------------------------------
110 // Path implementation
111 //
112
ProfilePath(unsigned int number,unsigned int count,double countStdDev,PathProfileInfo * ppi)113 ProfilePath::ProfilePath (unsigned int number, unsigned int count,
114 double countStdDev, PathProfileInfo* ppi)
115 : _number(number) , _count(count), _countStdDev(countStdDev), _ppi(ppi) {}
116
getFrequency() const117 double ProfilePath::getFrequency() const {
118 return 100 * double(_count) /
119 double(_ppi->_functionPathCounts[_ppi->_currentFunction]);
120 }
121
getNextEdge(BallLarusNode * node,unsigned int pathNumber)122 static BallLarusEdge* getNextEdge (BallLarusNode* node,
123 unsigned int pathNumber) {
124 BallLarusEdge* best = 0;
125
126 for( BLEdgeIterator next = node->succBegin(),
127 end = node->succEnd(); next != end; next++ ) {
128 if( (*next)->getType() != BallLarusEdge::BACKEDGE && // no backedges
129 (*next)->getType() != BallLarusEdge::SPLITEDGE && // no split edges
130 (*next)->getWeight() <= pathNumber && // weight must be <= pathNumber
131 (!best || (best->getWeight() < (*next)->getWeight())) ) // best one?
132 best = *next;
133 }
134
135 return best;
136 }
137
getPathEdges() const138 ProfilePathEdgeVector* ProfilePath::getPathEdges() const {
139 BallLarusNode* currentNode = _ppi->_currentDag->getRoot ();
140 unsigned int increment = _number;
141 ProfilePathEdgeVector* pev = new ProfilePathEdgeVector;
142
143 while (currentNode != _ppi->_currentDag->getExit()) {
144 BallLarusEdge* next = getNextEdge(currentNode, increment);
145
146 increment -= next->getWeight();
147
148 if( next->getType() != BallLarusEdge::BACKEDGE_PHONY &&
149 next->getType() != BallLarusEdge::SPLITEDGE_PHONY &&
150 next->getTarget() != _ppi->_currentDag->getExit() )
151 pev->push_back(ProfilePathEdge(
152 next->getSource()->getBlock(),
153 next->getTarget()->getBlock(),
154 next->getDuplicateNumber()));
155
156 if( next->getType() == BallLarusEdge::BACKEDGE_PHONY &&
157 next->getTarget() == _ppi->_currentDag->getExit() )
158 pev->push_back(ProfilePathEdge(
159 next->getRealEdge()->getSource()->getBlock(),
160 next->getRealEdge()->getTarget()->getBlock(),
161 next->getDuplicateNumber()));
162
163 if( next->getType() == BallLarusEdge::SPLITEDGE_PHONY &&
164 next->getSource() == _ppi->_currentDag->getRoot() )
165 pev->push_back(ProfilePathEdge(
166 next->getRealEdge()->getSource()->getBlock(),
167 next->getRealEdge()->getTarget()->getBlock(),
168 next->getDuplicateNumber()));
169
170 // set the new node
171 currentNode = next->getTarget();
172 }
173
174 return pev;
175 }
176
getPathBlocks() const177 ProfilePathBlockVector* ProfilePath::getPathBlocks() const {
178 BallLarusNode* currentNode = _ppi->_currentDag->getRoot ();
179 unsigned int increment = _number;
180 ProfilePathBlockVector* pbv = new ProfilePathBlockVector;
181
182 while (currentNode != _ppi->_currentDag->getExit()) {
183 BallLarusEdge* next = getNextEdge(currentNode, increment);
184 increment -= next->getWeight();
185
186 // add block to the block list if it is a real edge
187 if( next->getType() == BallLarusEdge::NORMAL)
188 pbv->push_back (currentNode->getBlock());
189 // make the back edge the last edge since we are at the end
190 else if( next->getTarget() == _ppi->_currentDag->getExit() ) {
191 pbv->push_back (currentNode->getBlock());
192 pbv->push_back (next->getRealEdge()->getTarget()->getBlock());
193 }
194
195 // set the new node
196 currentNode = next->getTarget();
197 }
198
199 return pbv;
200 }
201
getFirstBlockInPath() const202 BasicBlock* ProfilePath::getFirstBlockInPath() const {
203 BallLarusNode* root = _ppi->_currentDag->getRoot();
204 BallLarusEdge* edge = getNextEdge(root, _number);
205
206 if( edge && (edge->getType() == BallLarusEdge::BACKEDGE_PHONY ||
207 edge->getType() == BallLarusEdge::SPLITEDGE_PHONY) )
208 return edge->getTarget()->getBlock();
209
210 return root->getBlock();
211 }
212
213 // ----------------------------------------------------------------------------
214 // PathProfileInfo implementation
215 //
216
217 // Pass identification
218 char llvm::PathProfileInfo::ID = 0;
219
PathProfileInfo()220 PathProfileInfo::PathProfileInfo () : _currentDag(0) , _currentFunction(0) {
221 }
222
~PathProfileInfo()223 PathProfileInfo::~PathProfileInfo() {
224 if (_currentDag)
225 delete _currentDag;
226 }
227
228 // set the function for which paths are currently begin processed
setCurrentFunction(Function * F)229 void PathProfileInfo::setCurrentFunction(Function* F) {
230 // Make sure it exists
231 if (!F) return;
232
233 if (_currentDag)
234 delete _currentDag;
235
236 _currentFunction = F;
237 _currentDag = new BallLarusDag(*F);
238 _currentDag->init();
239 _currentDag->calculatePathNumbers();
240 }
241
242 // get the function for which paths are currently being processed
getCurrentFunction() const243 Function* PathProfileInfo::getCurrentFunction() const {
244 return _currentFunction;
245 }
246
247 // get the entry block of the function
getCurrentFunctionEntry()248 BasicBlock* PathProfileInfo::getCurrentFunctionEntry() {
249 return _currentDag->getRoot()->getBlock();
250 }
251
252 // return the path based on its number
getPath(unsigned int number)253 ProfilePath* PathProfileInfo::getPath(unsigned int number) {
254 return _functionPaths[_currentFunction][number];
255 }
256
257 // return the number of paths which a function may potentially execute
getPotentialPathCount()258 unsigned int PathProfileInfo::getPotentialPathCount() {
259 return _currentDag ? _currentDag->getNumberOfPaths() : 0;
260 }
261
262 // return an iterator for the beginning of a functions executed paths
pathBegin()263 ProfilePathIterator PathProfileInfo::pathBegin() {
264 return _functionPaths[_currentFunction].begin();
265 }
266
267 // return an iterator for the end of a functions executed paths
pathEnd()268 ProfilePathIterator PathProfileInfo::pathEnd() {
269 return _functionPaths[_currentFunction].end();
270 }
271
272 // returns the total number of paths run in the function
pathsRun()273 unsigned int PathProfileInfo::pathsRun() {
274 return _currentFunction ? _functionPaths[_currentFunction].size() : 0;
275 }
276
277 // ----------------------------------------------------------------------------
278 // PathLoader implementation
279 //
280
281 // remove all generated paths
~PathProfileLoaderPass()282 PathProfileLoaderPass::~PathProfileLoaderPass() {
283 for( FunctionPathIterator funcNext = _functionPaths.begin(),
284 funcEnd = _functionPaths.end(); funcNext != funcEnd; funcNext++)
285 for( ProfilePathIterator pathNext = funcNext->second.begin(),
286 pathEnd = funcNext->second.end(); pathNext != pathEnd; pathNext++)
287 delete pathNext->second;
288 }
289
290 // entry point of the pass; this loads and parses a file
runOnModule(Module & M)291 bool PathProfileLoaderPass::runOnModule(Module &M) {
292 // get the filename and setup the module's function references
293 _filename = PathProfileInfoFilename;
294 buildFunctionRefs (M);
295
296 if (!(_file = fopen(_filename.c_str(), "rb"))) {
297 errs () << "error: input '" << _filename << "' file does not exist.\n";
298 return false;
299 }
300
301 ProfilingType profType;
302
303 while( fread(&profType, sizeof(ProfilingType), 1, _file) ) {
304 switch (profType) {
305 case ArgumentInfo:
306 handleArgumentInfo ();
307 break;
308 case PathInfo:
309 handlePathInfo ();
310 break;
311 default:
312 errs () << "error: bad path profiling file syntax, " << profType << "\n";
313 fclose (_file);
314 return false;
315 }
316 }
317
318 fclose (_file);
319
320 return true;
321 }
322
323 // create a reference table for functions defined in the path profile file
buildFunctionRefs(Module & M)324 void PathProfileLoaderPass::buildFunctionRefs (Module &M) {
325 _functions.push_back(0); // make the 0 index a null pointer
326
327 for (Module::iterator F = M.begin(), E = M.end(); F != E; F++) {
328 if (F->isDeclaration())
329 continue;
330 _functions.push_back(F);
331 }
332 }
333
334 // handle command like argument infor in the output file
handleArgumentInfo()335 void PathProfileLoaderPass::handleArgumentInfo() {
336 // get the argument list's length
337 unsigned savedArgsLength;
338 if( fread(&savedArgsLength, sizeof(unsigned), 1, _file) != 1 ) {
339 errs() << "warning: argument info header/data mismatch\n";
340 return;
341 }
342
343 // allocate a buffer, and get the arguments
344 char* args = new char[savedArgsLength+1];
345 if( fread(args, 1, savedArgsLength, _file) != savedArgsLength )
346 errs() << "warning: argument info header/data mismatch\n";
347
348 args[savedArgsLength] = '\0';
349 argList = std::string(args);
350 delete [] args; // cleanup dynamic string
351
352 // byte alignment
353 if (savedArgsLength & 3)
354 fseek(_file, 4-(savedArgsLength&3), SEEK_CUR);
355 }
356
357 // Handle path profile information in the output file
handlePathInfo()358 void PathProfileLoaderPass::handlePathInfo () {
359 // get the number of functions in this profile
360 unsigned functionCount;
361 if( fread(&functionCount, sizeof(functionCount), 1, _file) != 1 ) {
362 errs() << "warning: path info header/data mismatch\n";
363 return;
364 }
365
366 // gather path information for each function
367 for (unsigned i = 0; i < functionCount; i++) {
368 PathProfileHeader pathHeader;
369 if( fread(&pathHeader, sizeof(pathHeader), 1, _file) != 1 ) {
370 errs() << "warning: bad header for path function info\n";
371 break;
372 }
373
374 Function* f = _functions[pathHeader.fnNumber];
375
376 // dynamically allocate a table to store path numbers
377 PathProfileTableEntry* pathTable =
378 new PathProfileTableEntry[pathHeader.numEntries];
379
380 if( fread(pathTable, sizeof(PathProfileTableEntry),
381 pathHeader.numEntries, _file) != pathHeader.numEntries) {
382 delete [] pathTable;
383 errs() << "warning: path function info header/data mismatch\n";
384 return;
385 }
386
387 // Build a new path for the current function
388 unsigned int totalPaths = 0;
389 for (unsigned int j = 0; j < pathHeader.numEntries; j++) {
390 totalPaths += pathTable[j].pathCounter;
391 _functionPaths[f][pathTable[j].pathNumber]
392 = new ProfilePath(pathTable[j].pathNumber, pathTable[j].pathCounter,
393 0, this);
394 }
395
396 _functionPathCounts[f] = totalPaths;
397
398 delete [] pathTable;
399 }
400 }
401
402 //===----------------------------------------------------------------------===//
403 // NoProfile PathProfileInfo implementation
404 //
405
406 namespace {
407 struct NoPathProfileInfo : public ImmutablePass, public PathProfileInfo {
408 static char ID; // Class identification, replacement for typeinfo
NoPathProfileInfo__anon19d37f130211::NoPathProfileInfo409 NoPathProfileInfo() : ImmutablePass(ID) {
410 initializeNoPathProfileInfoPass(*PassRegistry::getPassRegistry());
411 }
412
413 /// getAdjustedAnalysisPointer - This method is used when a pass implements
414 /// an analysis interface through multiple inheritance. If needed, it
415 /// should override this to adjust the this pointer as needed for the
416 /// specified pass info.
getAdjustedAnalysisPointer__anon19d37f130211::NoPathProfileInfo417 virtual void *getAdjustedAnalysisPointer(AnalysisID PI) {
418 if (PI == &PathProfileInfo::ID)
419 return (PathProfileInfo*)this;
420 return this;
421 }
422
getPassName__anon19d37f130211::NoPathProfileInfo423 virtual const char *getPassName() const {
424 return "NoPathProfileInfo";
425 }
426 };
427 } // End of anonymous namespace
428
429 char NoPathProfileInfo::ID = 0;
430 // Register this pass...
431 INITIALIZE_AG_PASS(NoPathProfileInfo, PathProfileInfo, "no-path-profile",
432 "No Path Profile Information", false, true, true)
433
createNoPathProfileInfoPass()434 ImmutablePass *llvm::createNoPathProfileInfoPass() { return new NoPathProfileInfo(); }
435