123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150 |
- //===- LoopTransformWarning.cpp - ----------------------------------------===//
- //
- // 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
- //
- //===----------------------------------------------------------------------===//
- //
- // Emit warnings if forced code transformations have not been performed.
- //
- //===----------------------------------------------------------------------===//
- #include "llvm/Transforms/Scalar/WarnMissedTransforms.h"
- #include "llvm/Analysis/LoopInfo.h"
- #include "llvm/Analysis/OptimizationRemarkEmitter.h"
- #include "llvm/InitializePasses.h"
- #include "llvm/Transforms/Utils/LoopUtils.h"
- using namespace llvm;
- #define DEBUG_TYPE "transform-warning"
- /// Emit warnings for forced (i.e. user-defined) loop transformations which have
- /// still not been performed.
- static void warnAboutLeftoverTransformations(Loop *L,
- OptimizationRemarkEmitter *ORE) {
- if (hasUnrollTransformation(L) == TM_ForcedByUser) {
- LLVM_DEBUG(dbgs() << "Leftover unroll transformation\n");
- ORE->emit(
- DiagnosticInfoOptimizationFailure(DEBUG_TYPE,
- "FailedRequestedUnrolling",
- L->getStartLoc(), L->getHeader())
- << "loop not unrolled: the optimizer was unable to perform the "
- "requested transformation; the transformation might be disabled or "
- "specified as part of an unsupported transformation ordering");
- }
- if (hasUnrollAndJamTransformation(L) == TM_ForcedByUser) {
- LLVM_DEBUG(dbgs() << "Leftover unroll-and-jam transformation\n");
- ORE->emit(
- DiagnosticInfoOptimizationFailure(DEBUG_TYPE,
- "FailedRequestedUnrollAndJamming",
- L->getStartLoc(), L->getHeader())
- << "loop not unroll-and-jammed: the optimizer was unable to perform "
- "the requested transformation; the transformation might be disabled "
- "or specified as part of an unsupported transformation ordering");
- }
- if (hasVectorizeTransformation(L) == TM_ForcedByUser) {
- LLVM_DEBUG(dbgs() << "Leftover vectorization transformation\n");
- Optional<ElementCount> VectorizeWidth =
- getOptionalElementCountLoopAttribute(L);
- Optional<int> InterleaveCount =
- getOptionalIntLoopAttribute(L, "llvm.loop.interleave.count");
- if (!VectorizeWidth || VectorizeWidth->isVector())
- ORE->emit(
- DiagnosticInfoOptimizationFailure(DEBUG_TYPE,
- "FailedRequestedVectorization",
- L->getStartLoc(), L->getHeader())
- << "loop not vectorized: the optimizer was unable to perform the "
- "requested transformation; the transformation might be disabled "
- "or specified as part of an unsupported transformation ordering");
- else if (InterleaveCount.getValueOr(0) != 1)
- ORE->emit(
- DiagnosticInfoOptimizationFailure(DEBUG_TYPE,
- "FailedRequestedInterleaving",
- L->getStartLoc(), L->getHeader())
- << "loop not interleaved: the optimizer was unable to perform the "
- "requested transformation; the transformation might be disabled "
- "or specified as part of an unsupported transformation ordering");
- }
- if (hasDistributeTransformation(L) == TM_ForcedByUser) {
- LLVM_DEBUG(dbgs() << "Leftover distribute transformation\n");
- ORE->emit(
- DiagnosticInfoOptimizationFailure(DEBUG_TYPE,
- "FailedRequestedDistribution",
- L->getStartLoc(), L->getHeader())
- << "loop not distributed: the optimizer was unable to perform the "
- "requested transformation; the transformation might be disabled or "
- "specified as part of an unsupported transformation ordering");
- }
- }
- static void warnAboutLeftoverTransformations(Function *F, LoopInfo *LI,
- OptimizationRemarkEmitter *ORE) {
- for (auto *L : LI->getLoopsInPreorder())
- warnAboutLeftoverTransformations(L, ORE);
- }
- // New pass manager boilerplate
- PreservedAnalyses
- WarnMissedTransformationsPass::run(Function &F, FunctionAnalysisManager &AM) {
- // Do not warn about not applied transformations if optimizations are
- // disabled.
- if (F.hasOptNone())
- return PreservedAnalyses::all();
- auto &ORE = AM.getResult<OptimizationRemarkEmitterAnalysis>(F);
- auto &LI = AM.getResult<LoopAnalysis>(F);
- warnAboutLeftoverTransformations(&F, &LI, &ORE);
- return PreservedAnalyses::all();
- }
- // Legacy pass manager boilerplate
- namespace {
- class WarnMissedTransformationsLegacy : public FunctionPass {
- public:
- static char ID;
- explicit WarnMissedTransformationsLegacy() : FunctionPass(ID) {
- initializeWarnMissedTransformationsLegacyPass(
- *PassRegistry::getPassRegistry());
- }
- bool runOnFunction(Function &F) override {
- if (skipFunction(F))
- return false;
- auto &ORE = getAnalysis<OptimizationRemarkEmitterWrapperPass>().getORE();
- auto &LI = getAnalysis<LoopInfoWrapperPass>().getLoopInfo();
- warnAboutLeftoverTransformations(&F, &LI, &ORE);
- return false;
- }
- void getAnalysisUsage(AnalysisUsage &AU) const override {
- AU.addRequired<OptimizationRemarkEmitterWrapperPass>();
- AU.addRequired<LoopInfoWrapperPass>();
- AU.setPreservesAll();
- }
- };
- } // end anonymous namespace
- char WarnMissedTransformationsLegacy::ID = 0;
- INITIALIZE_PASS_BEGIN(WarnMissedTransformationsLegacy, "transform-warning",
- "Warn about non-applied transformations", false, false)
- INITIALIZE_PASS_DEPENDENCY(LoopInfoWrapperPass)
- INITIALIZE_PASS_DEPENDENCY(OptimizationRemarkEmitterWrapperPass)
- INITIALIZE_PASS_END(WarnMissedTransformationsLegacy, "transform-warning",
- "Warn about non-applied transformations", false, false)
- Pass *llvm::createWarnMissedTransformationsPass() {
- return new WarnMissedTransformationsLegacy();
- }
|