; NOTE: Assertions have been autogenerated by utils/update_test_checks.py
; RUN: opt -S -passes=instcombine %s | FileCheck %s
; --------------------------------------------------------------------
; (@canonicalize(x) == @canonicalize(y)) is equivalent to to (x == y)
; --------------------------------------------------------------------
define i1 @canonicalize_oeq_canonicalize_f32(float %x, float %y) {
; CHECK-LABEL: @canonicalize_oeq_canonicalize_f32(
; CHECK-NEXT: [[CMP:%.*]] = fcmp oeq float [[X:%.*]], [[Y:%.*]]
; CHECK-NEXT: ret i1 [[CMP]]
;
%canon.x = call float @llvm.canonicalize.f32(float %x)
%canon.y = call float @llvm.canonicalize.f32(float %y)
%cmp = fcmp oeq float %canon.x, %canon.y
ret i1 %cmp
}
define i1 @canonicalize_oeq_canonicalize_f32_flags(float %x, float %y) {
; CHECK-LABEL: @canonicalize_oeq_canonicalize_f32_flags(
; CHECK-NEXT: [[CMP:%.*]] = fcmp nsz oeq float [[X:%.*]], [[Y:%.*]]
; CHECK-NEXT: ret i1 [[CMP]]
;
%canon.x = call float @llvm.canonicalize.f32(float %x)
%canon.y = call float @llvm.canonicalize.f32(float %y)
%cmp = fcmp nsz oeq float %canon.x, %canon.y
ret i1 %cmp
}
define <2 x i1> @canonicalize_oeq_canonicalize_v2f32(<2 x float> %x, <2 x float> %y) {
; CHECK-LABEL: @canonicalize_oeq_canonicalize_v2f32(
; CHECK-NEXT: [[CMP:%.*]] = fcmp oeq <2 x float> [[X:%.*]], [[Y:%.*]]
; CHECK-NEXT: ret <2 x i1> [[CMP]]
;
%canon.x = call <2 x float> @llvm.canonicalize.v2f32(<2 x float> %x)
%canon.y = call <2 x float> @llvm.canonicalize.v2f32(<2 x float> %y)
%cmp = fcmp oeq <2 x float> %canon.x, %canon.y
ret <2 x i1> %cmp
}
define i1 @canonicalize_ueq_canonicalize_f32(float %x, float %y) {
; CHECK-LABEL: @canonicalize_ueq_canonicalize_f32(
; CHECK-NEXT: [[CMP:%.*]] = fcmp ueq float [[X:%.*]], [[Y:%.*]]
; CHECK-NEXT: ret i1 [[CMP]]
;
%canon.x = call float @llvm.canonicalize.f32(float %x)
%canon.y = call float @llvm.canonicalize.f32(float %y)
%cmp = fcmp ueq float %canon.x, %canon.y
ret i1 %cmp
}
define <2 x i1> @canonicalize_ueq_canonicalize_v2f32(<2 x float> %x, <2 x float> %y) {
; CHECK-LABEL: @canonicalize_ueq_canonicalize_v2f32(
; CHECK-NEXT: [[CMP:%.*]] = fcmp ueq <2 x float> [[X:%.*]], [[Y:%.*]]
; CHECK-NEXT: ret <2 x i1> [[CMP]]
;
%canon.x = call <2 x float> @llvm.canonicalize.v2f32(<2 x float> %x)
%canon.y = call <2 x float> @llvm.canonicalize.v2f32(<2 x float> %y)
%cmp = fcmp ueq <2 x float> %canon.x, %canon.y
ret <2 x i1> %cmp
}
define i1 @canonicalize_one_canonicalize_f32(float %x, float %y) {
; CHECK-LABEL: @canonicalize_one_canonicalize_f32(
; CHECK-NEXT: [[CMP:%.*]] = fcmp one float [[X:%.*]], [[Y:%.*]]
; CHECK-NEXT: ret i1 [[CMP]]
;
%canon.x = call float @llvm.canonicalize.f32(float %x)
%canon.y = call float @llvm.canonicalize.f32(float %y)
%cmp = fcmp one float %canon.x, %canon.y
ret i1 %cmp
}
define <2 x i1> @canonicalize_one_canonicalize_v2f32(<2 x float> %x, <2 x float> %y) {
; CHECK-LABEL: @canonicalize_one_canonicalize_v2f32(
; CHECK-NEXT: [[CMP:%.*]] = fcmp one <2 x float> [[X:%.*]], [[Y:%.*]]
; CHECK-NEXT: ret <2 x i1> [[CMP]]
;
%canon.x = call <2 x float> @llvm.canonicalize.v2f32(<2 x float> %x)
%canon.y = call <2 x float> @llvm.canonicalize.v2f32(<2 x float> %y)
%cmp = fcmp one <2 x float> %canon.x, %canon.y
ret <2 x i1> %cmp
}
define i1 @canonicalize_une_canonicalize_f32(float %x, float %y) {
; CHECK-LABEL: @canonicalize_une_canonicalize_f32(
; CHECK-NEXT: [[CMP:%.*]] = fcmp une float [[X:%.*]], [[Y:%.*]]
; CHECK-NEXT: ret i1 [[CMP]]
;
%canon.x = call float @llvm.canonicalize.f32(float %x)
%canon.y = call float @llvm.canonicalize.f32(float %y)
%cmp = fcmp une float %canon.x, %canon.y
ret i1 %cmp
}
define <2 x i1> @canonicalize_une_canonicalize_v2f32(<2 x float> %x, <2 x float> %y) {
; CHECK-LABEL: @canonicalize_une_canonicalize_v2f32(
; CHECK-NEXT: [[CMP:%.*]] = fcmp une <2 x float> [[X:%.*]], [[Y:%.*]]
; CHECK-NEXT: ret <2 x i1> [[CMP]]
;
%canon.x = call <2 x float> @llvm.canonicalize.v2f32(<2 x float> %x)
%canon.y = call <2 x float> @llvm.canonicalize.v2f32(<2 x float> %y)
%cmp = fcmp une <2 x float> %canon.x, %canon.y
ret <2 x i1> %cmp
}
define i1 @canonicalize_ogt_canonicalize_f32(float %x, float %y) {
; CHECK-LABEL: @canonicalize_ogt_canonicalize_f32(
; CHECK-NEXT: [[CMP:%.*]] = fcmp ogt float [[X:%.*]], [[Y:%.*]]
; CHECK-NEXT: ret i1 [[CMP]]
;
%canon.x = call float @llvm.canonicalize.f32(float %x)
%canon.y = call float @llvm.canonicalize.f32(float %y)
%cmp = fcmp ogt float %canon.x, %canon.y
ret i1 %cmp
}
define i1 @canonicalize_oge_canonicalize_f32(float %x, float %y) {
; CHECK-LABEL: @canonicalize_oge_canonicalize_f32(
; CHECK-NEXT: [[CMP:%.*]] = fcmp oge float [[X:%.*]], [[Y:%.*]]
; CHECK-NEXT: ret i1 [[CMP]]
;
%canon.x = call float @llvm.canonicalize.f32(float %x)
%canon.y = call float @llvm.canonicalize.f32(float %y)
%cmp = fcmp oge float %canon.x, %canon.y
ret i1 %cmp
}
define i1 @canonicalize_olt_canonicalize_f32(float %x, float %y) {
; CHECK-LABEL: @canonicalize_olt_canonicalize_f32(
; CHECK-NEXT: [[CMP:%.*]] = fcmp olt float [[X:%.*]], [[Y:%.*]]
; CHECK-NEXT: ret i1 [[CMP]]
;
%canon.x = call float @llvm.canonicalize.f32(float %x)
%canon.y = call float @llvm.canonicalize.f32(float %y)
%cmp = fcmp olt float %canon.x, %canon.y
ret i1 %cmp
}
define i1 @canonicalize_ole_canonicalize_f32(float %x, float %y) {
; CHECK-LABEL: @canonicalize_ole_canonicalize_f32(
; CHECK-NEXT: [[CMP:%.*]] = fcmp ole float [[X:%.*]], [[Y:%.*]]
; CHECK-NEXT: ret i1 [[CMP]]
;
%canon.x = call float @llvm.canonicalize.f32(float %x)
%canon.y = call float @llvm.canonicalize.f32(float %y)
%cmp = fcmp ole float %canon.x, %canon.y
ret i1 %cmp
}
define i1 @canonicalize_ord_canonicalize_f32(float %x, float %y) {
; CHECK-LABEL: @canonicalize_ord_canonicalize_f32(
; CHECK-NEXT: [[CMP:%.*]] = fcmp ord float [[X:%.*]], [[Y:%.*]]
; CHECK-NEXT: ret i1 [[CMP]]
;
%canon.x = call float @llvm.canonicalize.f32(float %x)
%canon.y = call float @llvm.canonicalize.f32(float %y)
%cmp = fcmp ord float %canon.x, %canon.y
ret i1 %cmp
}
define i1 @canonicalize_ugt_canonicalize_f32(float %x, float %y) {
; CHECK-LABEL: @canonicalize_ugt_canonicalize_f32(
; CHECK-NEXT: [[CMP:%.*]] = fcmp ugt float [[X:%.*]], [[Y:%.*]]
; CHECK-NEXT: ret i1 [[CMP]]
;
%canon.x = call float @llvm.canonicalize.f32(float %x)
%canon.y = call float @llvm.canonicalize.f32(float %y)
%cmp = fcmp ugt float %canon.x, %canon.y
ret i1 %cmp
}
define i1 @canonicalize_uge_canonicalize_f32(float %x, float %y) {
; CHECK-LABEL: @canonicalize_uge_canonicalize_f32(
; CHECK-NEXT: [[CMP:%.*]] = fcmp uge float [[X:%.*]], [[Y:%.*]]
; CHECK-NEXT: ret i1 [[CMP]]
;
%canon.x = call float @llvm.canonicalize.f32(float %x)
%canon.y = call float @llvm.canonicalize.f32(float %y)
%cmp = fcmp uge float %canon.x, %canon.y
ret i1 %cmp
}
define i1 @canonicalize_ult_canonicalize_f32(float %x, float %y) {
; CHECK-LABEL: @canonicalize_ult_canonicalize_f32(
; CHECK-NEXT: [[CMP:%.*]] = fcmp ult float [[X:%.*]], [[Y:%.*]]
; CHECK-NEXT: ret i1 [[CMP]]
;
%canon.x = call float @llvm.canonicalize.f32(float %x)
%canon.y = call float @llvm.canonicalize.f32(float %y)
%cmp = fcmp ult float %canon.x, %canon.y
ret i1 %cmp
}
define i1 @canonicalize_ule_canonicalize_f32(float %x, float %y) {
; CHECK-LABEL: @canonicalize_ule_canonicalize_f32(
; CHECK-NEXT: [[CMP:%.*]] = fcmp ule float [[X:%.*]], [[Y:%.*]]
; CHECK-NEXT: ret i1 [[CMP]]
;
%canon.x = call float @llvm.canonicalize.f32(float %x)
%canon.y = call float @llvm.canonicalize.f32(float %y)
%cmp = fcmp ule float %canon.x, %canon.y
ret i1 %cmp
}
define i1 @canonicalize_uno_canonicalize_f32(float %x, float %y) {
; CHECK-LABEL: @canonicalize_uno_canonicalize_f32(
; CHECK-NEXT: [[CMP:%.*]] = fcmp uno float [[X:%.*]], [[Y:%.*]]
; CHECK-NEXT: ret i1 [[CMP]]
;
%canon.x = call float @llvm.canonicalize.f32(float %x)
%canon.y = call float @llvm.canonicalize.f32(float %y)
%cmp = fcmp uno float %canon.x, %canon.y
ret i1 %cmp
}
define i1 @canonicalize_oeq_y_f32() {
; CHECK-LABEL: @canonicalize_oeq_y_f32(
; CHECK-NEXT: [[X:%.*]] = call float @gen_f32()
; CHECK-NEXT: [[Y:%.*]] = call float @gen_f32()
; CHECK-NEXT: [[CANON_X:%.*]] = call float @llvm.canonicalize.f32(float [[X]])
; CHECK-NEXT: [[CMP:%.*]] = fcmp oeq float [[CANON_X]], [[Y]]
; CHECK-NEXT: ret i1 [[CMP]]
;
%x = call float @gen_f32()
%y = call float @gen_f32()
%canon.x = call float @llvm.canonicalize.f32(float %x)
%cmp = fcmp oeq float %canon.x, %y
ret i1 %cmp
}
define i1 @canonicalize_oeq_y_commute_f32() {
; CHECK-LABEL: @canonicalize_oeq_y_commute_f32(
; CHECK-NEXT: [[X:%.*]] = call float @gen_f32()
; CHECK-NEXT: [[Y:%.*]] = call float @gen_f32()
; CHECK-NEXT: [[CANON_Y:%.*]] = call float @llvm.canonicalize.f32(float [[Y]])
; CHECK-NEXT: [[CMP:%.*]] = fcmp oeq float [[X]], [[CANON_Y]]
; CHECK-NEXT: ret i1 [[CMP]]
;
%x = call float @gen_f32()
%y = call float @gen_f32()
%canon.y = call float @llvm.canonicalize.f32(float %y)
%cmp = fcmp oeq float %x, %canon.y
ret i1 %cmp
}
; --------------------------------------------------------------------
; (@canonicalize(x) == x) is equivalent to (x == x)
; --------------------------------------------------------------------
define i1 @canonicalize_oeq_arg_f32(float %x) {
; CHECK-LABEL: @canonicalize_oeq_arg_f32(
; CHECK-NEXT: [[CMP:%.*]] = fcmp ord float [[X:%.*]], 0.000000e+00
; CHECK-NEXT: ret i1 [[CMP]]
;
%canon.x = call float @llvm.canonicalize.f32(float %x)
%cmp = fcmp oeq float %canon.x, %x
ret i1 %cmp
}
define i1 @canonicalize_oeq_arg_f32_flags(float %x) {
; CHECK-LABEL: @canonicalize_oeq_arg_f32_flags(
; CHECK-NEXT: [[CMP:%.*]] = fcmp nsz ord float [[X:%.*]], 0.000000e+00
; CHECK-NEXT: ret i1 [[CMP]]
;
%canon.x = call float @llvm.canonicalize.f32(float %x)
%cmp = fcmp nsz oeq float %canon.x, %x
ret i1 %cmp
}
declare float @gen_f32()
declare <2 x float> @gen_v2f32()
define i1 @canonicalize_oeq_arg_f32_commute() {
; CHECK-LABEL: @canonicalize_oeq_arg_f32_commute(
; CHECK-NEXT: [[X:%.*]] = call float @gen_f32()
; CHECK-NEXT: [[CMP:%.*]] = fcmp nsz ord float [[X]], 0.000000e+00
; CHECK-NEXT: ret i1 [[CMP]]
;
%x = call float @gen_f32() ; thwart complexity-based canonicalization
%canon.x = call float @llvm.canonicalize.f32(float %x)
%cmp = fcmp nsz oeq float %x, %canon.x
ret i1 %cmp
}
define <2 x i1> @canonicalize_oeq_arg_v2f32(<2 x float> %x) {
; CHECK-LABEL: @canonicalize_oeq_arg_v2f32(
; CHECK-NEXT: [[CMP:%.*]] = fcmp ord <2 x float> [[X:%.*]], zeroinitializer
; CHECK-NEXT: ret <2 x i1> [[CMP]]
;
%canon.x = call <2 x float> @llvm.canonicalize.v2f32(<2 x float> %x)
%cmp = fcmp oeq <2 x float> %canon.x, %x
ret <2 x i1> %cmp
}
define <2 x i1> @canonicalize_oeq_arg_v2f32_commute() {
; CHECK-LABEL: @canonicalize_oeq_arg_v2f32_commute(
; CHECK-NEXT: [[X:%.*]] = call <2 x float> @gen_v2f32()
; CHECK-NEXT: [[CMP:%.*]] = fcmp ord <2 x float> [[X]], zeroinitializer
; CHECK-NEXT: ret <2 x i1> [[CMP]]
;
%x = call <2 x float> @gen_v2f32() ; thwart complexity-based canonicalization
%canon.x = call <2 x float> @llvm.canonicalize.v2f32(<2 x float> %x)
%cmp = fcmp oeq <2 x float> %x, %canon.x
ret <2 x i1> %cmp
}
define i1 @canonicalize_ueq_arg_f32(float %x) {
; CHECK-LABEL: @canonicalize_ueq_arg_f32(
; CHECK-NEXT: ret i1 true
;
%canon.x = call float @llvm.canonicalize.f32(float %x)
%cmp = fcmp ueq float %canon.x, %x
ret i1 %cmp
}
define <2 x i1> @canonicalize_ueq_arg_v2f32(<2 x float> %x) {
; CHECK-LABEL: @canonicalize_ueq_arg_v2f32(
; CHECK-NEXT: ret <2 x i1> <i1 true, i1 true>
;
%canon.x = call <2 x float> @llvm.canonicalize.v2f32(<2 x float> %x)
%cmp = fcmp ueq <2 x float> %canon.x, %x
ret <2 x i1> %cmp
}
define i1 @canonicalize_one_arg_f32(float %x) {
; CHECK-LABEL: @canonicalize_one_arg_f32(
; CHECK-NEXT: ret i1 false
;
%canon.x = call float @llvm.canonicalize.f32(float %x)
%cmp = fcmp one float %canon.x, %x
ret i1 %cmp
}
define <2 x i1> @canonicalize_one_arg_v2f32(<2 x float> %x) {
; CHECK-LABEL: @canonicalize_one_arg_v2f32(
; CHECK-NEXT: ret <2 x i1> zeroinitializer
;
%canon.x = call <2 x float> @llvm.canonicalize.v2f32(<2 x float> %x)
%cmp = fcmp one <2 x float> %canon.x, %x
ret <2 x i1> %cmp
}
define i1 @canonicalize_une_arg_f32(float %x) {
; CHECK-LABEL: @canonicalize_une_arg_f32(
; CHECK-NEXT: [[CMP:%.*]] = fcmp uno float [[X:%.*]], 0.000000e+00
; CHECK-NEXT: ret i1 [[CMP]]
;
%canon.x = call float @llvm.canonicalize.f32(float %x)
%cmp = fcmp une float %canon.x, %x
ret i1 %cmp
}
define <2 x i1> @canonicalize_une_arg_v2f32(<2 x float> %x) {
; CHECK-LABEL: @canonicalize_une_arg_v2f32(
; CHECK-NEXT: [[CMP:%.*]] = fcmp uno <2 x float> [[X:%.*]], zeroinitializer
; CHECK-NEXT: ret <2 x i1> [[CMP]]
;
%canon.x = call <2 x float> @llvm.canonicalize.v2f32(<2 x float> %x)
%cmp = fcmp une <2 x float> %canon.x, %x
ret <2 x i1> %cmp
}
define i1 @canonicalize_ogt_arg_f32(float %x) {
; CHECK-LABEL: @canonicalize_ogt_arg_f32(
; CHECK-NEXT: ret i1 false
;
%canon.x = call float @llvm.canonicalize.f32(float %x)
%cmp = fcmp ogt float %canon.x, %x
ret i1 %cmp
}
define i1 @canonicalize_oge_arg_f32(float %x) {
; CHECK-LABEL: @canonicalize_oge_arg_f32(
; CHECK-NEXT: [[CMP:%.*]] = fcmp ord float [[X:%.*]], 0.000000e+00
; CHECK-NEXT: ret i1 [[CMP]]
;
%canon.x = call float @llvm.canonicalize.f32(float %x)
%cmp = fcmp oge float %canon.x, %x
ret i1 %cmp
}
define i1 @canonicalize_olt_arg_f32(float %x) {
; CHECK-LABEL: @canonicalize_olt_arg_f32(
; CHECK-NEXT: ret i1 false
;
%canon.x = call float @llvm.canonicalize.f32(float %x)
%cmp = fcmp olt float %canon.x, %x
ret i1 %cmp
}
define i1 @canonicalize_ole_arg_f32(float %x) {
; CHECK-LABEL: @canonicalize_ole_arg_f32(
; CHECK-NEXT: [[CMP:%.*]] = fcmp ord float [[X:%.*]], 0.000000e+00
; CHECK-NEXT: ret i1 [[CMP]]
;
%canon.x = call float @llvm.canonicalize.f32(float %x)
%cmp = fcmp ole float %canon.x, %x
ret i1 %cmp
}
define i1 @canonicalize_ord_arg_f32(float %x) {
; CHECK-LABEL: @canonicalize_ord_arg_f32(
; CHECK-NEXT: [[CMP:%.*]] = fcmp ord float [[X:%.*]], 0.000000e+00
; CHECK-NEXT: ret i1 [[CMP]]
;
%canon.x = call float @llvm.canonicalize.f32(float %x)
%cmp = fcmp ord float %canon.x, %x
ret i1 %cmp
}
define i1 @canonicalize_ugt_arg_f32(float %x) {
; CHECK-LABEL: @canonicalize_ugt_arg_f32(
; CHECK-NEXT: [[CMP:%.*]] = fcmp uno float [[X:%.*]], 0.000000e+00
; CHECK-NEXT: ret i1 [[CMP]]
;
%canon.x = call float @llvm.canonicalize.f32(float %x)
%cmp = fcmp ugt float %canon.x, %x
ret i1 %cmp
}
define i1 @canonicalize_uge_arg_f32(float %x) {
; CHECK-LABEL: @canonicalize_uge_arg_f32(
; CHECK-NEXT: ret i1 true
;
%canon.x = call float @llvm.canonicalize.f32(float %x)
%cmp = fcmp uge float %canon.x, %x
ret i1 %cmp
}
define i1 @canonicalize_ult_arg_f32(float %x) {
; CHECK-LABEL: @canonicalize_ult_arg_f32(
; CHECK-NEXT: [[CMP:%.*]] = fcmp uno float [[X:%.*]], 0.000000e+00
; CHECK-NEXT: ret i1 [[CMP]]
;
%canon.x = call float @llvm.canonicalize.f32(float %x)
%cmp = fcmp ult float %canon.x, %x
ret i1 %cmp
}
define i1 @canonicalize_ule_arg_f32(float %x) {
; CHECK-LABEL: @canonicalize_ule_arg_f32(
; CHECK-NEXT: ret i1 true
;
%canon.x = call float @llvm.canonicalize.f32(float %x)
%cmp = fcmp ule float %canon.x, %x
ret i1 %cmp
}
define i1 @canonicalize_uno_arg_f32(float %x) {
; CHECK-LABEL: @canonicalize_uno_arg_f32(
; CHECK-NEXT: [[CMP:%.*]] = fcmp uno float [[X:%.*]], 0.000000e+00
; CHECK-NEXT: ret i1 [[CMP]]
;
%canon.x = call float @llvm.canonicalize.f32(float %x)
%cmp = fcmp uno float %canon.x, %x
ret i1 %cmp
}
; --------------------------------------------------------------------
; Others
; --------------------------------------------------------------------
; Regression test checking that the vector version of llvm.canonicalize works.
define <2 x i1> @vec_canonicalize_with_fpclass(<2 x float> %x) {
; CHECK-LABEL: @vec_canonicalize_with_fpclass(
; CHECK-NEXT: ret <2 x i1> zeroinitializer
;
%canon = call <2 x float> @llvm.canonicalize.v2f32(<2 x float> %x)
%fpclass = call <2 x i1> @llvm.is.fpclass.v2f32(<2 x float> %canon, i32 1)
ret <2 x i1> %fpclass
}
declare float @llvm.canonicalize.f32(float)
declare <2 x float> @llvm.canonicalize.v2f32(<2 x float>)
declare <2 x i1> @llvm.is.fpclass.v2f32(<2 x float>, i32 immarg)