2018-08-29 05:06:51 +08:00
|
|
|
//===- PassTimingInfo.cpp - LLVM Pass Timing Implementation ---------------===//
|
|
|
|
//
|
|
|
|
// The LLVM Compiler Infrastructure
|
|
|
|
//
|
|
|
|
// This file is distributed under the University of Illinois Open Source
|
|
|
|
// License. See LICENSE.TXT for details.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// This file implements the LLVM Pass Timing infrastructure for both
|
|
|
|
// new and legacy pass managers.
|
|
|
|
//
|
|
|
|
// TimingInfo Class - This class is used to calculate information about the
|
|
|
|
// amount of time each pass takes to execute. This only happens when
|
|
|
|
// -time-passes is enabled on the command line.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "llvm/IR/PassTimingInfo.h"
|
|
|
|
#include "llvm/ADT/DenseMap.h"
|
|
|
|
#include "llvm/ADT/Statistic.h"
|
|
|
|
#include "llvm/ADT/StringRef.h"
|
|
|
|
#include "llvm/Pass.h"
|
|
|
|
#include "llvm/Support/CommandLine.h"
|
|
|
|
#include "llvm/Support/Debug.h"
|
2018-09-04 14:12:28 +08:00
|
|
|
#include "llvm/Support/FormatVariadic.h"
|
2018-08-29 05:06:51 +08:00
|
|
|
#include "llvm/Support/ManagedStatic.h"
|
|
|
|
#include "llvm/Support/Mutex.h"
|
|
|
|
#include "llvm/Support/Timer.h"
|
|
|
|
#include "llvm/Support/raw_ostream.h"
|
|
|
|
#include <string>
|
|
|
|
|
|
|
|
using namespace llvm;
|
|
|
|
|
|
|
|
namespace llvm {
|
|
|
|
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
// TimingInfo implementation
|
|
|
|
|
|
|
|
bool TimePassesIsEnabled = false;
|
|
|
|
static cl::opt<bool, true> EnableTiming(
|
|
|
|
"time-passes", cl::location(TimePassesIsEnabled), cl::Hidden,
|
|
|
|
cl::desc("Time each pass, printing elapsed time for each on exit"));
|
|
|
|
|
|
|
|
namespace {
|
|
|
|
static ManagedStatic<sys::SmartMutex<true>> TimingInfoMutex;
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename PassT>
|
|
|
|
PassTimingInfo<PassT>::PassTimingInfo()
|
|
|
|
: TG("pass", "... Pass execution timing report ...") {}
|
|
|
|
|
|
|
|
template <typename PassT> PassTimingInfo<PassT>::~PassTimingInfo() {
|
|
|
|
// Deleting the timers accumulates their info into the TG member.
|
|
|
|
// Then TG member is (implicitly) deleted, actually printing the report.
|
|
|
|
for (auto &I : TimingData)
|
2018-09-04 14:12:28 +08:00
|
|
|
delete I.getSecond();
|
2018-08-29 05:06:51 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
template <typename PassT> void PassTimingInfo<PassT>::init() {
|
|
|
|
if (!TimePassesIsEnabled || TheTimeInfo)
|
|
|
|
return;
|
|
|
|
|
|
|
|
// Constructed the first time this is called, iff -time-passes is enabled.
|
2018-09-04 14:12:28 +08:00
|
|
|
// This guarantees that the object will be constructed after static globals,
|
2018-08-29 05:06:51 +08:00
|
|
|
// thus it will be destroyed before them.
|
|
|
|
static ManagedStatic<PassTimingInfo> TTI;
|
|
|
|
TheTimeInfo = &*TTI;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Prints out timing information and then resets the timers.
|
|
|
|
template <typename PassT> void PassTimingInfo<PassT>::print() {
|
|
|
|
TG.print(*CreateInfoOutputFile());
|
|
|
|
}
|
|
|
|
|
2018-09-04 14:12:28 +08:00
|
|
|
template <typename PassInfoT>
|
|
|
|
Timer *PassTimingInfo<PassInfoT>::newPassTimer(StringRef PassID,
|
|
|
|
StringRef PassDesc) {
|
|
|
|
unsigned &num = PassIDCountMap[PassID];
|
|
|
|
num++;
|
|
|
|
// Appending description with a pass-instance number for all but the first one
|
|
|
|
std::string PassDescNumbered =
|
|
|
|
num <= 1 ? PassDesc.str() : formatv("{0} #{1}", PassDesc, num).str();
|
|
|
|
return new Timer(PassID, PassDescNumbered, TG);
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Returns the timer for the specified pass instance \p Pass.
|
|
|
|
/// Instances of the same pass type (uniquely identified by \p PassID) are
|
|
|
|
/// numbered by the order of appearance.
|
|
|
|
template <>
|
|
|
|
Timer *PassTimingInfo<StringRef>::getPassTimer(StringRef PassID,
|
|
|
|
PassInstanceID Pass) {
|
2018-08-29 05:06:51 +08:00
|
|
|
init();
|
|
|
|
sys::SmartScopedLock<true> Lock(*TimingInfoMutex);
|
2018-09-04 14:12:28 +08:00
|
|
|
Timer *&T = TimingData[Pass];
|
2018-08-29 05:06:51 +08:00
|
|
|
if (!T)
|
2018-09-04 14:12:28 +08:00
|
|
|
T = newPassTimer(PassID, PassID);
|
2018-08-29 05:06:51 +08:00
|
|
|
return T;
|
|
|
|
}
|
|
|
|
|
2018-09-04 14:12:28 +08:00
|
|
|
template <>
|
|
|
|
Timer *PassTimingInfo<Pass *>::getPassTimer(Pass *P, PassInstanceID Pass) {
|
2018-08-29 05:06:51 +08:00
|
|
|
if (P->getAsPMDataManager())
|
|
|
|
return nullptr;
|
|
|
|
|
|
|
|
init();
|
|
|
|
sys::SmartScopedLock<true> Lock(*TimingInfoMutex);
|
2018-09-04 14:12:28 +08:00
|
|
|
Timer *&T = TimingData[Pass];
|
2018-08-29 05:06:51 +08:00
|
|
|
|
|
|
|
if (!T) {
|
2018-09-04 14:12:28 +08:00
|
|
|
StringRef PassName = P->getPassName();
|
2018-08-29 05:06:51 +08:00
|
|
|
StringRef PassArgument;
|
|
|
|
if (const PassInfo *PI = Pass::lookupPassInfo(P->getPassID()))
|
|
|
|
PassArgument = PI->getPassArgument();
|
2018-09-04 14:12:28 +08:00
|
|
|
T = newPassTimer(PassArgument.empty() ? PassName : PassArgument, PassName);
|
2018-08-29 05:06:51 +08:00
|
|
|
}
|
|
|
|
return T;
|
|
|
|
}
|
|
|
|
|
|
|
|
template <typename PassInfoT>
|
|
|
|
PassTimingInfo<PassInfoT> *PassTimingInfo<PassInfoT>::TheTimeInfo;
|
|
|
|
|
|
|
|
template class PassTimingInfo<Pass *>;
|
|
|
|
template class PassTimingInfo<StringRef>;
|
|
|
|
|
|
|
|
Timer *getPassTimer(Pass *P) {
|
|
|
|
PassTimingInfo<Pass *>::init();
|
|
|
|
if (PassTimingInfo<Pass *>::TheTimeInfo)
|
2018-09-04 14:12:28 +08:00
|
|
|
return PassTimingInfo<Pass *>::TheTimeInfo->getPassTimer(P, P);
|
2018-08-29 05:06:51 +08:00
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
Timer *getPassTimer(StringRef PassName) {
|
|
|
|
PassTimingInfo<StringRef>::init();
|
|
|
|
if (PassTimingInfo<StringRef>::TheTimeInfo)
|
2018-09-04 14:12:28 +08:00
|
|
|
return PassTimingInfo<StringRef>::TheTimeInfo->getPassTimer(PassName,
|
|
|
|
nullptr);
|
2018-08-29 05:06:51 +08:00
|
|
|
return nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
/// If timing is enabled, report the times collected up to now and then reset
|
|
|
|
/// them.
|
|
|
|
void reportAndResetTimings() {
|
|
|
|
if (PassTimingInfo<StringRef>::TheTimeInfo)
|
|
|
|
PassTimingInfo<StringRef>::TheTimeInfo->print();
|
|
|
|
if (PassTimingInfo<Pass *>::TheTimeInfo)
|
|
|
|
PassTimingInfo<Pass *>::TheTimeInfo->print();
|
|
|
|
}
|
|
|
|
|
|
|
|
} // namespace llvm
|