This adjusts mayHaveSideEffect() to return true for !willReturn() instructions. Just like other side-effects, non-willreturn calls (aka "divergence") cannot be removed and cannot be reordered relative to other side effects. This fixes a number of bugs where non-willreturn calls are either incorrectly dropped or moved. In particular, it also fixes the last open problem in https://bugs.llvm.org/show_bug.cgi?id=50511. I performed a cursory review of all current mayHaveSideEffect() uses, which convinced me that these are indeed the desired default semantics. Places that do not want to consider non-willreturn as a sideeffect generally do not want mayHaveSideEffect() semantics at all. I identified two such cases, which are addressed by D106591 and D106742. Finally, there is a use in SCEV for which we don't really have an appropriate API right now -- what it wants is basically "would this be considered forward progress". I've just spelled out the previous semantics there. Differential Revision: https://reviews.llvm.org/D106749
45 lines
1.3 KiB
LLVM
45 lines
1.3 KiB
LLVM
; NOTE: Assertions have been autogenerated by utils/update_test_checks.py
|
|
; RUN: opt < %s -sccp -loop-deletion -simplifycfg -simplifycfg-require-and-preserve-domtree=1 -S | FileCheck %s
|
|
|
|
declare double @sqrt(double) readnone nounwind willreturn
|
|
%empty = type {}
|
|
declare %empty @has_side_effects()
|
|
|
|
; No matter how hard you try, sqrt(1.0) is always 1.0. This allows the
|
|
; optimizer to delete this loop.
|
|
define double @test_0(i32 %param) {
|
|
; CHECK-LABEL: @test_0(
|
|
; CHECK-NEXT: entry:
|
|
; CHECK-NEXT: ret double 1.000000e+00
|
|
;
|
|
entry:
|
|
br label %Loop
|
|
Loop: ; preds = %Loop, %entry
|
|
%I2 = phi i32 [ 0, %entry ], [ %I3, %Loop ]
|
|
%V = phi double [ 1.000000e+00, %entry ], [ %V2, %Loop ]
|
|
%V2 = call double @sqrt( double %V )
|
|
%I3 = add i32 %I2, 1
|
|
%tmp.7 = icmp ne i32 %I3, %param
|
|
br i1 %tmp.7, label %Loop, label %Exit
|
|
Exit: ; preds = %Loop
|
|
ret double %V
|
|
}
|
|
|
|
define i32 @test_1() {
|
|
; CHECK-LABEL: @test_1(
|
|
; CHECK-NEXT: [[TMP1:%.*]] = call [[EMPTY:%.*]] @has_side_effects()
|
|
; CHECK-NEXT: ret i32 0
|
|
;
|
|
%1 = call %empty @has_side_effects()
|
|
ret i32 0
|
|
}
|
|
|
|
define i32 @test_not_willreturn() {
|
|
; CHECK-LABEL: @test_not_willreturn(
|
|
; CHECK-NEXT: [[TMP1:%.*]] = call [[EMPTY:%.*]] @has_side_effects() #[[ATTR1:[0-9]+]]
|
|
; CHECK-NEXT: ret i32 0
|
|
;
|
|
%1 = call %empty @has_side_effects() nounwind readonly
|
|
ret i32 0
|
|
}
|