-
Notifications
You must be signed in to change notification settings - Fork 12.3k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Inform pass manager when child loops are deleted
As part of the nontrivial unswitching we could end up removing child loops. This patch add a notification to the pass manager when that happens (using the markLoopAsDeleted callback). Without this there could be stale LoopAccessAnalysis results cached in the analysis manager. Those analysis results are cached based on a Loop* as key. Since the BumpPtrAllocator used to allocate Loop objects could be resetted between different runs of for example the loop-distribute pass (running on different functions), a new Loop object could be created using the same Loop pointer. And then when requiring the LoopAccessAnalysis for the loop we got the stale (corrupt) result from the destroyed loop. Reviewed By: aeubanks Differential Revision: https://reviews.llvm.org/D109257 (fixes PR51754) (cherry-picked from commit 0f0344d)
- Loading branch information
Showing
2 changed files
with
102 additions
and
12 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
71 changes: 71 additions & 0 deletions
71
llvm/test/Transforms/SimpleLoopUnswitch/nontrivial-unswitch-markloopasdeleted.ll
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,71 @@ | ||
; RUN: opt < %s -enable-loop-distribute -passes='loop-distribute,loop-mssa(simple-loop-unswitch<nontrivial>),loop-distribute' -o /dev/null -S -debug-pass-manager=verbose 2>&1 | FileCheck %s | ||
|
||
|
||
; Running loop-distribute will result in LoopAccessAnalysis being required and | ||
; cached in the LoopAnalysisManagerFunctionProxy. | ||
; | ||
; CHECK: Running analysis: LoopAccessAnalysis on Loop at depth 2 containing: %loop_a_inner<header><latch><exiting> | ||
|
||
|
||
; Then simple-loop-unswitch is removing/replacing some loops (resulting in | ||
; Loop objects used as key in the analyses cache is destroyed). So here we | ||
; want to see that any analysis results cached on the destroyed loop is | ||
; cleared. A special case here is that loop_a_inner is destroyed when | ||
; unswitching the parent loop. | ||
; | ||
; The bug solved and verified by this test case was related to the | ||
; SimpleLoopUnswitch not marking the Loop as removed, so we missed clearing | ||
; the analysis caches. | ||
; | ||
; CHECK: Running pass: SimpleLoopUnswitchPass on Loop at depth 1 containing: %loop_begin<header>,%loop_b,%loop_b_inner,%loop_b_inner_exit,%loop_a,%loop_a_inner,%loop_a_inner_exit,%latch<latch><exiting> | ||
; CHECK-NEXT: Clearing all analysis results for: loop_a_inner | ||
|
||
|
||
; When running loop-distribute the second time we can see that loop_a_inner | ||
; isn't analysed because the loop no longer exists (instead we find a new loop, | ||
; loop_a_inner.us). This kind of verifies that it was correct to remove the | ||
; loop_a_inner related analysis above. | ||
; | ||
; CHECK: Running analysis: LoopAccessAnalysis on Loop at depth 2 containing: %loop_a_inner.us<header><latch><exiting> | ||
|
||
|
||
define i32 @test6(i1* %ptr, i1 %cond1, i32* %a.ptr, i32* %b.ptr) { | ||
entry: | ||
br label %loop_begin | ||
|
||
loop_begin: | ||
%v = load i1, i1* %ptr | ||
br i1 %cond1, label %loop_a, label %loop_b | ||
|
||
loop_a: | ||
br label %loop_a_inner | ||
|
||
loop_a_inner: | ||
%va = load i1, i1* %ptr | ||
%a = load i32, i32* %a.ptr | ||
br i1 %va, label %loop_a_inner, label %loop_a_inner_exit | ||
|
||
loop_a_inner_exit: | ||
%a.lcssa = phi i32 [ %a, %loop_a_inner ] | ||
br label %latch | ||
|
||
loop_b: | ||
br label %loop_b_inner | ||
|
||
loop_b_inner: | ||
%vb = load i1, i1* %ptr | ||
%b = load i32, i32* %b.ptr | ||
br i1 %vb, label %loop_b_inner, label %loop_b_inner_exit | ||
|
||
loop_b_inner_exit: | ||
%b.lcssa = phi i32 [ %b, %loop_b_inner ] | ||
br label %latch | ||
|
||
latch: | ||
%ab.phi = phi i32 [ %a.lcssa, %loop_a_inner_exit ], [ %b.lcssa, %loop_b_inner_exit ] | ||
br i1 %v, label %loop_begin, label %loop_exit | ||
|
||
loop_exit: | ||
%ab.lcssa = phi i32 [ %ab.phi, %latch ] | ||
ret i32 %ab.lcssa | ||
} |