
Unfortunately alive2 cannot prove the correctness due to fails by timeout even for float type half. However it should be correct. If a and b are not NaN, maximum and minimum will just return different values (a and b) and take into account a + b == b + a this is the same. If a or b is NaN, than maximum and minimum are equal to NaN and NaN + NaN is NaN. a + b is also a NaN. In terms of preserving fast flags, we cannot preserve ninf due to minimum(NaN, Infinity) == maximum(NaN, Infinity) == NaN, minimum(NaN, Infinity) +ninf maximum(NaN, Infinity) == NaN +ninf NaN = NaN However transformation will change minimum(NaN, Infinity) + maximum(NaN, Infinity) to NaN +ninf Infinity == poison. But if fadd is marked as nnan, we can preserve because NaN +ninf/nnan NaN = poison as well. The same optimization for maximum(a,b) * minimum(a,b) => a * b is added. All said above for fadd is correct for fmul. Reviewed By: mkazantsev Differential Revision: https://reviews.llvm.org/D147299
99 lines
3.1 KiB
LLVM
99 lines
3.1 KiB
LLVM
; NOTE: Assertions have been autogenerated by utils/update_test_checks.py
|
|
; RUN: opt < %s -passes=instcombine -S | FileCheck %s
|
|
|
|
declare float @llvm.minimum.f32(float %Val0, float %Val1)
|
|
declare float @llvm.maximum.f32(float %Val0, float %Val1)
|
|
declare <4 x float> @llvm.minimum.v4f32(<4 x float> %Val0, <4 x float> %Val1)
|
|
declare <4 x float> @llvm.maximum.v4f32(<4 x float> %Val0, <4 x float> %Val1)
|
|
|
|
define float @test(float %a, float %b) {
|
|
; CHECK-LABEL: @test(
|
|
; CHECK-NEXT: entry:
|
|
; CHECK-NEXT: [[RES:%.*]] = fadd float [[A:%.*]], [[B:%.*]]
|
|
; CHECK-NEXT: ret float [[RES]]
|
|
;
|
|
entry:
|
|
%min = call float @llvm.minimum.f32(float %a, float %b)
|
|
%max = call float @llvm.maximum.f32(float %a, float %b)
|
|
%res = fadd float %min, %max
|
|
ret float %res
|
|
}
|
|
|
|
define float @test_comm1(float %a, float %b) {
|
|
; CHECK-LABEL: @test_comm1(
|
|
; CHECK-NEXT: entry:
|
|
; CHECK-NEXT: [[RES:%.*]] = fadd float [[A:%.*]], [[B:%.*]]
|
|
; CHECK-NEXT: ret float [[RES]]
|
|
;
|
|
entry:
|
|
%min = call float @llvm.minimum.f32(float %a, float %b)
|
|
%max = call float @llvm.maximum.f32(float %a, float %b)
|
|
%res = fadd float %max, %min
|
|
ret float %res
|
|
}
|
|
|
|
define float @test_comm2(float %a, float %b) {
|
|
; CHECK-LABEL: @test_comm2(
|
|
; CHECK-NEXT: entry:
|
|
; CHECK-NEXT: [[RES:%.*]] = fadd float [[B:%.*]], [[A:%.*]]
|
|
; CHECK-NEXT: ret float [[RES]]
|
|
;
|
|
entry:
|
|
%min = call float @llvm.minimum.f32(float %a, float %b)
|
|
%max = call float @llvm.maximum.f32(float %b, float %a)
|
|
%res = fadd float %min, %max
|
|
ret float %res
|
|
}
|
|
|
|
define float @test_comm3(float %a, float %b) {
|
|
; CHECK-LABEL: @test_comm3(
|
|
; CHECK-NEXT: entry:
|
|
; CHECK-NEXT: [[RES:%.*]] = fadd float [[B:%.*]], [[A:%.*]]
|
|
; CHECK-NEXT: ret float [[RES]]
|
|
;
|
|
entry:
|
|
%min = call float @llvm.minimum.f32(float %a, float %b)
|
|
%max = call float @llvm.maximum.f32(float %b, float %a)
|
|
%res = fadd float %max, %min
|
|
ret float %res
|
|
}
|
|
|
|
define <4 x float> @test_vect(<4 x float> %a, <4 x float> %b) {
|
|
; CHECK-LABEL: @test_vect(
|
|
; CHECK-NEXT: entry:
|
|
; CHECK-NEXT: [[RES:%.*]] = fadd <4 x float> [[B:%.*]], [[A:%.*]]
|
|
; CHECK-NEXT: ret <4 x float> [[RES]]
|
|
;
|
|
entry:
|
|
%min = call <4 x float> @llvm.minimum.v4f32(<4 x float> %a, <4 x float> %b)
|
|
%max = call <4 x float> @llvm.maximum.v4f32(<4 x float> %b, <4 x float> %a)
|
|
%res = fadd <4 x float> %min, %max
|
|
ret <4 x float> %res
|
|
}
|
|
|
|
define float @test_flags(float %a, float %b) {
|
|
; CHECK-LABEL: @test_flags(
|
|
; CHECK-NEXT: entry:
|
|
; CHECK-NEXT: [[RES:%.*]] = fadd fast float [[A:%.*]], [[B:%.*]]
|
|
; CHECK-NEXT: ret float [[RES]]
|
|
;
|
|
entry:
|
|
%min = call float @llvm.minimum.f32(float %a, float %b)
|
|
%max = call float @llvm.maximum.f32(float %a, float %b)
|
|
%res = fadd fast float %min, %max
|
|
ret float %res
|
|
}
|
|
|
|
define float @test_flags2(float %a, float %b) {
|
|
; CHECK-LABEL: @test_flags2(
|
|
; CHECK-NEXT: entry:
|
|
; CHECK-NEXT: [[RES:%.*]] = fadd reassoc nsz arcp contract afn float [[A:%.*]], [[B:%.*]]
|
|
; CHECK-NEXT: ret float [[RES]]
|
|
;
|
|
entry:
|
|
%min = call float @llvm.minimum.f32(float %a, float %b)
|
|
%max = call float @llvm.maximum.f32(float %a, float %b)
|
|
%res = fadd reassoc ninf nsz arcp contract afn float %min, %max
|
|
ret float %res
|
|
}
|