summaryrefslogtreecommitdiffstats
path: root/polly/include
diff options
context:
space:
mode:
authorMichael Kruse <llvm@meinersbur.de>2018-04-25 18:53:33 +0000
committerMichael Kruse <llvm@meinersbur.de>2018-04-25 18:53:33 +0000
commitbeffdb9daac6ed4545fcf241abad559387c6ea51 (patch)
tree6f10f13db14bc72b888e1830bf2466084d123503 /polly/include
parentd8803d3d9251d35499659538244e5dd4550f5aaf (diff)
downloadbcm5719-llvm-beffdb9daac6ed4545fcf241abad559387c6ea51.tar.gz
bcm5719-llvm-beffdb9daac6ed4545fcf241abad559387c6ea51.zip
[ScopDetect] Reject loop with multiple exit blocks.
The current statement domain derivation algorithm does not (always) consider that different exit blocks of a loop can have different conditions to be reached. From the code for (int i = n; ; i-=2) { if (i <= 0) goto even; if (i <= 1) goto odd; A[i] = i; } even: A[0] = 42; return; odd: A[1] = 21; return; Polly currently derives the following domains: Stmt_even_critedge Domain := [n] -> { Stmt_even_critedge[] }; Stmt_odd Domain := [n] -> { Stmt_odd[] : (1 + n) mod 2 = 0 and n > 0 }; while the domain for the odd case is correct, Stmt_even is assumed to be executed unconditionally, which is obviously wrong. While projecting out the loop dimension in `adjustDomainDimensions`, it does not consider that there are other exit condition that have matched before. I don't know a how to fix this without changing a lot of code. Therefore This patch rejects loops with multiple exist blocks to fix the miscompile of test-suite's uuencode. The odd condition is transformed by LLVM to %cmp1 = icmp eq i64 %indvars.iv, 1 such that the project_out in adjustDomainDimensions() indeed only matches for odd n (using this condition only, we'd have an infinite loop otherwise). The even condition manifests as %cmp = icmp slt i64 %indvars.iv, 3 Because buildDomainsWithBranchConstraints() does not consider other exit conditions, it has to assume that the induction variable will eventually be lower than 3 and taking this exit. IMHO we need to reuse the algorithm that determines the number of iterations (addLoopBoundsToHeaderDomain) to determine which exit condition applies first. It has to happen in buildDomainsWithBranchConstraints() because the result will need to propagate to successor BBs. Currently addLoopBoundsToHeaderDomain() just look for union of all backedge conditions (which means leaving not the loop here). The patch in llvm.org/PR35465 changes it to look for exit conditions instead. This is required because there might be other exit conditions that do not alternatively go back to the loop header. Differential Revision: https://reviews.llvm.org/D45649 llvm-svn: 330858
Diffstat (limited to 'polly/include')
-rw-r--r--polly/include/polly/ScopDetectionDiagnostic.h29
1 files changed, 29 insertions, 0 deletions
diff --git a/polly/include/polly/ScopDetectionDiagnostic.h b/polly/include/polly/ScopDetectionDiagnostic.h
index 12f79d14409..200c77f852c 100644
--- a/polly/include/polly/ScopDetectionDiagnostic.h
+++ b/polly/include/polly/ScopDetectionDiagnostic.h
@@ -90,6 +90,7 @@ enum class RejectReasonKind {
LoopBound,
LoopHasNoExit,
+ LoopHasMultipleExits,
LoopOnlySomeLatches,
FuncCall,
@@ -585,6 +586,34 @@ public:
};
//===----------------------------------------------------------------------===//
+/// Captures errors when a loop has multiple exists.
+class ReportLoopHasMultipleExits : public RejectReason {
+ /// The loop that has multiple exits.
+ Loop *L;
+
+ const DebugLoc Loc;
+
+public:
+ ReportLoopHasMultipleExits(Loop *L)
+ : RejectReason(RejectReasonKind::LoopHasMultipleExits), L(L),
+ Loc(L->getStartLoc()) {}
+
+ /// @name LLVM-RTTI interface
+ //@{
+ static bool classof(const RejectReason *RR);
+ //@}
+
+ /// @name RejectReason interface
+ //@{
+ std::string getRemarkName() const override;
+ const Value *getRemarkBB() const override;
+ std::string getMessage() const override;
+ const DebugLoc &getDebugLoc() const override;
+ std::string getEndUserMessage() const override;
+ //@}
+};
+
+//===----------------------------------------------------------------------===//
/// Captures errors when not all loop latches are part of the scop.
class ReportLoopOnlySomeLatches : public RejectReason {
/// The loop for which not all loop latches are part of the scop.
OpenPOWER on IntegriCloud