; NOTE: Assertions have been autogenerated by utils/update_llc_test_checks.py UTC_ARGS: --version 6 ; RUN: llc -mtriple=amdgcn -mcpu=tahiti < %s | FileCheck -check-prefix=GFX6 %s ; RUN: llc -mtriple=amdgcn -mcpu=gfx1100 < %s | FileCheck -check-prefix=GFX11 %s declare i32 @llvm.ctlz.i32(i32, i1) declare i64 @llvm.ctlz.i64(i64, i1) declare i32 @llvm.amdgcn.sffbh.i32(i32) ; Test that ctls(x) is lowered to umin(ffbh_i32(x), bitwidth) - 1 ; ctls is formed by the DAG combiner from: ctlz(x ^ ashr(x, 31)) - 1 define i32 @ctls_i32(i32 %x) { ; GFX6-LABEL: ctls_i32: ; GFX6: ; %bb.0: ; GFX6-NEXT: s_waitcnt vmcnt(0) expcnt(0) lgkmcnt(0) ; GFX6-NEXT: v_ffbh_i32_e32 v0, v0 ; GFX6-NEXT: v_min_u32_e32 v0, 32, v0 ; GFX6-NEXT: v_add_i32_e32 v0, vcc, -1, v0 ; GFX6-NEXT: s_setpc_b64 s[30:31] ; ; GFX11-LABEL: ctls_i32: ; GFX11: ; %bb.0: ; GFX11-NEXT: s_waitcnt vmcnt(0) expcnt(0) lgkmcnt(0) ; GFX11-NEXT: v_cls_i32_e32 v0, v0 ; GFX11-NEXT: s_delay_alu instid0(VALU_DEP_1) | instskip(NEXT) | instid1(VALU_DEP_1) ; GFX11-NEXT: v_min_u32_e32 v0, 32, v0 ; GFX11-NEXT: v_add_nc_u32_e32 v0, -1, v0 ; GFX11-NEXT: s_setpc_b64 s[30:31] %a = ashr i32 %x, 31 %b = xor i32 %x, %a %c = call i32 @llvm.ctlz.i32(i32 %b, i1 false) %d = sub i32 %c, 1 ret i32 %d } define i32 @ctls_i32_known_positive(i32 %x) { ; GFX6-LABEL: ctls_i32_known_positive: ; GFX6: ; %bb.0: ; GFX6-NEXT: s_waitcnt vmcnt(0) expcnt(0) lgkmcnt(0) ; GFX6-NEXT: v_and_b32_e32 v0, 0x7fffffff, v0 ; GFX6-NEXT: v_ffbh_i32_e32 v0, v0 ; GFX6-NEXT: v_min_u32_e32 v0, 32, v0 ; GFX6-NEXT: v_add_i32_e32 v0, vcc, -1, v0 ; GFX6-NEXT: s_setpc_b64 s[30:31] ; ; GFX11-LABEL: ctls_i32_known_positive: ; GFX11: ; %bb.0: ; GFX11-NEXT: s_waitcnt vmcnt(0) expcnt(0) lgkmcnt(0) ; GFX11-NEXT: v_and_b32_e32 v0, 0x7fffffff, v0 ; GFX11-NEXT: s_delay_alu instid0(VALU_DEP_1) | instskip(NEXT) | instid1(VALU_DEP_1) ; GFX11-NEXT: v_cls_i32_e32 v0, v0 ; GFX11-NEXT: v_min_u32_e32 v0, 32, v0 ; GFX11-NEXT: s_delay_alu instid0(VALU_DEP_1) ; GFX11-NEXT: v_add_nc_u32_e32 v0, -1, v0 ; GFX11-NEXT: s_setpc_b64 s[30:31] %pos = and i32 %x, 2147483647 %a = ashr i32 %pos, 31 %b = xor i32 %pos, %a %c = call i32 @llvm.ctlz.i32(i32 %b, i1 false) %d = sub i32 %c, 1 ret i32 %d } ; sub(ctlz(xor(x, sra(x, 31))), 1) -> ctls(x) define i32 @ctls_i32_xor_commuted(i32 %x) { ; GFX6-LABEL: ctls_i32_xor_commuted: ; GFX6: ; %bb.0: ; GFX6-NEXT: s_waitcnt vmcnt(0) expcnt(0) lgkmcnt(0) ; GFX6-NEXT: v_ffbh_i32_e32 v0, v0 ; GFX6-NEXT: v_min_u32_e32 v0, 32, v0 ; GFX6-NEXT: v_add_i32_e32 v0, vcc, -1, v0 ; GFX6-NEXT: s_setpc_b64 s[30:31] ; ; GFX11-LABEL: ctls_i32_xor_commuted: ; GFX11: ; %bb.0: ; GFX11-NEXT: s_waitcnt vmcnt(0) expcnt(0) lgkmcnt(0) ; GFX11-NEXT: v_cls_i32_e32 v0, v0 ; GFX11-NEXT: s_delay_alu instid0(VALU_DEP_1) | instskip(NEXT) | instid1(VALU_DEP_1) ; GFX11-NEXT: v_min_u32_e32 v0, 32, v0 ; GFX11-NEXT: v_add_nc_u32_e32 v0, -1, v0 ; GFX11-NEXT: s_setpc_b64 s[30:31] %a = ashr i32 %x, 31 %b = xor i32 %a, %x ; note: reversed order compared to ctls_i32 %c = call i32 @llvm.ctlz.i32(i32 %b, i1 false) %d = sub i32 %c, 1 ret i32 %d } define i32 @ctls_i32_zero_undef(i32 %x) { ; GFX6-LABEL: ctls_i32_zero_undef: ; GFX6: ; %bb.0: ; GFX6-NEXT: s_waitcnt vmcnt(0) expcnt(0) lgkmcnt(0) ; GFX6-NEXT: v_ffbh_i32_e32 v0, v0 ; GFX6-NEXT: v_min_u32_e32 v0, 32, v0 ; GFX6-NEXT: v_add_i32_e32 v0, vcc, -1, v0 ; GFX6-NEXT: s_setpc_b64 s[30:31] ; ; GFX11-LABEL: ctls_i32_zero_undef: ; GFX11: ; %bb.0: ; GFX11-NEXT: s_waitcnt vmcnt(0) expcnt(0) lgkmcnt(0) ; GFX11-NEXT: v_cls_i32_e32 v0, v0 ; GFX11-NEXT: s_delay_alu instid0(VALU_DEP_1) | instskip(NEXT) | instid1(VALU_DEP_1) ; GFX11-NEXT: v_min_u32_e32 v0, 32, v0 ; GFX11-NEXT: v_add_nc_u32_e32 v0, -1, v0 ; GFX11-NEXT: s_setpc_b64 s[30:31] %a = ashr i32 %x, 31 %b = xor i32 %x, %a %c = call i32 @llvm.ctlz.i32(i32 %b, i1 true) ; zero_undef = true %d = sub i32 %c, 1 ret i32 %d } ; umin(ffbh_i32(x), 32) -> ffbh_i32(x). define i32 @ctls_i32_known_mixed_bits(i32 %x) { ; GFX6-LABEL: ctls_i32_known_mixed_bits: ; GFX6: ; %bb.0: ; GFX6-NEXT: s_waitcnt vmcnt(0) expcnt(0) lgkmcnt(0) ; GFX6-NEXT: v_or_b32_e32 v0, 1, v0 ; GFX6-NEXT: v_and_b32_e32 v0, 0x7fffffff, v0 ; GFX6-NEXT: v_ffbh_i32_e32 v0, v0 ; GFX6-NEXT: v_add_i32_e32 v0, vcc, -1, v0 ; GFX6-NEXT: s_setpc_b64 s[30:31] ; ; GFX11-LABEL: ctls_i32_known_mixed_bits: ; GFX11: ; %bb.0: ; GFX11-NEXT: s_waitcnt vmcnt(0) expcnt(0) lgkmcnt(0) ; GFX11-NEXT: v_or_b32_e32 v0, 1, v0 ; GFX11-NEXT: s_delay_alu instid0(VALU_DEP_1) | instskip(NEXT) | instid1(VALU_DEP_1) ; GFX11-NEXT: v_and_b32_e32 v0, 0x7fffffff, v0 ; GFX11-NEXT: v_cls_i32_e32 v0, v0 ; GFX11-NEXT: s_delay_alu instid0(VALU_DEP_1) ; GFX11-NEXT: v_add_nc_u32_e32 v0, -1, v0 ; GFX11-NEXT: s_setpc_b64 s[30:31] ; Force bit 31 = 0 and bit 0 = 1, so value is neither all-0s nor all-1s %cleared = and i32 %x, 2147483647 ; clear bit 31 %mixed = or i32 %cleared, 1 ; set bit 0 %a = ashr i32 %mixed, 31 %b = xor i32 %mixed, %a %c = call i32 @llvm.ctlz.i32(i32 %b, i1 false) %d = sub i32 %c, 1 ret i32 %d } ; test for i64 CTLS. define i32 @ctls_i64(i64 %x) { ; GFX6-LABEL: ctls_i64: ; GFX6: ; %bb.0: ; GFX6-NEXT: s_waitcnt vmcnt(0) expcnt(0) lgkmcnt(0) ; GFX6-NEXT: v_ashrrev_i32_e32 v2, 31, v1 ; GFX6-NEXT: v_xor_b32_e32 v0, v0, v2 ; GFX6-NEXT: v_ffbh_u32_e32 v0, v0 ; GFX6-NEXT: v_xor_b32_e32 v1, v1, v2 ; GFX6-NEXT: v_min_u32_e32 v0, 0xffffffdf, v0 ; GFX6-NEXT: v_add_i32_e32 v0, vcc, 32, v0 ; GFX6-NEXT: v_ffbh_u32_e32 v1, v1 ; GFX6-NEXT: v_min3_u32 v0, v0, v1, 64 ; GFX6-NEXT: v_add_i32_e32 v0, vcc, -1, v0 ; GFX6-NEXT: s_setpc_b64 s[30:31] ; ; GFX11-LABEL: ctls_i64: ; GFX11: ; %bb.0: ; GFX11-NEXT: s_waitcnt vmcnt(0) expcnt(0) lgkmcnt(0) ; GFX11-NEXT: v_ashrrev_i32_e32 v2, 31, v1 ; GFX11-NEXT: s_delay_alu instid0(VALU_DEP_1) | instskip(SKIP_1) | instid1(VALU_DEP_2) ; GFX11-NEXT: v_xor_b32_e32 v0, v0, v2 ; GFX11-NEXT: v_xor_b32_e32 v1, v1, v2 ; GFX11-NEXT: v_clz_i32_u32_e32 v0, v0 ; GFX11-NEXT: s_delay_alu instid0(VALU_DEP_2) | instskip(NEXT) | instid1(VALU_DEP_2) ; GFX11-NEXT: v_clz_i32_u32_e32 v1, v1 ; GFX11-NEXT: v_add_nc_u32_e64 v0, v0, 32 clamp ; GFX11-NEXT: s_delay_alu instid0(VALU_DEP_1) | instskip(NEXT) | instid1(VALU_DEP_1) ; GFX11-NEXT: v_min3_u32 v0, v0, v1, 64 ; GFX11-NEXT: v_add_nc_u32_e32 v0, -1, v0 ; GFX11-NEXT: s_setpc_b64 s[30:31] %a = ashr i64 %x, 63 %b = xor i64 %x, %a %c = call i64 @llvm.ctlz.i64(i64 %b, i1 false) %d = sub i64 %c, 1 %e = trunc i64 %d to i32 ret i32 %e } ; i16 CTLS via the sub(ctlz(xor(x, sra(x, 15))), 1) pattern. declare i16 @llvm.ctlz.i16(i16, i1) define i16 @ctls_i16(i16 %x) { ; GFX6-LABEL: ctls_i16: ; GFX6: ; %bb.0: ; GFX6-NEXT: s_waitcnt vmcnt(0) expcnt(0) lgkmcnt(0) ; GFX6-NEXT: v_bfe_i32 v0, v0, 0, 16 ; GFX6-NEXT: v_ffbh_i32_e32 v0, v0 ; GFX6-NEXT: v_min_u32_e32 v0, 32, v0 ; GFX6-NEXT: v_subrev_i32_e32 v0, vcc, 17, v0 ; GFX6-NEXT: s_setpc_b64 s[30:31] ; ; GFX11-LABEL: ctls_i16: ; GFX11: ; %bb.0: ; GFX11-NEXT: s_waitcnt vmcnt(0) expcnt(0) lgkmcnt(0) ; GFX11-NEXT: v_ashrrev_i16 v0.h, 15, v0.l ; GFX11-NEXT: s_delay_alu instid0(VALU_DEP_1) | instskip(SKIP_1) | instid1(VALU_DEP_1) ; GFX11-NEXT: v_xor_b16 v0.l, v0.l, v0.h ; GFX11-NEXT: v_mov_b16_e32 v0.h, 0 ; GFX11-NEXT: v_clz_i32_u32_e32 v0, v0 ; GFX11-NEXT: s_delay_alu instid0(VALU_DEP_1) | instskip(NEXT) | instid1(VALU_DEP_1) ; GFX11-NEXT: v_min_u32_e32 v0, 32, v0 ; GFX11-NEXT: v_add_nc_u32_e32 v0, -16, v0 ; GFX11-NEXT: s_delay_alu instid0(VALU_DEP_1) ; GFX11-NEXT: v_add_nc_u16 v0.l, v0.l, -1 ; GFX11-NEXT: s_setpc_b64 s[30:31] %a = ashr i16 %x, 15 %b = xor i16 %x, %a %c = call i16 @llvm.ctlz.i16(i16 %b, i1 false) %d = sub i16 %c, 1 ret i16 %d } ; uniform input should use scalar sffbh. define amdgpu_ps i32 @ctls_i32_salu(i32 inreg %x) { ; GFX6-LABEL: ctls_i32_salu: ; GFX6: ; %bb.0: ; GFX6-NEXT: s_flbit_i32 s0, s0 ; GFX6-NEXT: s_min_u32 s0, s0, 32 ; GFX6-NEXT: s_add_i32 s0, s0, -1 ; GFX6-NEXT: ; return to shader part epilog ; ; GFX11-LABEL: ctls_i32_salu: ; GFX11: ; %bb.0: ; GFX11-NEXT: s_cls_i32 s0, s0 ; GFX11-NEXT: s_delay_alu instid0(SALU_CYCLE_1) | instskip(NEXT) | instid1(SALU_CYCLE_1) ; GFX11-NEXT: s_min_u32 s0, s0, 32 ; GFX11-NEXT: s_add_i32 s0, s0, -1 ; GFX11-NEXT: ; return to shader part epilog %a = ashr i32 %x, 31 %b = xor i32 %x, %a %c = call i32 @llvm.ctlz.i32(i32 %b, i1 false) %d = sub i32 %c, 1 ret i32 %d } define <2 x i32> @ctls_v2i32(<2 x i32> %x) { ; GFX6-LABEL: ctls_v2i32: ; GFX6: ; %bb.0: ; GFX6-NEXT: s_waitcnt vmcnt(0) expcnt(0) lgkmcnt(0) ; GFX6-NEXT: v_ashrrev_i32_e32 v2, 31, v1 ; GFX6-NEXT: v_ashrrev_i32_e32 v3, 31, v0 ; GFX6-NEXT: v_xor_b32_e32 v1, v1, v2 ; GFX6-NEXT: v_xor_b32_e32 v0, v0, v3 ; GFX6-NEXT: v_ffbh_u32_e32 v1, v1 ; GFX6-NEXT: v_ffbh_u32_e32 v0, v0 ; GFX6-NEXT: v_min_u32_e32 v1, 32, v1 ; GFX6-NEXT: v_min_u32_e32 v0, 32, v0 ; GFX6-NEXT: v_add_i32_e32 v0, vcc, -1, v0 ; GFX6-NEXT: v_add_i32_e32 v1, vcc, -1, v1 ; GFX6-NEXT: s_setpc_b64 s[30:31] ; ; GFX11-LABEL: ctls_v2i32: ; GFX11: ; %bb.0: ; GFX11-NEXT: s_waitcnt vmcnt(0) expcnt(0) lgkmcnt(0) ; GFX11-NEXT: v_ashrrev_i32_e32 v2, 31, v0 ; GFX11-NEXT: v_ashrrev_i32_e32 v3, 31, v1 ; GFX11-NEXT: s_delay_alu instid0(VALU_DEP_2) | instskip(NEXT) | instid1(VALU_DEP_2) ; GFX11-NEXT: v_xor_b32_e32 v0, v0, v2 ; GFX11-NEXT: v_xor_b32_e32 v1, v1, v3 ; GFX11-NEXT: s_delay_alu instid0(VALU_DEP_2) | instskip(NEXT) | instid1(VALU_DEP_2) ; GFX11-NEXT: v_clz_i32_u32_e32 v0, v0 ; GFX11-NEXT: v_clz_i32_u32_e32 v1, v1 ; GFX11-NEXT: s_delay_alu instid0(VALU_DEP_2) | instskip(NEXT) | instid1(VALU_DEP_2) ; GFX11-NEXT: v_min_u32_e32 v0, 32, v0 ; GFX11-NEXT: v_min_u32_e32 v1, 32, v1 ; GFX11-NEXT: s_delay_alu instid0(VALU_DEP_2) | instskip(NEXT) | instid1(VALU_DEP_2) ; GFX11-NEXT: v_add_nc_u32_e32 v0, -1, v0 ; GFX11-NEXT: v_add_nc_u32_e32 v1, -1, v1 ; GFX11-NEXT: s_setpc_b64 s[30:31] %a = ashr <2 x i32> %x, %b = xor <2 x i32> %x, %a %c = call <2 x i32> @llvm.ctlz.v2i32(<2 x i32> %b, i1 false) %d = sub <2 x i32> %c, ret <2 x i32> %d } declare <2 x i32> @llvm.ctlz.v2i32(<2 x i32>, i1) declare <4 x i32> @llvm.ctlz.v4i32(<4 x i32>, i1) define <4 x i32> @ctls_v4i32(<4 x i32> %x) { ; GFX6-LABEL: ctls_v4i32: ; GFX6: ; %bb.0: ; GFX6-NEXT: s_waitcnt vmcnt(0) expcnt(0) lgkmcnt(0) ; GFX6-NEXT: v_ffbh_i32_e32 v0, v0 ; GFX6-NEXT: v_ffbh_i32_e32 v1, v1 ; GFX6-NEXT: v_ffbh_i32_e32 v2, v2 ; GFX6-NEXT: v_ffbh_i32_e32 v3, v3 ; GFX6-NEXT: v_min_u32_e32 v0, 32, v0 ; GFX6-NEXT: v_min_u32_e32 v1, 32, v1 ; GFX6-NEXT: v_min_u32_e32 v2, 32, v2 ; GFX6-NEXT: v_min_u32_e32 v3, 32, v3 ; GFX6-NEXT: v_add_i32_e32 v0, vcc, -1, v0 ; GFX6-NEXT: v_add_i32_e32 v1, vcc, -1, v1 ; GFX6-NEXT: v_add_i32_e32 v2, vcc, -1, v2 ; GFX6-NEXT: v_add_i32_e32 v3, vcc, -1, v3 ; GFX6-NEXT: s_setpc_b64 s[30:31] ; ; GFX11-LABEL: ctls_v4i32: ; GFX11: ; %bb.0: ; GFX11-NEXT: s_waitcnt vmcnt(0) expcnt(0) lgkmcnt(0) ; GFX11-NEXT: v_cls_i32_e32 v0, v0 ; GFX11-NEXT: v_cls_i32_e32 v1, v1 ; GFX11-NEXT: v_cls_i32_e32 v2, v2 ; GFX11-NEXT: v_cls_i32_e32 v3, v3 ; GFX11-NEXT: s_delay_alu instid0(VALU_DEP_4) | instskip(NEXT) | instid1(VALU_DEP_4) ; GFX11-NEXT: v_min_u32_e32 v0, 32, v0 ; GFX11-NEXT: v_min_u32_e32 v1, 32, v1 ; GFX11-NEXT: s_delay_alu instid0(VALU_DEP_4) | instskip(NEXT) | instid1(VALU_DEP_4) ; GFX11-NEXT: v_min_u32_e32 v2, 32, v2 ; GFX11-NEXT: v_min_u32_e32 v3, 32, v3 ; GFX11-NEXT: s_delay_alu instid0(VALU_DEP_4) | instskip(NEXT) | instid1(VALU_DEP_4) ; GFX11-NEXT: v_add_nc_u32_e32 v0, -1, v0 ; GFX11-NEXT: v_add_nc_u32_e32 v1, -1, v1 ; GFX11-NEXT: s_delay_alu instid0(VALU_DEP_4) | instskip(NEXT) | instid1(VALU_DEP_4) ; GFX11-NEXT: v_add_nc_u32_e32 v2, -1, v2 ; GFX11-NEXT: v_add_nc_u32_e32 v3, -1, v3 ; GFX11-NEXT: s_setpc_b64 s[30:31] %a = ashr <4 x i32> %x, %b = xor <4 x i32> %x, %a %c = call <4 x i32> @llvm.ctlz.v4i32(<4 x i32> %b, i1 false) %d = sub <4 x i32> %c, ret <4 x i32> %d } ; umin should be folded away per element per element. define <2 x i32> @ctls_v2i32_known_mixed_bits(<2 x i32> %x) { ; GFX6-LABEL: ctls_v2i32_known_mixed_bits: ; GFX6: ; %bb.0: ; GFX6-NEXT: s_waitcnt vmcnt(0) expcnt(0) lgkmcnt(0) ; GFX6-NEXT: v_or_b32_e32 v1, 1, v1 ; GFX6-NEXT: v_or_b32_e32 v0, 1, v0 ; GFX6-NEXT: v_and_b32_e32 v1, 0x7fffffff, v1 ; GFX6-NEXT: v_and_b32_e32 v0, 0x7fffffff, v0 ; GFX6-NEXT: v_ffbh_u32_e32 v1, v1 ; GFX6-NEXT: v_ffbh_u32_e32 v0, v0 ; GFX6-NEXT: v_add_i32_e32 v0, vcc, -1, v0 ; GFX6-NEXT: v_add_i32_e32 v1, vcc, -1, v1 ; GFX6-NEXT: s_setpc_b64 s[30:31] ; ; GFX11-LABEL: ctls_v2i32_known_mixed_bits: ; GFX11: ; %bb.0: ; GFX11-NEXT: s_waitcnt vmcnt(0) expcnt(0) lgkmcnt(0) ; GFX11-NEXT: v_or_b32_e32 v0, 1, v0 ; GFX11-NEXT: v_or_b32_e32 v1, 1, v1 ; GFX11-NEXT: s_delay_alu instid0(VALU_DEP_2) | instskip(NEXT) | instid1(VALU_DEP_2) ; GFX11-NEXT: v_and_b32_e32 v0, 0x7fffffff, v0 ; GFX11-NEXT: v_and_b32_e32 v1, 0x7fffffff, v1 ; GFX11-NEXT: s_delay_alu instid0(VALU_DEP_2) | instskip(NEXT) | instid1(VALU_DEP_2) ; GFX11-NEXT: v_clz_i32_u32_e32 v0, v0 ; GFX11-NEXT: v_clz_i32_u32_e32 v1, v1 ; GFX11-NEXT: s_delay_alu instid0(VALU_DEP_2) | instskip(NEXT) | instid1(VALU_DEP_2) ; GFX11-NEXT: v_add_nc_u32_e32 v0, -1, v0 ; GFX11-NEXT: v_add_nc_u32_e32 v1, -1, v1 ; GFX11-NEXT: s_setpc_b64 s[30:31] %cleared = and <2 x i32> %x, %mixed = or <2 x i32> %cleared, %a = ashr <2 x i32> %mixed, %b = xor <2 x i32> %mixed, %a %c = call <2 x i32> @llvm.ctlz.v2i32(<2 x i32> %b, i1 false) %d = sub <2 x i32> %c, ret <2 x i32> %d } ; Vector with ctlz_zero_undef. define <2 x i32> @ctls_v2i32_zero_undef(<2 x i32> %x) { ; GFX6-LABEL: ctls_v2i32_zero_undef: ; GFX6: ; %bb.0: ; GFX6-NEXT: s_waitcnt vmcnt(0) expcnt(0) lgkmcnt(0) ; GFX6-NEXT: v_ashrrev_i32_e32 v2, 31, v1 ; GFX6-NEXT: v_ashrrev_i32_e32 v3, 31, v0 ; GFX6-NEXT: v_xor_b32_e32 v1, v1, v2 ; GFX6-NEXT: v_xor_b32_e32 v0, v0, v3 ; GFX6-NEXT: v_ffbh_u32_e32 v1, v1 ; GFX6-NEXT: v_ffbh_u32_e32 v0, v0 ; GFX6-NEXT: v_add_i32_e32 v0, vcc, -1, v0 ; GFX6-NEXT: v_add_i32_e32 v1, vcc, -1, v1 ; GFX6-NEXT: s_setpc_b64 s[30:31] ; ; GFX11-LABEL: ctls_v2i32_zero_undef: ; GFX11: ; %bb.0: ; GFX11-NEXT: s_waitcnt vmcnt(0) expcnt(0) lgkmcnt(0) ; GFX11-NEXT: v_ashrrev_i32_e32 v2, 31, v0 ; GFX11-NEXT: v_ashrrev_i32_e32 v3, 31, v1 ; GFX11-NEXT: s_delay_alu instid0(VALU_DEP_2) | instskip(NEXT) | instid1(VALU_DEP_2) ; GFX11-NEXT: v_xor_b32_e32 v0, v0, v2 ; GFX11-NEXT: v_xor_b32_e32 v1, v1, v3 ; GFX11-NEXT: s_delay_alu instid0(VALU_DEP_2) | instskip(NEXT) | instid1(VALU_DEP_2) ; GFX11-NEXT: v_clz_i32_u32_e32 v0, v0 ; GFX11-NEXT: v_clz_i32_u32_e32 v1, v1 ; GFX11-NEXT: s_delay_alu instid0(VALU_DEP_2) | instskip(NEXT) | instid1(VALU_DEP_2) ; GFX11-NEXT: v_add_nc_u32_e32 v0, -1, v0 ; GFX11-NEXT: v_add_nc_u32_e32 v1, -1, v1 ; GFX11-NEXT: s_setpc_b64 s[30:31] %a = ashr <2 x i32> %x, %b = xor <2 x i32> %x, %a %c = call <2 x i32> @llvm.ctlz.v2i32(<2 x i32> %b, i1 true) %d = sub <2 x i32> %c, ret <2 x i32> %d } ; Vector commuted XOR operands. define <2 x i32> @ctls_v2i32_xor_commuted(<2 x i32> %x) { ; GFX6-LABEL: ctls_v2i32_xor_commuted: ; GFX6: ; %bb.0: ; GFX6-NEXT: s_waitcnt vmcnt(0) expcnt(0) lgkmcnt(0) ; GFX6-NEXT: v_ashrrev_i32_e32 v2, 31, v1 ; GFX6-NEXT: v_ashrrev_i32_e32 v3, 31, v0 ; GFX6-NEXT: v_xor_b32_e32 v1, v2, v1 ; GFX6-NEXT: v_xor_b32_e32 v0, v3, v0 ; GFX6-NEXT: v_ffbh_u32_e32 v1, v1 ; GFX6-NEXT: v_ffbh_u32_e32 v0, v0 ; GFX6-NEXT: v_min_u32_e32 v1, 32, v1 ; GFX6-NEXT: v_min_u32_e32 v0, 32, v0 ; GFX6-NEXT: v_add_i32_e32 v0, vcc, -1, v0 ; GFX6-NEXT: v_add_i32_e32 v1, vcc, -1, v1 ; GFX6-NEXT: s_setpc_b64 s[30:31] ; ; GFX11-LABEL: ctls_v2i32_xor_commuted: ; GFX11: ; %bb.0: ; GFX11-NEXT: s_waitcnt vmcnt(0) expcnt(0) lgkmcnt(0) ; GFX11-NEXT: v_ashrrev_i32_e32 v2, 31, v0 ; GFX11-NEXT: v_ashrrev_i32_e32 v3, 31, v1 ; GFX11-NEXT: s_delay_alu instid0(VALU_DEP_2) | instskip(NEXT) | instid1(VALU_DEP_2) ; GFX11-NEXT: v_xor_b32_e32 v0, v2, v0 ; GFX11-NEXT: v_xor_b32_e32 v1, v3, v1 ; GFX11-NEXT: s_delay_alu instid0(VALU_DEP_2) | instskip(NEXT) | instid1(VALU_DEP_2) ; GFX11-NEXT: v_clz_i32_u32_e32 v0, v0 ; GFX11-NEXT: v_clz_i32_u32_e32 v1, v1 ; GFX11-NEXT: s_delay_alu instid0(VALU_DEP_2) | instskip(NEXT) | instid1(VALU_DEP_2) ; GFX11-NEXT: v_min_u32_e32 v0, 32, v0 ; GFX11-NEXT: v_min_u32_e32 v1, 32, v1 ; GFX11-NEXT: s_delay_alu instid0(VALU_DEP_2) | instskip(NEXT) | instid1(VALU_DEP_2) ; GFX11-NEXT: v_add_nc_u32_e32 v0, -1, v0 ; GFX11-NEXT: v_add_nc_u32_e32 v1, -1, v1 ; GFX11-NEXT: s_setpc_b64 s[30:31] %a = ashr <2 x i32> %x, %b = xor <2 x i32> %a, %x %c = call <2 x i32> @llvm.ctlz.v2i32(<2 x i32> %b, i1 false) %d = sub <2 x i32> %c, ret <2 x i32> %d } ; Vector known positive: umin should NOT be folded. define <2 x i32> @ctls_v2i32_known_positive(<2 x i32> %x) { ; GFX6-LABEL: ctls_v2i32_known_positive: ; GFX6: ; %bb.0: ; GFX6-NEXT: s_waitcnt vmcnt(0) expcnt(0) lgkmcnt(0) ; GFX6-NEXT: v_and_b32_e32 v1, 0x7fffffff, v1 ; GFX6-NEXT: v_and_b32_e32 v0, 0x7fffffff, v0 ; GFX6-NEXT: v_ffbh_u32_e32 v1, v1 ; GFX6-NEXT: v_ffbh_u32_e32 v0, v0 ; GFX6-NEXT: v_min_u32_e32 v1, 32, v1 ; GFX6-NEXT: v_min_u32_e32 v0, 32, v0 ; GFX6-NEXT: v_add_i32_e32 v0, vcc, -1, v0 ; GFX6-NEXT: v_add_i32_e32 v1, vcc, -1, v1 ; GFX6-NEXT: s_setpc_b64 s[30:31] ; ; GFX11-LABEL: ctls_v2i32_known_positive: ; GFX11: ; %bb.0: ; GFX11-NEXT: s_waitcnt vmcnt(0) expcnt(0) lgkmcnt(0) ; GFX11-NEXT: v_and_b32_e32 v0, 0x7fffffff, v0 ; GFX11-NEXT: v_and_b32_e32 v1, 0x7fffffff, v1 ; GFX11-NEXT: s_delay_alu instid0(VALU_DEP_2) | instskip(NEXT) | instid1(VALU_DEP_2) ; GFX11-NEXT: v_clz_i32_u32_e32 v0, v0 ; GFX11-NEXT: v_clz_i32_u32_e32 v1, v1 ; GFX11-NEXT: s_delay_alu instid0(VALU_DEP_2) | instskip(NEXT) | instid1(VALU_DEP_2) ; GFX11-NEXT: v_min_u32_e32 v0, 32, v0 ; GFX11-NEXT: v_min_u32_e32 v1, 32, v1 ; GFX11-NEXT: s_delay_alu instid0(VALU_DEP_2) | instskip(NEXT) | instid1(VALU_DEP_2) ; GFX11-NEXT: v_add_nc_u32_e32 v0, -1, v0 ; GFX11-NEXT: v_add_nc_u32_e32 v1, -1, v1 ; GFX11-NEXT: s_setpc_b64 s[30:31] %pos = and <2 x i32> %x, %a = ashr <2 x i32> %pos, %b = xor <2 x i32> %pos, %a %c = call <2 x i32> @llvm.ctlz.v2i32(<2 x i32> %b, i1 false) %d = sub <2 x i32> %c, ret <2 x i32> %d } ; @llvm.amdgcn.sffbh must still produce raw hardware result. define i32 @sffbh_intrinsic(i32 %x) { ; GFX6-LABEL: sffbh_intrinsic: ; GFX6: ; %bb.0: ; GFX6-NEXT: s_waitcnt vmcnt(0) expcnt(0) lgkmcnt(0) ; GFX6-NEXT: v_ffbh_i32_e32 v0, v0 ; GFX6-NEXT: s_setpc_b64 s[30:31] ; ; GFX11-LABEL: sffbh_intrinsic: ; GFX11: ; %bb.0: ; GFX11-NEXT: s_waitcnt vmcnt(0) expcnt(0) lgkmcnt(0) ; GFX11-NEXT: v_cls_i32_e32 v0, v0 ; GFX11-NEXT: s_setpc_b64 s[30:31] %r = call i32 @llvm.amdgcn.sffbh.i32(i32 %x) ret i32 %r } ; sitofp i64 to f32 uses sffbh(Hi)-1, not CTLS. define float @sitofp_i64_to_f32(i64 %x) { ; GFX6-LABEL: sitofp_i64_to_f32: ; GFX6: ; %bb.0: ; GFX6-NEXT: s_waitcnt vmcnt(0) expcnt(0) lgkmcnt(0) ; GFX6-NEXT: v_xor_b32_e32 v2, v0, v1 ; GFX6-NEXT: v_ashrrev_i32_e32 v2, 31, v2 ; GFX6-NEXT: v_ffbh_i32_e32 v3, v1 ; GFX6-NEXT: v_add_i32_e32 v2, vcc, 32, v2 ; GFX6-NEXT: v_add_i32_e32 v3, vcc, -1, v3 ; GFX6-NEXT: v_min_u32_e32 v2, v3, v2 ; GFX6-NEXT: v_lshl_b64 v[0:1], v[0:1], v2 ; GFX6-NEXT: v_min_u32_e32 v0, 1, v0 ; GFX6-NEXT: v_or_b32_e32 v0, v1, v0 ; GFX6-NEXT: v_cvt_f32_i32_e32 v0, v0 ; GFX6-NEXT: v_sub_i32_e32 v1, vcc, 32, v2 ; GFX6-NEXT: v_ldexp_f32_e32 v0, v0, v1 ; GFX6-NEXT: s_setpc_b64 s[30:31] ; ; GFX11-LABEL: sitofp_i64_to_f32: ; GFX11: ; %bb.0: ; GFX11-NEXT: s_waitcnt vmcnt(0) expcnt(0) lgkmcnt(0) ; GFX11-NEXT: v_xor_b32_e32 v2, v0, v1 ; GFX11-NEXT: v_cls_i32_e32 v3, v1 ; GFX11-NEXT: s_delay_alu instid0(VALU_DEP_2) | instskip(NEXT) | instid1(VALU_DEP_2) ; GFX11-NEXT: v_ashrrev_i32_e32 v2, 31, v2 ; GFX11-NEXT: v_add_nc_u32_e32 v3, -1, v3 ; GFX11-NEXT: s_delay_alu instid0(VALU_DEP_2) | instskip(NEXT) | instid1(VALU_DEP_1) ; GFX11-NEXT: v_add_nc_u32_e32 v2, 32, v2 ; GFX11-NEXT: v_min_u32_e32 v2, v3, v2 ; GFX11-NEXT: s_delay_alu instid0(VALU_DEP_1) | instskip(NEXT) | instid1(VALU_DEP_1) ; GFX11-NEXT: v_lshlrev_b64 v[0:1], v2, v[0:1] ; GFX11-NEXT: v_min_u32_e32 v0, 1, v0 ; GFX11-NEXT: s_delay_alu instid0(VALU_DEP_1) | instskip(SKIP_1) | instid1(VALU_DEP_2) ; GFX11-NEXT: v_or_b32_e32 v0, v1, v0 ; GFX11-NEXT: v_sub_nc_u32_e32 v1, 32, v2 ; GFX11-NEXT: v_cvt_f32_i32_e32 v0, v0 ; GFX11-NEXT: s_delay_alu instid0(VALU_DEP_1) ; GFX11-NEXT: v_ldexp_f32 v0, v0, v1 ; GFX11-NEXT: s_setpc_b64 s[30:31] %r = sitofp i64 %x to float ret float %r } ; Negative tests: define i32 @no_ctls_wrong_shift(i32 %x) { ; GFX6-LABEL: no_ctls_wrong_shift: ; GFX6: ; %bb.0: ; GFX6-NEXT: s_waitcnt vmcnt(0) expcnt(0) lgkmcnt(0) ; GFX6-NEXT: v_ashrrev_i32_e32 v1, 30, v0 ; GFX6-NEXT: v_xor_b32_e32 v0, v0, v1 ; GFX6-NEXT: v_ffbh_u32_e32 v0, v0 ; GFX6-NEXT: v_min_u32_e32 v0, 32, v0 ; GFX6-NEXT: v_add_i32_e32 v0, vcc, -1, v0 ; GFX6-NEXT: s_setpc_b64 s[30:31] ; ; GFX11-LABEL: no_ctls_wrong_shift: ; GFX11: ; %bb.0: ; GFX11-NEXT: s_waitcnt vmcnt(0) expcnt(0) lgkmcnt(0) ; GFX11-NEXT: v_ashrrev_i32_e32 v1, 30, v0 ; GFX11-NEXT: s_delay_alu instid0(VALU_DEP_1) | instskip(NEXT) | instid1(VALU_DEP_1) ; GFX11-NEXT: v_xor_b32_e32 v0, v0, v1 ; GFX11-NEXT: v_clz_i32_u32_e32 v0, v0 ; GFX11-NEXT: s_delay_alu instid0(VALU_DEP_1) | instskip(NEXT) | instid1(VALU_DEP_1) ; GFX11-NEXT: v_min_u32_e32 v0, 32, v0 ; GFX11-NEXT: v_add_nc_u32_e32 v0, -1, v0 ; GFX11-NEXT: s_setpc_b64 s[30:31] %a = ashr i32 %x, 30 %b = xor i32 %x, %a %c = call i32 @llvm.ctlz.i32(i32 %b, i1 false) %d = sub i32 %c, 1 ret i32 %d } define i32 @no_ctls_xor_different_value(i32 %x, i32 %y) { ; GFX6-LABEL: no_ctls_xor_different_value: ; GFX6: ; %bb.0: ; GFX6-NEXT: s_waitcnt vmcnt(0) expcnt(0) lgkmcnt(0) ; GFX6-NEXT: v_ashrrev_i32_e32 v1, 31, v1 ; GFX6-NEXT: v_xor_b32_e32 v0, v0, v1 ; GFX6-NEXT: v_ffbh_u32_e32 v0, v0 ; GFX6-NEXT: v_min_u32_e32 v0, 32, v0 ; GFX6-NEXT: v_add_i32_e32 v0, vcc, -1, v0 ; GFX6-NEXT: s_setpc_b64 s[30:31] ; ; GFX11-LABEL: no_ctls_xor_different_value: ; GFX11: ; %bb.0: ; GFX11-NEXT: s_waitcnt vmcnt(0) expcnt(0) lgkmcnt(0) ; GFX11-NEXT: v_ashrrev_i32_e32 v1, 31, v1 ; GFX11-NEXT: s_delay_alu instid0(VALU_DEP_1) | instskip(NEXT) | instid1(VALU_DEP_1) ; GFX11-NEXT: v_xor_b32_e32 v0, v0, v1 ; GFX11-NEXT: v_clz_i32_u32_e32 v0, v0 ; GFX11-NEXT: s_delay_alu instid0(VALU_DEP_1) | instskip(NEXT) | instid1(VALU_DEP_1) ; GFX11-NEXT: v_min_u32_e32 v0, 32, v0 ; GFX11-NEXT: v_add_nc_u32_e32 v0, -1, v0 ; GFX11-NEXT: s_setpc_b64 s[30:31] %a = ashr i32 %y, 31 %b = xor i32 %x, %a %c = call i32 @llvm.ctlz.i32(i32 %b, i1 false) %d = sub i32 %c, 1 ret i32 %d } define i32 @no_ctls_sub_2(i32 %x) { ; GFX6-LABEL: no_ctls_sub_2: ; GFX6: ; %bb.0: ; GFX6-NEXT: s_waitcnt vmcnt(0) expcnt(0) lgkmcnt(0) ; GFX6-NEXT: v_ffbh_i32_e32 v0, v0 ; GFX6-NEXT: v_min_u32_e32 v0, 32, v0 ; GFX6-NEXT: v_add_i32_e32 v0, vcc, -2, v0 ; GFX6-NEXT: s_setpc_b64 s[30:31] ; ; GFX11-LABEL: no_ctls_sub_2: ; GFX11: ; %bb.0: ; GFX11-NEXT: s_waitcnt vmcnt(0) expcnt(0) lgkmcnt(0) ; GFX11-NEXT: v_cls_i32_e32 v0, v0 ; GFX11-NEXT: s_delay_alu instid0(VALU_DEP_1) | instskip(NEXT) | instid1(VALU_DEP_1) ; GFX11-NEXT: v_min_u32_e32 v0, 32, v0 ; GFX11-NEXT: v_add_nc_u32_e32 v0, -2, v0 ; GFX11-NEXT: s_setpc_b64 s[30:31] %a = ashr i32 %x, 31 %b = xor i32 %x, %a %c = call i32 @llvm.ctlz.i32(i32 %b, i1 false) %d = sub i32 %c, 2 ret i32 %d }