2017-08-26 05:35:27 +08:00
|
|
|
//===- PruneUnprofitable.cpp ----------------------------------------------===//
|
2017-03-17 21:09:52 +08:00
|
|
|
//
|
2019-01-19 16:50:56 +08:00
|
|
|
// Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions.
|
|
|
|
// See https://llvm.org/LICENSE.txt for license information.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception
|
2017-03-17 21:09:52 +08:00
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
//
|
|
|
|
// Mark a SCoP as unfeasible if not deemed profitable to optimize.
|
|
|
|
//
|
|
|
|
//===----------------------------------------------------------------------===//
|
|
|
|
|
|
|
|
#include "polly/PruneUnprofitable.h"
|
2017-08-26 05:35:27 +08:00
|
|
|
#include "polly/ScopDetection.h"
|
2017-03-17 21:09:52 +08:00
|
|
|
#include "polly/ScopInfo.h"
|
|
|
|
#include "polly/ScopPass.h"
|
2017-08-26 05:35:27 +08:00
|
|
|
#include "llvm/ADT/Statistic.h"
|
|
|
|
#include "llvm/IR/DebugLoc.h"
|
|
|
|
#include "llvm/Support/Debug.h"
|
|
|
|
#include "llvm/Support/raw_ostream.h"
|
2017-03-17 21:09:52 +08:00
|
|
|
|
|
|
|
using namespace llvm;
|
2017-08-26 05:35:27 +08:00
|
|
|
using namespace polly;
|
|
|
|
|
|
|
|
#define DEBUG_TYPE "polly-prune-unprofitable"
|
2017-03-17 21:09:52 +08:00
|
|
|
|
|
|
|
namespace {
|
|
|
|
|
|
|
|
STATISTIC(ScopsProcessed,
|
|
|
|
"Number of SCoPs considered for unprofitability pruning");
|
|
|
|
STATISTIC(ScopsPruned, "Number of pruned SCoPs because it they cannot be "
|
|
|
|
"optimized in a significant way");
|
2017-08-23 21:50:30 +08:00
|
|
|
STATISTIC(ScopsSurvived, "Number of SCoPs after pruning");
|
|
|
|
|
|
|
|
STATISTIC(NumPrunedLoops, "Number of pruned loops");
|
|
|
|
STATISTIC(NumPrunedBoxedLoops, "Number of pruned boxed loops");
|
|
|
|
STATISTIC(NumPrunedAffineLoops, "Number of pruned affine loops");
|
|
|
|
|
|
|
|
STATISTIC(NumLoopsInScop, "Number of loops in scops after pruning");
|
|
|
|
STATISTIC(NumBoxedLoops, "Number of boxed loops in SCoPs after pruning");
|
|
|
|
STATISTIC(NumAffineLoops, "Number of affine loops in SCoPs after pruning");
|
2017-03-17 21:09:52 +08:00
|
|
|
|
2021-02-10 11:53:14 +08:00
|
|
|
static void updateStatistics(Scop &S, bool Pruned) {
|
|
|
|
Scop::ScopStatistics ScopStats = S.getStatistics();
|
|
|
|
if (Pruned) {
|
|
|
|
ScopsPruned++;
|
|
|
|
NumPrunedLoops += ScopStats.NumAffineLoops + ScopStats.NumBoxedLoops;
|
|
|
|
NumPrunedBoxedLoops += ScopStats.NumBoxedLoops;
|
|
|
|
NumPrunedAffineLoops += ScopStats.NumAffineLoops;
|
|
|
|
} else {
|
|
|
|
ScopsSurvived++;
|
|
|
|
NumLoopsInScop += ScopStats.NumAffineLoops + ScopStats.NumBoxedLoops;
|
|
|
|
NumBoxedLoops += ScopStats.NumBoxedLoops;
|
|
|
|
NumAffineLoops += ScopStats.NumAffineLoops;
|
2017-08-23 21:50:30 +08:00
|
|
|
}
|
2021-02-10 11:53:14 +08:00
|
|
|
}
|
2017-08-23 21:50:30 +08:00
|
|
|
|
2021-02-10 11:53:14 +08:00
|
|
|
static bool runPruneUnprofitable(Scop &S) {
|
|
|
|
if (PollyProcessUnprofitable) {
|
|
|
|
LLVM_DEBUG(
|
|
|
|
dbgs() << "NOTE: -polly-process-unprofitable active, won't prune "
|
|
|
|
"anything\n");
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
ScopsProcessed++;
|
|
|
|
|
|
|
|
if (!S.isProfitable(true)) {
|
|
|
|
LLVM_DEBUG(
|
|
|
|
dbgs() << "SCoP pruned because it probably cannot be optimized in "
|
|
|
|
"a significant way\n");
|
|
|
|
S.invalidate(PROFITABLE, DebugLoc());
|
|
|
|
updateStatistics(S, true);
|
|
|
|
} else {
|
|
|
|
updateStatistics(S, false);
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
class PruneUnprofitableWrapperPass : public ScopPass {
|
2017-03-17 21:09:52 +08:00
|
|
|
public:
|
|
|
|
static char ID;
|
2017-08-26 05:35:27 +08:00
|
|
|
|
2021-02-10 11:53:14 +08:00
|
|
|
explicit PruneUnprofitableWrapperPass() : ScopPass(ID) {}
|
|
|
|
PruneUnprofitableWrapperPass(const PruneUnprofitableWrapperPass &) = delete;
|
|
|
|
PruneUnprofitableWrapperPass &
|
|
|
|
operator=(const PruneUnprofitableWrapperPass &) = delete;
|
2017-03-17 21:09:52 +08:00
|
|
|
|
2017-08-26 05:35:27 +08:00
|
|
|
void getAnalysisUsage(AnalysisUsage &AU) const override {
|
2017-03-17 21:09:52 +08:00
|
|
|
AU.addRequired<ScopInfoRegionPass>();
|
|
|
|
AU.setPreservesAll();
|
|
|
|
}
|
|
|
|
|
2021-02-10 11:53:14 +08:00
|
|
|
bool runOnScop(Scop &S) override { return runPruneUnprofitable(S); }
|
2017-03-17 21:09:52 +08:00
|
|
|
};
|
2017-08-26 05:35:27 +08:00
|
|
|
} // namespace
|
|
|
|
|
2021-02-10 11:53:14 +08:00
|
|
|
char PruneUnprofitableWrapperPass::ID;
|
2017-03-17 21:09:52 +08:00
|
|
|
|
2021-02-10 11:53:14 +08:00
|
|
|
Pass *polly::createPruneUnprofitableWrapperPass() {
|
|
|
|
return new PruneUnprofitableWrapperPass();
|
|
|
|
}
|
2017-03-17 21:09:52 +08:00
|
|
|
|
2021-02-10 11:53:14 +08:00
|
|
|
INITIALIZE_PASS_BEGIN(PruneUnprofitableWrapperPass, "polly-prune-unprofitable",
|
2017-03-17 21:09:52 +08:00
|
|
|
"Polly - Prune unprofitable SCoPs", false, false)
|
2021-02-10 11:53:14 +08:00
|
|
|
INITIALIZE_PASS_END(PruneUnprofitableWrapperPass, "polly-prune-unprofitable",
|
2017-03-17 21:09:52 +08:00
|
|
|
"Polly - Prune unprofitable SCoPs", false, false)
|
2021-02-10 11:53:14 +08:00
|
|
|
|
|
|
|
llvm::PreservedAnalyses
|
|
|
|
PruneUnprofitablePass::run(Scop &S, ScopAnalysisManager &SAM,
|
|
|
|
ScopStandardAnalysisResults &SAR, SPMUpdater &U) {
|
|
|
|
bool Changed = runPruneUnprofitable(S);
|
|
|
|
|
|
|
|
if (!Changed)
|
|
|
|
return PreservedAnalyses::all();
|
|
|
|
|
|
|
|
PreservedAnalyses PA;
|
|
|
|
PA.preserveSet<AllAnalysesOn<Module>>();
|
|
|
|
PA.preserveSet<AllAnalysesOn<Function>>();
|
|
|
|
PA.preserveSet<AllAnalysesOn<Loop>>();
|
|
|
|
return PA;
|
|
|
|
}
|