mirror of
https://github.com/RPCSX/llvm.git
synced 2025-01-21 03:37:47 +00:00
f81b3074f7
When eliminating or merging almost empty basic blocks, the existence of non-trivial PHI nodes is currently used to recognize potential loops of which the block is the header and keep the block. However, the current algorithm fails if the loops' exit condition is evaluated only with volatile values hence no PHI nodes in the header. Especially when such a loop is an outer loop of a nested loop, the loop is collapsed into a single loop which prevent later optimizations from being applied (e.g., transforming nested loops into simplified forms and loop vectorization). The patch augments the existing PHI node-based check by adding a pre-test if the BB actually belongs to a set of loop headers and not eliminating it if yes. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@264697 91177308-0d34-0410-b5e6-96231b3b80d8
59 lines
1.5 KiB
LLVM
59 lines
1.5 KiB
LLVM
; REQUIRES: asserts
|
|
; RUN: opt -loop-unswitch -disable-output -stats -info-output-file - < %s | FileCheck --check-prefix=STATS %s
|
|
; RUN: opt -loop-unswitch -simplifycfg -S < %s | FileCheck %s
|
|
; PR5373
|
|
|
|
; Loop unswitching shouldn't trivially unswitch the true case of condition %a
|
|
; in the code here because it leads to an infinite loop. While this doesn't
|
|
; contain any instructions with side effects, it's still a kind of side effect.
|
|
; It can trivially unswitch on the false cas of condition %a though.
|
|
|
|
; STATS: 2 loop-unswitch - Number of branches unswitched
|
|
; STATS: 2 loop-unswitch - Number of unswitches that are trivial
|
|
|
|
; CHECK-LABEL: @func_16(
|
|
; CHECK-NEXT: entry:
|
|
; CHECK-NEXT: br i1 %a, label %entry.split, label %abort0.split
|
|
|
|
; CHECK: entry.split:
|
|
; CHECK-NEXT: br i1 %b, label %for.body, label %abort1.split
|
|
|
|
; CHECK: for.body:
|
|
; CHECK-NEXT: br label %for.body
|
|
|
|
; CHECK: abort0.split:
|
|
; CHECK-NEXT: call void @end0() [[NOR_NUW:#[0-9]+]]
|
|
; CHECK-NEXT: unreachable
|
|
|
|
; CHECK: abort1.split:
|
|
; CHECK-NEXT: call void @end1() [[NOR_NUW]]
|
|
; CHECK-NEXT: unreachable
|
|
|
|
; CHECK: }
|
|
|
|
define void @func_16(i1 %a, i1 %b) nounwind {
|
|
entry:
|
|
br label %for.body
|
|
|
|
for.body:
|
|
br i1 %a, label %cond.end, label %abort0
|
|
|
|
cond.end:
|
|
br i1 %b, label %for.body, label %abort1
|
|
|
|
abort0:
|
|
call void @end0() noreturn nounwind
|
|
unreachable
|
|
|
|
abort1:
|
|
call void @end1() noreturn nounwind
|
|
unreachable
|
|
}
|
|
|
|
declare void @end0() noreturn
|
|
declare void @end1() noreturn
|
|
|
|
; CHECK: attributes #0 = { nounwind }
|
|
; CHECK: attributes #1 = { noreturn }
|
|
; CHECK: attributes [[NOR_NUW]] = { noreturn nounwind }
|