2018-02-13 03:14:01 +08:00
|
|
|
; NOTE: Assertions have been autogenerated by utils/update_test_checks.py
|
2011-03-30 23:42:35 +08:00
|
|
|
; RUN: opt -S -instcombine < %s | FileCheck %s
|
|
|
|
|
2018-02-15 03:54:51 +08:00
|
|
|
define float @exact_inverse(float %x) {
|
|
|
|
; CHECK-LABEL: @exact_inverse(
|
|
|
|
; CHECK-NEXT: [[DIV:%.*]] = fmul float [[X:%.*]], 1.250000e-01
|
|
|
|
; CHECK-NEXT: ret float [[DIV]]
|
|
|
|
;
|
|
|
|
%div = fdiv float %x, 8.0
|
|
|
|
ret float %div
|
|
|
|
}
|
|
|
|
|
|
|
|
; Min normal float = 1.17549435E-38
|
|
|
|
|
|
|
|
define float @exact_inverse2(float %x) {
|
|
|
|
; CHECK-LABEL: @exact_inverse2(
|
2018-02-13 03:14:01 +08:00
|
|
|
; CHECK-NEXT: [[DIV:%.*]] = fmul float [[X:%.*]], 0x47D0000000000000
|
|
|
|
; CHECK-NEXT: ret float [[DIV]]
|
|
|
|
;
|
2011-03-30 23:42:35 +08:00
|
|
|
%div = fdiv float %x, 0x3810000000000000
|
|
|
|
ret float %div
|
|
|
|
}
|
|
|
|
|
2018-02-15 03:54:51 +08:00
|
|
|
; Max exponent = 1.70141183E+38; don't transform to multiply with denormal.
|
|
|
|
|
|
|
|
define float @exact_inverse_but_denorm(float %x) {
|
|
|
|
; CHECK-LABEL: @exact_inverse_but_denorm(
|
2018-02-13 03:14:01 +08:00
|
|
|
; CHECK-NEXT: [[DIV:%.*]] = fdiv float [[X:%.*]], 0x47E0000000000000
|
|
|
|
; CHECK-NEXT: ret float [[DIV]]
|
|
|
|
;
|
2011-03-30 23:42:35 +08:00
|
|
|
%div = fdiv float %x, 0x47E0000000000000
|
|
|
|
ret float %div
|
|
|
|
}
|
|
|
|
|
2018-02-15 03:54:51 +08:00
|
|
|
; Denormal = float 1.40129846E-45; inverse can't be represented.
|
|
|
|
|
|
|
|
define float @not_exact_inverse2(float %x) {
|
|
|
|
; CHECK-LABEL: @not_exact_inverse2(
|
2018-02-13 03:14:01 +08:00
|
|
|
; CHECK-NEXT: [[DIV:%.*]] = fdiv float [[X:%.*]], 0x36A0000000000000
|
|
|
|
; CHECK-NEXT: ret float [[DIV]]
|
|
|
|
;
|
2011-03-30 23:42:35 +08:00
|
|
|
%div = fdiv float %x, 0x36A0000000000000
|
|
|
|
ret float %div
|
|
|
|
}
|
2014-01-17 05:07:52 +08:00
|
|
|
|
2018-02-15 03:54:51 +08:00
|
|
|
; Fast math allows us to replace this fdiv.
|
|
|
|
|
|
|
|
define float @not_exact_but_allow_recip(float %x) {
|
|
|
|
; CHECK-LABEL: @not_exact_but_allow_recip(
|
|
|
|
; CHECK-NEXT: [[DIV:%.*]] = fmul arcp float [[X:%.*]], 0x3FD5555560000000
|
|
|
|
; CHECK-NEXT: ret float [[DIV]]
|
|
|
|
;
|
|
|
|
%div = fdiv arcp float %x, 3.0
|
|
|
|
ret float %div
|
|
|
|
}
|
|
|
|
|
|
|
|
; Fast math allows us to replace this fdiv, but we don't to avoid a denormal.
|
|
|
|
; TODO: What if the function attributes tell us that denormals are flushed?
|
|
|
|
|
|
|
|
define float @not_exact_but_allow_recip_but_denorm(float %x) {
|
|
|
|
; CHECK-LABEL: @not_exact_but_allow_recip_but_denorm(
|
|
|
|
; CHECK-NEXT: [[DIV:%.*]] = fdiv arcp float [[X:%.*]], 0x47E0000100000000
|
2018-02-13 03:14:01 +08:00
|
|
|
; CHECK-NEXT: ret float [[DIV]]
|
|
|
|
;
|
2018-02-15 03:54:51 +08:00
|
|
|
%div = fdiv arcp float %x, 0x47E0000100000000
|
2014-01-17 05:07:52 +08:00
|
|
|
ret float %div
|
|
|
|
}
|
2014-01-20 15:44:53 +08:00
|
|
|
|
2018-02-15 03:54:51 +08:00
|
|
|
define <2 x float> @exact_inverse_splat(<2 x float> %x) {
|
|
|
|
; CHECK-LABEL: @exact_inverse_splat(
|
2018-02-15 21:55:52 +08:00
|
|
|
; CHECK-NEXT: [[DIV:%.*]] = fmul <2 x float> [[X:%.*]], <float 2.500000e-01, float 2.500000e-01>
|
2018-02-15 03:54:51 +08:00
|
|
|
; CHECK-NEXT: ret <2 x float> [[DIV]]
|
|
|
|
;
|
|
|
|
%div = fdiv <2 x float> %x, <float 4.0, float 4.0>
|
|
|
|
ret <2 x float> %div
|
|
|
|
}
|
|
|
|
|
2018-02-15 21:55:52 +08:00
|
|
|
; Fast math allows us to replace this fdiv.
|
|
|
|
|
|
|
|
define <2 x float> @not_exact_but_allow_recip_splat(<2 x float> %x) {
|
|
|
|
; CHECK-LABEL: @not_exact_but_allow_recip_splat(
|
|
|
|
; CHECK-NEXT: [[DIV:%.*]] = fmul arcp <2 x float> [[X:%.*]], <float 0x3FD5555560000000, float 0x3FD5555560000000>
|
|
|
|
; CHECK-NEXT: ret <2 x float> [[DIV]]
|
|
|
|
;
|
|
|
|
%div = fdiv arcp <2 x float> %x, <float 3.0, float 3.0>
|
|
|
|
ret <2 x float> %div
|
|
|
|
}
|
|
|
|
|
|
|
|
; FIXME: Vector neglect.
|
|
|
|
|
2018-02-15 03:54:51 +08:00
|
|
|
define <2 x float> @exact_inverse_vec(<2 x float> %x) {
|
|
|
|
; CHECK-LABEL: @exact_inverse_vec(
|
|
|
|
; CHECK-NEXT: [[DIV:%.*]] = fdiv <2 x float> [[X:%.*]], <float 4.000000e+00, float 8.000000e+00>
|
|
|
|
; CHECK-NEXT: ret <2 x float> [[DIV]]
|
|
|
|
;
|
|
|
|
%div = fdiv <2 x float> %x, <float 4.0, float 8.0>
|
|
|
|
ret <2 x float> %div
|
|
|
|
}
|
|
|
|
|
|
|
|
define <2 x float> @not_exact_inverse_splat(<2 x float> %x) {
|
|
|
|
; CHECK-LABEL: @not_exact_inverse_splat(
|
|
|
|
; CHECK-NEXT: [[DIV:%.*]] = fdiv <2 x float> [[X:%.*]], <float 3.000000e+00, float 3.000000e+00>
|
|
|
|
; CHECK-NEXT: ret <2 x float> [[DIV]]
|
|
|
|
;
|
|
|
|
%div = fdiv <2 x float> %x, <float 3.0, float 3.0>
|
|
|
|
ret <2 x float> %div
|
|
|
|
}
|
|
|
|
|
|
|
|
define <2 x float> @not_exact_inverse_vec(<2 x float> %x) {
|
|
|
|
; CHECK-LABEL: @not_exact_inverse_vec(
|
|
|
|
; CHECK-NEXT: [[DIV:%.*]] = fdiv <2 x float> [[X:%.*]], <float 4.000000e+00, float 3.000000e+00>
|
|
|
|
; CHECK-NEXT: ret <2 x float> [[DIV]]
|
|
|
|
;
|
|
|
|
%div = fdiv <2 x float> %x, <float 4.0, float 3.0>
|
|
|
|
ret <2 x float> %div
|
|
|
|
}
|
|
|
|
|
2018-02-13 03:14:01 +08:00
|
|
|
define float @test5(float %x, float %y, float %z) {
|
|
|
|
; CHECK-LABEL: @test5(
|
|
|
|
; CHECK-NEXT: [[TMP1:%.*]] = fmul fast float [[Y:%.*]], [[Z:%.*]]
|
|
|
|
; CHECK-NEXT: [[DIV2:%.*]] = fdiv fast float [[X:%.*]], [[TMP1]]
|
|
|
|
; CHECK-NEXT: ret float [[DIV2]]
|
|
|
|
;
|
2014-01-20 15:44:53 +08:00
|
|
|
%div1 = fdiv fast float %x, %y
|
|
|
|
%div2 = fdiv fast float %div1, %z
|
|
|
|
ret float %div2
|
|
|
|
}
|
|
|
|
|
2018-02-13 03:14:01 +08:00
|
|
|
define float @test6(float %x, float %y, float %z) {
|
|
|
|
; CHECK-LABEL: @test6(
|
|
|
|
; CHECK-NEXT: [[TMP1:%.*]] = fmul fast float [[Z:%.*]], [[Y:%.*]]
|
|
|
|
; CHECK-NEXT: [[DIV2:%.*]] = fdiv fast float [[TMP1]], [[X:%.*]]
|
|
|
|
; CHECK-NEXT: ret float [[DIV2]]
|
|
|
|
;
|
2014-01-20 15:44:53 +08:00
|
|
|
%div1 = fdiv fast float %x, %y
|
|
|
|
%div2 = fdiv fast float %z, %div1
|
|
|
|
ret float %div2
|
|
|
|
}
|
2017-01-11 07:08:54 +08:00
|
|
|
|
|
|
|
define float @fdiv_fneg_fneg(float %x, float %y) {
|
2018-02-13 03:14:01 +08:00
|
|
|
; CHECK-LABEL: @fdiv_fneg_fneg(
|
|
|
|
; CHECK-NEXT: [[DIV:%.*]] = fdiv float [[X:%.*]], [[Y:%.*]]
|
|
|
|
; CHECK-NEXT: ret float [[DIV]]
|
|
|
|
;
|
2017-01-11 07:08:54 +08:00
|
|
|
%x.fneg = fsub float -0.0, %x
|
|
|
|
%y.fneg = fsub float -0.0, %y
|
|
|
|
%div = fdiv float %x.fneg, %y.fneg
|
|
|
|
ret float %div
|
|
|
|
}
|
|
|
|
|
2018-02-13 03:14:01 +08:00
|
|
|
; The test above shows that no FMF are needed, but show that we are not dropping FMF.
|
|
|
|
|
2017-01-11 07:08:54 +08:00
|
|
|
define float @fdiv_fneg_fneg_fast(float %x, float %y) {
|
2018-02-13 03:14:01 +08:00
|
|
|
; CHECK-LABEL: @fdiv_fneg_fneg_fast(
|
|
|
|
; CHECK-NEXT: [[DIV:%.*]] = fdiv fast float [[X:%.*]], [[Y:%.*]]
|
|
|
|
; CHECK-NEXT: ret float [[DIV]]
|
|
|
|
;
|
2017-01-11 07:08:54 +08:00
|
|
|
%x.fneg = fsub float -0.0, %x
|
|
|
|
%y.fneg = fsub float -0.0, %y
|
|
|
|
%div = fdiv fast float %x.fneg, %y.fneg
|
|
|
|
ret float %div
|
|
|
|
}
|
2018-02-13 03:14:01 +08:00
|
|
|
|
2018-02-13 03:23:39 +08:00
|
|
|
; X / (X * Y) --> 1.0 / Y
|
|
|
|
|
|
|
|
define float @div_factor(float %x, float %y) {
|
|
|
|
; CHECK-LABEL: @div_factor(
|
2018-02-13 03:39:21 +08:00
|
|
|
; CHECK-NEXT: [[D:%.*]] = fdiv reassoc nnan float 1.000000e+00, [[Y:%.*]]
|
2018-02-13 03:23:39 +08:00
|
|
|
; CHECK-NEXT: ret float [[D]]
|
|
|
|
;
|
|
|
|
%m = fmul float %x, %y
|
|
|
|
%d = fdiv nnan reassoc float %x, %m
|
|
|
|
ret float %d;
|
|
|
|
}
|
|
|
|
|
|
|
|
; We can't do the transform without 'nnan' because if x is NAN and y is a number, this should return NAN.
|
|
|
|
|
|
|
|
define float @div_factor_too_strict(float %x, float %y) {
|
|
|
|
; CHECK-LABEL: @div_factor_too_strict(
|
|
|
|
; CHECK-NEXT: [[M:%.*]] = fmul float [[X:%.*]], [[Y:%.*]]
|
|
|
|
; CHECK-NEXT: [[D:%.*]] = fdiv reassoc float [[X]], [[M]]
|
|
|
|
; CHECK-NEXT: ret float [[D]]
|
|
|
|
;
|
|
|
|
%m = fmul float %x, %y
|
|
|
|
%d = fdiv reassoc float %x, %m
|
|
|
|
ret float %d
|
|
|
|
}
|
|
|
|
|
|
|
|
; Commute, verify vector types, and show that we are not dropping extra FMF.
|
|
|
|
; X / (Y * X) --> 1.0 / Y
|
|
|
|
|
|
|
|
define <2 x float> @div_factor_commute(<2 x float> %x, <2 x float> %y) {
|
|
|
|
; CHECK-LABEL: @div_factor_commute(
|
2018-02-13 03:39:21 +08:00
|
|
|
; CHECK-NEXT: [[D:%.*]] = fdiv reassoc nnan ninf nsz <2 x float> <float 1.000000e+00, float 1.000000e+00>, [[Y:%.*]]
|
2018-02-13 03:23:39 +08:00
|
|
|
; CHECK-NEXT: ret <2 x float> [[D]]
|
|
|
|
;
|
|
|
|
%m = fmul <2 x float> %y, %x
|
|
|
|
%d = fdiv nnan ninf nsz reassoc <2 x float> %x, %m
|
|
|
|
ret <2 x float> %d
|
|
|
|
}
|
|
|
|
|