2017-07-03 22:29:45 +08:00
|
|
|
; NOTE: Assertions have been autogenerated by utils/update_llc_test_checks.py
|
2013-10-03 12:16:45 +08:00
|
|
|
; RUN: llc -mtriple=x86_64-unknown-unknown -mattr=+tbm < %s | FileCheck %s
|
|
|
|
|
2017-08-13 20:16:00 +08:00
|
|
|
; TODO - Patterns fail to fold with ZF flags and prevents TBM instruction selection.
|
|
|
|
|
2017-07-03 22:29:45 +08:00
|
|
|
define i32 @test_x86_tbm_bextri_u32(i32 %a) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_bextri_u32:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2018-01-12 14:21:36 +08:00
|
|
|
; CHECK-NEXT: bextrl $3076, %edi, %eax # imm = 0xC04
|
2017-07-03 22:29:45 +08:00
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = lshr i32 %a, 4
|
|
|
|
%t1 = and i32 %t0, 4095
|
|
|
|
ret i32 %t1
|
|
|
|
}
|
|
|
|
|
2017-09-13 01:40:25 +08:00
|
|
|
; Make sure we still use AH subreg trick for extracting bits 15:8
|
|
|
|
define i32 @test_x86_tbm_bextri_u32_subreg(i32 %a) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_bextri_u32_subreg:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2017-09-13 01:40:25 +08:00
|
|
|
; CHECK-NEXT: movl %edi, %eax
|
2018-01-23 13:37:00 +08:00
|
|
|
; CHECK-NEXT: movzbl %ah, %eax
|
2017-09-13 01:40:25 +08:00
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = lshr i32 %a, 8
|
|
|
|
%t1 = and i32 %t0, 255
|
|
|
|
ret i32 %t1
|
|
|
|
}
|
|
|
|
|
2017-07-03 22:29:45 +08:00
|
|
|
define i32 @test_x86_tbm_bextri_u32_m(i32* nocapture %a) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_bextri_u32_m:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2018-01-12 14:21:36 +08:00
|
|
|
; CHECK-NEXT: bextrl $3076, (%rdi), %eax # imm = 0xC04
|
2017-07-03 22:29:45 +08:00
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = load i32, i32* %a
|
|
|
|
%t1 = lshr i32 %t0, 4
|
|
|
|
%t2 = and i32 %t1, 4095
|
|
|
|
ret i32 %t2
|
|
|
|
}
|
|
|
|
|
2017-08-13 20:16:00 +08:00
|
|
|
define i32 @test_x86_tbm_bextri_u32_z(i32 %a, i32 %b) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_bextri_u32_z:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2018-01-12 14:21:36 +08:00
|
|
|
; CHECK-NEXT: bextrl $3076, %edi, %eax # imm = 0xC04
|
[x86] Fix an amazing goof in the handling of sub, or, and xor lowering.
The comment for this code indicated that it should work similar to our
handling of add lowering above: if we see uses of an instruction other
than flag usage and store usage, it tries to avoid the specialized
X86ISD::* nodes that are designed for flag+op modeling and emits an
explicit test.
Problem is, only the add case actually did this. In all the other cases,
the logic was incomplete and inverted. Any time the value was used by
a store, we bailed on the specialized X86ISD node. All of this appears
to have been historical where we had different logic here. =/
Turns out, we have quite a few patterns designed around these nodes. We
should actually form them. I fixed the code to match what we do for add,
and it has quite a positive effect just within some of our test cases.
The only thing close to a regression I see is using:
notl %r
testl %r, %r
instead of:
xorl -1, %r
But we can add a pattern or something to fold that back out. The
improvements seem more than worth this.
I've also worked with Craig to update the comments to no longer be
actively contradicted by the code. =[ Some of this still remains
a mystery to both Craig and myself, but this seems like a large step in
the direction of consistency and slightly more accurate comments.
Many thanks to Craig for help figuring out this nasty stuff.
Differential Revision: https://reviews.llvm.org/D37096
llvm-svn: 311737
2017-08-25 08:34:07 +08:00
|
|
|
; CHECK-NEXT: cmovel %esi, %eax
|
2017-08-13 20:16:00 +08:00
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = lshr i32 %a, 4
|
|
|
|
%t1 = and i32 %t0, 4095
|
|
|
|
%t2 = icmp eq i32 %t1, 0
|
|
|
|
%t3 = select i1 %t2, i32 %b, i32 %t1
|
|
|
|
ret i32 %t3
|
|
|
|
}
|
|
|
|
|
2017-08-26 07:34:55 +08:00
|
|
|
define i32 @test_x86_tbm_bextri_u32_z2(i32 %a, i32 %b, i32 %c) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_bextri_u32_z2:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2017-09-13 01:40:25 +08:00
|
|
|
; CHECK-NEXT: shrl $4, %edi
|
2017-09-29 07:35:36 +08:00
|
|
|
; CHECK-NEXT: testl $4095, %edi # imm = 0xFFF
|
2017-08-26 07:34:55 +08:00
|
|
|
; CHECK-NEXT: cmovnel %edx, %esi
|
|
|
|
; CHECK-NEXT: movl %esi, %eax
|
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = lshr i32 %a, 4
|
|
|
|
%t1 = and i32 %t0, 4095
|
|
|
|
%t2 = icmp eq i32 %t1, 0
|
|
|
|
%t3 = select i1 %t2, i32 %b, i32 %c
|
|
|
|
ret i32 %t3
|
|
|
|
}
|
|
|
|
|
2017-07-03 22:29:45 +08:00
|
|
|
define i64 @test_x86_tbm_bextri_u64(i64 %a) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_bextri_u64:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2018-01-12 14:21:36 +08:00
|
|
|
; CHECK-NEXT: bextrl $3076, %edi, %eax # imm = 0xC04
|
2017-07-03 22:29:45 +08:00
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = lshr i64 %a, 4
|
|
|
|
%t1 = and i64 %t0, 4095
|
|
|
|
ret i64 %t1
|
|
|
|
}
|
|
|
|
|
2017-09-13 01:40:25 +08:00
|
|
|
; Make sure we still use AH subreg trick for extracting bits 15:8
|
|
|
|
define i64 @test_x86_tbm_bextri_u64_subreg(i64 %a) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_bextri_u64_subreg:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2017-09-13 01:40:25 +08:00
|
|
|
; CHECK-NEXT: movq %rdi, %rax
|
2018-01-23 13:37:00 +08:00
|
|
|
; CHECK-NEXT: movzbl %ah, %eax
|
2017-09-13 01:40:25 +08:00
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = lshr i64 %a, 8
|
|
|
|
%t1 = and i64 %t0, 255
|
|
|
|
ret i64 %t1
|
|
|
|
}
|
|
|
|
|
2017-07-03 22:29:45 +08:00
|
|
|
define i64 @test_x86_tbm_bextri_u64_m(i64* nocapture %a) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_bextri_u64_m:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2018-01-12 14:21:36 +08:00
|
|
|
; CHECK-NEXT: bextrl $3076, (%rdi), %eax # imm = 0xC04
|
2017-07-03 22:29:45 +08:00
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = load i64, i64* %a
|
|
|
|
%t1 = lshr i64 %t0, 4
|
|
|
|
%t2 = and i64 %t1, 4095
|
|
|
|
ret i64 %t2
|
|
|
|
}
|
|
|
|
|
2017-08-13 20:16:00 +08:00
|
|
|
define i64 @test_x86_tbm_bextri_u64_z(i64 %a, i64 %b) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_bextri_u64_z:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2018-01-12 14:21:36 +08:00
|
|
|
; CHECK-NEXT: bextrl $3076, %edi, %eax # imm = 0xC04
|
2017-08-25 09:59:06 +08:00
|
|
|
; CHECK-NEXT: cmoveq %rsi, %rax
|
2017-08-13 20:16:00 +08:00
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = lshr i64 %a, 4
|
|
|
|
%t1 = and i64 %t0, 4095
|
|
|
|
%t2 = icmp eq i64 %t1, 0
|
|
|
|
%t3 = select i1 %t2, i64 %b, i64 %t1
|
|
|
|
ret i64 %t3
|
|
|
|
}
|
|
|
|
|
2017-08-26 07:34:55 +08:00
|
|
|
define i64 @test_x86_tbm_bextri_u64_z2(i64 %a, i64 %b, i64 %c) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_bextri_u64_z2:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2017-09-13 01:40:25 +08:00
|
|
|
; CHECK-NEXT: shrl $4, %edi
|
2017-09-29 07:35:36 +08:00
|
|
|
; CHECK-NEXT: testl $4095, %edi # imm = 0xFFF
|
2017-08-26 07:34:55 +08:00
|
|
|
; CHECK-NEXT: cmovneq %rdx, %rsi
|
|
|
|
; CHECK-NEXT: movq %rsi, %rax
|
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = lshr i64 %a, 4
|
|
|
|
%t1 = and i64 %t0, 4095
|
|
|
|
%t2 = icmp eq i64 %t1, 0
|
|
|
|
%t3 = select i1 %t2, i64 %b, i64 %c
|
|
|
|
ret i64 %t3
|
|
|
|
}
|
|
|
|
|
2017-07-03 22:29:45 +08:00
|
|
|
define i32 @test_x86_tbm_blcfill_u32(i32 %a) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_blcfill_u32:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2018-01-12 14:21:36 +08:00
|
|
|
; CHECK-NEXT: blcfilll %edi, %eax
|
2017-07-03 22:29:45 +08:00
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = add i32 %a, 1
|
|
|
|
%t1 = and i32 %t0, %a
|
|
|
|
ret i32 %t1
|
|
|
|
}
|
|
|
|
|
2017-08-13 20:16:00 +08:00
|
|
|
define i32 @test_x86_tbm_blcfill_u32_z(i32 %a, i32 %b) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_blcfill_u32_z:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2018-01-12 14:21:36 +08:00
|
|
|
; CHECK-NEXT: blcfilll %edi, %eax
|
2017-08-13 20:16:00 +08:00
|
|
|
; CHECK-NEXT: cmovel %esi, %eax
|
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = add i32 %a, 1
|
|
|
|
%t1 = and i32 %t0, %a
|
|
|
|
%t2 = icmp eq i32 %t1, 0
|
|
|
|
%t3 = select i1 %t2, i32 %b, i32 %t1
|
|
|
|
ret i32 %t3
|
|
|
|
}
|
|
|
|
|
2017-08-26 07:34:55 +08:00
|
|
|
define i32 @test_x86_tbm_blcfill_u32_z2(i32 %a, i32 %b, i32 %c) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_blcfill_u32_z2:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK-NEXT: # kill: def $edi killed $edi def $rdi
|
2017-08-26 07:34:55 +08:00
|
|
|
; CHECK-NEXT: leal 1(%rdi), %eax
|
|
|
|
; CHECK-NEXT: testl %edi, %eax
|
|
|
|
; CHECK-NEXT: cmovnel %edx, %esi
|
|
|
|
; CHECK-NEXT: movl %esi, %eax
|
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = add i32 %a, 1
|
|
|
|
%t1 = and i32 %t0, %a
|
|
|
|
%t2 = icmp eq i32 %t1, 0
|
|
|
|
%t3 = select i1 %t2, i32 %b, i32 %c
|
|
|
|
ret i32 %t3
|
|
|
|
}
|
|
|
|
|
2017-07-03 22:29:45 +08:00
|
|
|
define i64 @test_x86_tbm_blcfill_u64(i64 %a) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_blcfill_u64:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2018-01-12 14:21:36 +08:00
|
|
|
; CHECK-NEXT: blcfillq %rdi, %rax
|
2017-07-03 22:29:45 +08:00
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = add i64 %a, 1
|
|
|
|
%t1 = and i64 %t0, %a
|
|
|
|
ret i64 %t1
|
|
|
|
}
|
|
|
|
|
2017-08-13 20:16:00 +08:00
|
|
|
define i64 @test_x86_tbm_blcfill_u64_z(i64 %a, i64 %b) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_blcfill_u64_z:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2018-01-12 14:21:36 +08:00
|
|
|
; CHECK-NEXT: blcfillq %rdi, %rax
|
2017-08-13 20:16:00 +08:00
|
|
|
; CHECK-NEXT: cmoveq %rsi, %rax
|
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = add i64 %a, 1
|
|
|
|
%t1 = and i64 %t0, %a
|
|
|
|
%t2 = icmp eq i64 %t1, 0
|
|
|
|
%t3 = select i1 %t2, i64 %b, i64 %t1
|
|
|
|
ret i64 %t3
|
|
|
|
}
|
|
|
|
|
2017-08-26 07:34:55 +08:00
|
|
|
define i64 @test_x86_tbm_blcfill_u64_z2(i64 %a, i64 %b, i64 %c) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_blcfill_u64_z2:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2017-08-26 07:34:55 +08:00
|
|
|
; CHECK-NEXT: leaq 1(%rdi), %rax
|
|
|
|
; CHECK-NEXT: testq %rdi, %rax
|
|
|
|
; CHECK-NEXT: cmovneq %rdx, %rsi
|
|
|
|
; CHECK-NEXT: movq %rsi, %rax
|
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = add i64 %a, 1
|
|
|
|
%t1 = and i64 %t0, %a
|
|
|
|
%t2 = icmp eq i64 %t1, 0
|
|
|
|
%t3 = select i1 %t2, i64 %b, i64 %c
|
|
|
|
ret i64 %t3
|
|
|
|
}
|
|
|
|
|
2017-07-03 22:29:45 +08:00
|
|
|
define i32 @test_x86_tbm_blci_u32(i32 %a) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_blci_u32:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2018-01-12 14:21:36 +08:00
|
|
|
; CHECK-NEXT: blcil %edi, %eax
|
2017-07-03 22:29:45 +08:00
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = add i32 1, %a
|
|
|
|
%t1 = xor i32 %t0, -1
|
|
|
|
%t2 = or i32 %t1, %a
|
|
|
|
ret i32 %t2
|
|
|
|
}
|
|
|
|
|
2017-08-13 20:16:00 +08:00
|
|
|
define i32 @test_x86_tbm_blci_u32_z(i32 %a, i32 %b) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_blci_u32_z:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2018-01-12 14:21:36 +08:00
|
|
|
; CHECK-NEXT: blcil %edi, %eax
|
2017-08-13 20:16:00 +08:00
|
|
|
; CHECK-NEXT: cmovel %esi, %eax
|
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = add i32 1, %a
|
|
|
|
%t1 = xor i32 %t0, -1
|
|
|
|
%t2 = or i32 %t1, %a
|
|
|
|
%t3 = icmp eq i32 %t2, 0
|
|
|
|
%t4 = select i1 %t3, i32 %b, i32 %t2
|
|
|
|
ret i32 %t4
|
|
|
|
}
|
|
|
|
|
2017-08-26 07:34:55 +08:00
|
|
|
define i32 @test_x86_tbm_blci_u32_z2(i32 %a, i32 %b, i32 %c) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_blci_u32_z2:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK-NEXT: # kill: def $edi killed $edi def $rdi
|
2017-08-26 07:34:55 +08:00
|
|
|
; CHECK-NEXT: leal 1(%rdi), %eax
|
|
|
|
; CHECK-NEXT: notl %eax
|
|
|
|
; CHECK-NEXT: orl %edi, %eax
|
|
|
|
; CHECK-NEXT: cmovnel %edx, %esi
|
|
|
|
; CHECK-NEXT: movl %esi, %eax
|
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = add i32 1, %a
|
|
|
|
%t1 = xor i32 %t0, -1
|
|
|
|
%t2 = or i32 %t1, %a
|
|
|
|
%t3 = icmp eq i32 %t2, 0
|
|
|
|
%t4 = select i1 %t3, i32 %b, i32 %c
|
|
|
|
ret i32 %t4
|
|
|
|
}
|
|
|
|
|
2017-07-03 22:29:45 +08:00
|
|
|
define i64 @test_x86_tbm_blci_u64(i64 %a) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_blci_u64:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2018-01-12 14:21:36 +08:00
|
|
|
; CHECK-NEXT: blciq %rdi, %rax
|
2017-07-03 22:29:45 +08:00
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = add i64 1, %a
|
|
|
|
%t1 = xor i64 %t0, -1
|
|
|
|
%t2 = or i64 %t1, %a
|
|
|
|
ret i64 %t2
|
|
|
|
}
|
|
|
|
|
2017-08-13 20:16:00 +08:00
|
|
|
define i64 @test_x86_tbm_blci_u64_z(i64 %a, i64 %b) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_blci_u64_z:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2018-01-12 14:21:36 +08:00
|
|
|
; CHECK-NEXT: blciq %rdi, %rax
|
2017-08-13 20:16:00 +08:00
|
|
|
; CHECK-NEXT: cmoveq %rsi, %rax
|
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = add i64 1, %a
|
|
|
|
%t1 = xor i64 %t0, -1
|
|
|
|
%t2 = or i64 %t1, %a
|
|
|
|
%t3 = icmp eq i64 %t2, 0
|
|
|
|
%t4 = select i1 %t3, i64 %b, i64 %t2
|
|
|
|
ret i64 %t4
|
|
|
|
}
|
|
|
|
|
2017-08-26 07:34:55 +08:00
|
|
|
define i64 @test_x86_tbm_blci_u64_z2(i64 %a, i64 %b, i64 %c) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_blci_u64_z2:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2017-08-26 07:34:55 +08:00
|
|
|
; CHECK-NEXT: leaq 1(%rdi), %rax
|
|
|
|
; CHECK-NEXT: notq %rax
|
|
|
|
; CHECK-NEXT: orq %rdi, %rax
|
|
|
|
; CHECK-NEXT: cmovneq %rdx, %rsi
|
|
|
|
; CHECK-NEXT: movq %rsi, %rax
|
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = add i64 1, %a
|
|
|
|
%t1 = xor i64 %t0, -1
|
|
|
|
%t2 = or i64 %t1, %a
|
|
|
|
%t3 = icmp eq i64 %t2, 0
|
|
|
|
%t4 = select i1 %t3, i64 %b, i64 %c
|
|
|
|
ret i64 %t4
|
|
|
|
}
|
|
|
|
|
2017-07-03 22:29:45 +08:00
|
|
|
define i32 @test_x86_tbm_blci_u32_b(i32 %a) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_blci_u32_b:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2018-01-12 14:21:36 +08:00
|
|
|
; CHECK-NEXT: blcil %edi, %eax
|
2017-07-03 22:29:45 +08:00
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = sub i32 -2, %a
|
|
|
|
%t1 = or i32 %t0, %a
|
|
|
|
ret i32 %t1
|
|
|
|
}
|
|
|
|
|
|
|
|
define i64 @test_x86_tbm_blci_u64_b(i64 %a) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_blci_u64_b:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2018-01-12 14:21:36 +08:00
|
|
|
; CHECK-NEXT: blciq %rdi, %rax
|
2017-07-03 22:29:45 +08:00
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = sub i64 -2, %a
|
|
|
|
%t1 = or i64 %t0, %a
|
|
|
|
ret i64 %t1
|
|
|
|
}
|
|
|
|
|
|
|
|
define i32 @test_x86_tbm_blcic_u32(i32 %a) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_blcic_u32:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2018-01-12 14:21:36 +08:00
|
|
|
; CHECK-NEXT: blcicl %edi, %eax
|
2017-07-03 22:29:45 +08:00
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = xor i32 %a, -1
|
|
|
|
%t1 = add i32 %a, 1
|
|
|
|
%t2 = and i32 %t1, %t0
|
|
|
|
ret i32 %t2
|
|
|
|
}
|
|
|
|
|
2017-08-13 20:16:00 +08:00
|
|
|
define i32 @test_x86_tbm_blcic_u32_z(i32 %a, i32 %b) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_blcic_u32_z:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2018-01-12 14:21:36 +08:00
|
|
|
; CHECK-NEXT: blcicl %edi, %eax
|
2017-08-13 20:16:00 +08:00
|
|
|
; CHECK-NEXT: cmovel %esi, %eax
|
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = xor i32 %a, -1
|
|
|
|
%t1 = add i32 %a, 1
|
|
|
|
%t2 = and i32 %t1, %t0
|
|
|
|
%t3 = icmp eq i32 %t2, 0
|
|
|
|
%t4 = select i1 %t3, i32 %b, i32 %t2
|
|
|
|
ret i32 %t4
|
|
|
|
}
|
|
|
|
|
2017-08-26 07:34:55 +08:00
|
|
|
define i32 @test_x86_tbm_blcic_u32_z2(i32 %a, i32 %b, i32 %c) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_blcic_u32_z2:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2017-08-26 07:34:55 +08:00
|
|
|
; CHECK-NEXT: movl %edi, %eax
|
|
|
|
; CHECK-NEXT: notl %eax
|
|
|
|
; CHECK-NEXT: incl %edi
|
|
|
|
; CHECK-NEXT: testl %eax, %edi
|
|
|
|
; CHECK-NEXT: cmovnel %edx, %esi
|
|
|
|
; CHECK-NEXT: movl %esi, %eax
|
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = xor i32 %a, -1
|
|
|
|
%t1 = add i32 %a, 1
|
|
|
|
%t2 = and i32 %t1, %t0
|
|
|
|
%t3 = icmp eq i32 %t2, 0
|
|
|
|
%t4 = select i1 %t3, i32 %b, i32 %c
|
|
|
|
ret i32 %t4
|
|
|
|
}
|
|
|
|
|
2017-07-03 22:29:45 +08:00
|
|
|
define i64 @test_x86_tbm_blcic_u64(i64 %a) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_blcic_u64:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2018-01-12 14:21:36 +08:00
|
|
|
; CHECK-NEXT: blcicq %rdi, %rax
|
2017-07-03 22:29:45 +08:00
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = xor i64 %a, -1
|
|
|
|
%t1 = add i64 %a, 1
|
|
|
|
%t2 = and i64 %t1, %t0
|
|
|
|
ret i64 %t2
|
|
|
|
}
|
|
|
|
|
2017-08-13 20:16:00 +08:00
|
|
|
define i64 @test_x86_tbm_blcic_u64_z(i64 %a, i64 %b) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_blcic_u64_z:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2018-01-12 14:21:36 +08:00
|
|
|
; CHECK-NEXT: blcicq %rdi, %rax
|
2017-08-13 20:16:00 +08:00
|
|
|
; CHECK-NEXT: cmoveq %rsi, %rax
|
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = xor i64 %a, -1
|
|
|
|
%t1 = add i64 %a, 1
|
|
|
|
%t2 = and i64 %t1, %t0
|
|
|
|
%t3 = icmp eq i64 %t2, 0
|
|
|
|
%t4 = select i1 %t3, i64 %b, i64 %t2
|
|
|
|
ret i64 %t4
|
|
|
|
}
|
|
|
|
|
2017-08-26 07:34:55 +08:00
|
|
|
define i64 @test_x86_tbm_blcic_u64_z2(i64 %a, i64 %b, i64 %c) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_blcic_u64_z2:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2017-08-26 07:34:55 +08:00
|
|
|
; CHECK-NEXT: movq %rdi, %rax
|
|
|
|
; CHECK-NEXT: notq %rax
|
|
|
|
; CHECK-NEXT: incq %rdi
|
|
|
|
; CHECK-NEXT: testq %rax, %rdi
|
|
|
|
; CHECK-NEXT: cmovneq %rdx, %rsi
|
|
|
|
; CHECK-NEXT: movq %rsi, %rax
|
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = xor i64 %a, -1
|
|
|
|
%t1 = add i64 %a, 1
|
|
|
|
%t2 = and i64 %t1, %t0
|
|
|
|
%t3 = icmp eq i64 %t2, 0
|
|
|
|
%t4 = select i1 %t3, i64 %b, i64 %c
|
|
|
|
ret i64 %t4
|
|
|
|
}
|
|
|
|
|
2017-07-03 22:29:45 +08:00
|
|
|
define i32 @test_x86_tbm_blcmsk_u32(i32 %a) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_blcmsk_u32:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2018-01-12 14:21:36 +08:00
|
|
|
; CHECK-NEXT: blcmskl %edi, %eax
|
2017-07-03 22:29:45 +08:00
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = add i32 %a, 1
|
|
|
|
%t1 = xor i32 %t0, %a
|
|
|
|
ret i32 %t1
|
|
|
|
}
|
|
|
|
|
2017-08-13 20:16:00 +08:00
|
|
|
define i32 @test_x86_tbm_blcmsk_u32_z(i32 %a, i32 %b) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_blcmsk_u32_z:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2018-01-12 14:21:36 +08:00
|
|
|
; CHECK-NEXT: blcmskl %edi, %eax
|
2017-08-13 20:16:00 +08:00
|
|
|
; CHECK-NEXT: cmovel %esi, %eax
|
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = add i32 %a, 1
|
|
|
|
%t1 = xor i32 %t0, %a
|
|
|
|
%t2 = icmp eq i32 %t1, 0
|
|
|
|
%t3 = select i1 %t2, i32 %b, i32 %t1
|
|
|
|
ret i32 %t3
|
|
|
|
}
|
|
|
|
|
2017-08-26 07:34:55 +08:00
|
|
|
define i32 @test_x86_tbm_blcmsk_u32_z2(i32 %a, i32 %b, i32 %c) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_blcmsk_u32_z2:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK-NEXT: # kill: def $edi killed $edi def $rdi
|
2017-08-26 07:34:55 +08:00
|
|
|
; CHECK-NEXT: leal 1(%rdi), %eax
|
|
|
|
; CHECK-NEXT: xorl %edi, %eax
|
|
|
|
; CHECK-NEXT: cmovnel %edx, %esi
|
|
|
|
; CHECK-NEXT: movl %esi, %eax
|
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = add i32 %a, 1
|
|
|
|
%t1 = xor i32 %t0, %a
|
|
|
|
%t2 = icmp eq i32 %t1, 0
|
|
|
|
%t3 = select i1 %t2, i32 %b, i32 %c
|
|
|
|
ret i32 %t3
|
|
|
|
}
|
|
|
|
|
2017-07-03 22:29:45 +08:00
|
|
|
define i64 @test_x86_tbm_blcmsk_u64(i64 %a) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_blcmsk_u64:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2018-01-12 14:21:36 +08:00
|
|
|
; CHECK-NEXT: blcmskq %rdi, %rax
|
2017-07-03 22:29:45 +08:00
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = add i64 %a, 1
|
|
|
|
%t1 = xor i64 %t0, %a
|
|
|
|
ret i64 %t1
|
|
|
|
}
|
|
|
|
|
2017-08-13 20:16:00 +08:00
|
|
|
define i64 @test_x86_tbm_blcmsk_u64_z(i64 %a, i64 %b) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_blcmsk_u64_z:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2018-01-12 14:21:36 +08:00
|
|
|
; CHECK-NEXT: blcmskq %rdi, %rax
|
2017-08-13 20:16:00 +08:00
|
|
|
; CHECK-NEXT: cmoveq %rsi, %rax
|
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = add i64 %a, 1
|
|
|
|
%t1 = xor i64 %t0, %a
|
|
|
|
%t2 = icmp eq i64 %t1, 0
|
|
|
|
%t3 = select i1 %t2, i64 %b, i64 %t1
|
|
|
|
ret i64 %t3
|
|
|
|
}
|
|
|
|
|
2017-08-26 07:34:55 +08:00
|
|
|
define i64 @test_x86_tbm_blcmsk_u64_z2(i64 %a, i64 %b, i64 %c) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_blcmsk_u64_z2:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2017-08-26 07:34:55 +08:00
|
|
|
; CHECK-NEXT: leaq 1(%rdi), %rax
|
|
|
|
; CHECK-NEXT: xorq %rdi, %rax
|
|
|
|
; CHECK-NEXT: cmovneq %rdx, %rsi
|
|
|
|
; CHECK-NEXT: movq %rsi, %rax
|
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = add i64 %a, 1
|
|
|
|
%t1 = xor i64 %t0, %a
|
|
|
|
%t2 = icmp eq i64 %t1, 0
|
|
|
|
%t3 = select i1 %t2, i64 %b, i64 %c
|
|
|
|
ret i64 %t3
|
|
|
|
}
|
|
|
|
|
2017-07-03 22:29:45 +08:00
|
|
|
define i32 @test_x86_tbm_blcs_u32(i32 %a) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_blcs_u32:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2018-01-12 14:21:36 +08:00
|
|
|
; CHECK-NEXT: blcsl %edi, %eax
|
2017-07-03 22:29:45 +08:00
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = add i32 %a, 1
|
|
|
|
%t1 = or i32 %t0, %a
|
|
|
|
ret i32 %t1
|
|
|
|
}
|
|
|
|
|
2017-08-13 20:16:00 +08:00
|
|
|
define i32 @test_x86_tbm_blcs_u32_z(i32 %a, i32 %b) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_blcs_u32_z:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2018-01-12 14:21:36 +08:00
|
|
|
; CHECK-NEXT: blcsl %edi, %eax
|
2017-08-13 20:16:00 +08:00
|
|
|
; CHECK-NEXT: cmovel %esi, %eax
|
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = add i32 %a, 1
|
|
|
|
%t1 = or i32 %t0, %a
|
|
|
|
%t2 = icmp eq i32 %t1, 0
|
|
|
|
%t3 = select i1 %t2, i32 %b, i32 %t1
|
|
|
|
ret i32 %t3
|
|
|
|
}
|
|
|
|
|
2017-08-26 07:34:55 +08:00
|
|
|
define i32 @test_x86_tbm_blcs_u32_z2(i32 %a, i32 %b, i32 %c) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_blcs_u32_z2:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK-NEXT: # kill: def $edi killed $edi def $rdi
|
2017-08-26 07:34:55 +08:00
|
|
|
; CHECK-NEXT: leal 1(%rdi), %eax
|
|
|
|
; CHECK-NEXT: orl %edi, %eax
|
|
|
|
; CHECK-NEXT: cmovnel %edx, %esi
|
|
|
|
; CHECK-NEXT: movl %esi, %eax
|
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = add i32 %a, 1
|
|
|
|
%t1 = or i32 %t0, %a
|
|
|
|
%t2 = icmp eq i32 %t1, 0
|
|
|
|
%t3 = select i1 %t2, i32 %b, i32 %c
|
|
|
|
ret i32 %t3
|
|
|
|
}
|
|
|
|
|
2017-07-03 22:29:45 +08:00
|
|
|
define i64 @test_x86_tbm_blcs_u64(i64 %a) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_blcs_u64:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2018-01-12 14:21:36 +08:00
|
|
|
; CHECK-NEXT: blcsq %rdi, %rax
|
2017-07-03 22:29:45 +08:00
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = add i64 %a, 1
|
|
|
|
%t1 = or i64 %t0, %a
|
|
|
|
ret i64 %t1
|
|
|
|
}
|
|
|
|
|
2017-08-13 20:16:00 +08:00
|
|
|
define i64 @test_x86_tbm_blcs_u64_z(i64 %a, i64 %b) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_blcs_u64_z:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2018-01-12 14:21:36 +08:00
|
|
|
; CHECK-NEXT: blcsq %rdi, %rax
|
2017-08-13 20:16:00 +08:00
|
|
|
; CHECK-NEXT: cmoveq %rsi, %rax
|
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = add i64 %a, 1
|
|
|
|
%t1 = or i64 %t0, %a
|
|
|
|
%t2 = icmp eq i64 %t1, 0
|
|
|
|
%t3 = select i1 %t2, i64 %b, i64 %t1
|
|
|
|
ret i64 %t3
|
|
|
|
}
|
|
|
|
|
2017-08-26 07:34:55 +08:00
|
|
|
define i64 @test_x86_tbm_blcs_u64_z2(i64 %a, i64 %b, i64 %c) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_blcs_u64_z2:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2017-08-26 07:34:55 +08:00
|
|
|
; CHECK-NEXT: leaq 1(%rdi), %rax
|
|
|
|
; CHECK-NEXT: orq %rdi, %rax
|
|
|
|
; CHECK-NEXT: cmovneq %rdx, %rsi
|
|
|
|
; CHECK-NEXT: movq %rsi, %rax
|
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = add i64 %a, 1
|
|
|
|
%t1 = or i64 %t0, %a
|
|
|
|
%t2 = icmp eq i64 %t1, 0
|
|
|
|
%t3 = select i1 %t2, i64 %b, i64 %c
|
|
|
|
ret i64 %t3
|
|
|
|
}
|
|
|
|
|
2017-07-03 22:29:45 +08:00
|
|
|
define i32 @test_x86_tbm_blsfill_u32(i32 %a) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_blsfill_u32:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2018-01-12 14:21:36 +08:00
|
|
|
; CHECK-NEXT: blsfilll %edi, %eax
|
2017-07-03 22:29:45 +08:00
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = add i32 %a, -1
|
|
|
|
%t1 = or i32 %t0, %a
|
|
|
|
ret i32 %t1
|
|
|
|
}
|
|
|
|
|
2017-08-13 20:16:00 +08:00
|
|
|
define i32 @test_x86_tbm_blsfill_u32_z(i32 %a, i32 %b) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_blsfill_u32_z:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2018-01-12 14:21:36 +08:00
|
|
|
; CHECK-NEXT: blsfilll %edi, %eax
|
2017-08-13 20:16:00 +08:00
|
|
|
; CHECK-NEXT: cmovel %esi, %eax
|
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = add i32 %a, -1
|
|
|
|
%t1 = or i32 %t0, %a
|
|
|
|
%t2 = icmp eq i32 %t1, 0
|
|
|
|
%t3 = select i1 %t2, i32 %b, i32 %t1
|
|
|
|
ret i32 %t3
|
|
|
|
}
|
|
|
|
|
2017-08-26 07:34:55 +08:00
|
|
|
define i32 @test_x86_tbm_blsfill_u32_z2(i32 %a, i32 %b, i32 %c) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_blsfill_u32_z2:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK-NEXT: # kill: def $edi killed $edi def $rdi
|
2017-08-26 07:34:55 +08:00
|
|
|
; CHECK-NEXT: leal -1(%rdi), %eax
|
|
|
|
; CHECK-NEXT: orl %edi, %eax
|
|
|
|
; CHECK-NEXT: cmovnel %edx, %esi
|
|
|
|
; CHECK-NEXT: movl %esi, %eax
|
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = add i32 %a, -1
|
|
|
|
%t1 = or i32 %t0, %a
|
|
|
|
%t2 = icmp eq i32 %t1, 0
|
|
|
|
%t3 = select i1 %t2, i32 %b, i32 %c
|
|
|
|
ret i32 %t3
|
|
|
|
}
|
|
|
|
|
2017-07-03 22:29:45 +08:00
|
|
|
define i64 @test_x86_tbm_blsfill_u64(i64 %a) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_blsfill_u64:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2018-01-12 14:21:36 +08:00
|
|
|
; CHECK-NEXT: blsfillq %rdi, %rax
|
2017-07-03 22:29:45 +08:00
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = add i64 %a, -1
|
|
|
|
%t1 = or i64 %t0, %a
|
|
|
|
ret i64 %t1
|
|
|
|
}
|
|
|
|
|
2017-08-13 20:16:00 +08:00
|
|
|
define i64 @test_x86_tbm_blsfill_u64_z(i64 %a, i64 %b) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_blsfill_u64_z:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2018-01-12 14:21:36 +08:00
|
|
|
; CHECK-NEXT: blsfillq %rdi, %rax
|
2017-08-13 20:16:00 +08:00
|
|
|
; CHECK-NEXT: cmoveq %rsi, %rax
|
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = add i64 %a, -1
|
|
|
|
%t1 = or i64 %t0, %a
|
|
|
|
%t2 = icmp eq i64 %t1, 0
|
|
|
|
%t3 = select i1 %t2, i64 %b, i64 %t1
|
|
|
|
ret i64 %t3
|
|
|
|
}
|
|
|
|
|
2017-08-26 07:34:55 +08:00
|
|
|
define i64 @test_x86_tbm_blsfill_u64_z2(i64 %a, i64 %b, i64 %c) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_blsfill_u64_z2:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2017-08-26 07:34:55 +08:00
|
|
|
; CHECK-NEXT: leaq -1(%rdi), %rax
|
|
|
|
; CHECK-NEXT: orq %rdi, %rax
|
|
|
|
; CHECK-NEXT: cmovneq %rdx, %rsi
|
|
|
|
; CHECK-NEXT: movq %rsi, %rax
|
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = add i64 %a, -1
|
|
|
|
%t1 = or i64 %t0, %a
|
|
|
|
%t2 = icmp eq i64 %t1, 0
|
|
|
|
%t3 = select i1 %t2, i64 %b, i64 %c
|
|
|
|
ret i64 %t3
|
|
|
|
}
|
|
|
|
|
2017-07-03 22:29:45 +08:00
|
|
|
define i32 @test_x86_tbm_blsic_u32(i32 %a) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_blsic_u32:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2018-01-12 14:21:36 +08:00
|
|
|
; CHECK-NEXT: blsicl %edi, %eax
|
2017-07-03 22:29:45 +08:00
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = xor i32 %a, -1
|
|
|
|
%t1 = add i32 %a, -1
|
|
|
|
%t2 = or i32 %t0, %t1
|
|
|
|
ret i32 %t2
|
|
|
|
}
|
|
|
|
|
2017-08-13 20:16:00 +08:00
|
|
|
define i32 @test_x86_tbm_blsic_u32_z(i32 %a, i32 %b) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_blsic_u32_z:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2018-01-12 14:21:36 +08:00
|
|
|
; CHECK-NEXT: blsicl %edi, %eax
|
[x86] Fix an amazing goof in the handling of sub, or, and xor lowering.
The comment for this code indicated that it should work similar to our
handling of add lowering above: if we see uses of an instruction other
than flag usage and store usage, it tries to avoid the specialized
X86ISD::* nodes that are designed for flag+op modeling and emits an
explicit test.
Problem is, only the add case actually did this. In all the other cases,
the logic was incomplete and inverted. Any time the value was used by
a store, we bailed on the specialized X86ISD node. All of this appears
to have been historical where we had different logic here. =/
Turns out, we have quite a few patterns designed around these nodes. We
should actually form them. I fixed the code to match what we do for add,
and it has quite a positive effect just within some of our test cases.
The only thing close to a regression I see is using:
notl %r
testl %r, %r
instead of:
xorl -1, %r
But we can add a pattern or something to fold that back out. The
improvements seem more than worth this.
I've also worked with Craig to update the comments to no longer be
actively contradicted by the code. =[ Some of this still remains
a mystery to both Craig and myself, but this seems like a large step in
the direction of consistency and slightly more accurate comments.
Many thanks to Craig for help figuring out this nasty stuff.
Differential Revision: https://reviews.llvm.org/D37096
llvm-svn: 311737
2017-08-25 08:34:07 +08:00
|
|
|
; CHECK-NEXT: cmovel %esi, %eax
|
2017-08-13 20:16:00 +08:00
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = xor i32 %a, -1
|
|
|
|
%t1 = add i32 %a, -1
|
|
|
|
%t2 = or i32 %t0, %t1
|
|
|
|
%t3 = icmp eq i32 %t2, 0
|
|
|
|
%t4 = select i1 %t3, i32 %b, i32 %t2
|
|
|
|
ret i32 %t4
|
|
|
|
}
|
|
|
|
|
2017-08-26 07:34:55 +08:00
|
|
|
define i32 @test_x86_tbm_blsic_u32_z2(i32 %a, i32 %b, i32 %c) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_blsic_u32_z2:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2017-08-26 07:34:55 +08:00
|
|
|
; CHECK-NEXT: movl %edi, %eax
|
|
|
|
; CHECK-NEXT: notl %eax
|
|
|
|
; CHECK-NEXT: decl %edi
|
|
|
|
; CHECK-NEXT: orl %eax, %edi
|
|
|
|
; CHECK-NEXT: cmovnel %edx, %esi
|
|
|
|
; CHECK-NEXT: movl %esi, %eax
|
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = xor i32 %a, -1
|
|
|
|
%t1 = add i32 %a, -1
|
|
|
|
%t2 = or i32 %t0, %t1
|
|
|
|
%t3 = icmp eq i32 %t2, 0
|
|
|
|
%t4 = select i1 %t3, i32 %b, i32 %c
|
|
|
|
ret i32 %t4
|
|
|
|
}
|
|
|
|
|
2017-07-03 22:29:45 +08:00
|
|
|
define i64 @test_x86_tbm_blsic_u64(i64 %a) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_blsic_u64:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2018-01-12 14:21:36 +08:00
|
|
|
; CHECK-NEXT: blsicq %rdi, %rax
|
2017-07-03 22:29:45 +08:00
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = xor i64 %a, -1
|
|
|
|
%t1 = add i64 %a, -1
|
|
|
|
%t2 = or i64 %t0, %t1
|
|
|
|
ret i64 %t2
|
|
|
|
}
|
|
|
|
|
2017-08-13 20:16:00 +08:00
|
|
|
define i64 @test_x86_tbm_blsic_u64_z(i64 %a, i64 %b) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_blsic_u64_z:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2018-01-12 14:21:36 +08:00
|
|
|
; CHECK-NEXT: blsicq %rdi, %rax
|
[x86] Fix an amazing goof in the handling of sub, or, and xor lowering.
The comment for this code indicated that it should work similar to our
handling of add lowering above: if we see uses of an instruction other
than flag usage and store usage, it tries to avoid the specialized
X86ISD::* nodes that are designed for flag+op modeling and emits an
explicit test.
Problem is, only the add case actually did this. In all the other cases,
the logic was incomplete and inverted. Any time the value was used by
a store, we bailed on the specialized X86ISD node. All of this appears
to have been historical where we had different logic here. =/
Turns out, we have quite a few patterns designed around these nodes. We
should actually form them. I fixed the code to match what we do for add,
and it has quite a positive effect just within some of our test cases.
The only thing close to a regression I see is using:
notl %r
testl %r, %r
instead of:
xorl -1, %r
But we can add a pattern or something to fold that back out. The
improvements seem more than worth this.
I've also worked with Craig to update the comments to no longer be
actively contradicted by the code. =[ Some of this still remains
a mystery to both Craig and myself, but this seems like a large step in
the direction of consistency and slightly more accurate comments.
Many thanks to Craig for help figuring out this nasty stuff.
Differential Revision: https://reviews.llvm.org/D37096
llvm-svn: 311737
2017-08-25 08:34:07 +08:00
|
|
|
; CHECK-NEXT: cmoveq %rsi, %rax
|
2017-08-13 20:16:00 +08:00
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = xor i64 %a, -1
|
|
|
|
%t1 = add i64 %a, -1
|
|
|
|
%t2 = or i64 %t0, %t1
|
|
|
|
%t3 = icmp eq i64 %t2, 0
|
|
|
|
%t4 = select i1 %t3, i64 %b, i64 %t2
|
|
|
|
ret i64 %t4
|
|
|
|
}
|
|
|
|
|
2017-08-26 07:34:55 +08:00
|
|
|
define i64 @test_x86_tbm_blsic_u64_z2(i64 %a, i64 %b, i64 %c) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_blsic_u64_z2:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2017-08-26 07:34:55 +08:00
|
|
|
; CHECK-NEXT: movq %rdi, %rax
|
|
|
|
; CHECK-NEXT: notq %rax
|
|
|
|
; CHECK-NEXT: decq %rdi
|
|
|
|
; CHECK-NEXT: orq %rax, %rdi
|
|
|
|
; CHECK-NEXT: cmovneq %rdx, %rsi
|
|
|
|
; CHECK-NEXT: movq %rsi, %rax
|
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = xor i64 %a, -1
|
|
|
|
%t1 = add i64 %a, -1
|
|
|
|
%t2 = or i64 %t0, %t1
|
|
|
|
%t3 = icmp eq i64 %t2, 0
|
|
|
|
%t4 = select i1 %t3, i64 %b, i64 %c
|
|
|
|
ret i64 %t4
|
|
|
|
}
|
|
|
|
|
2017-07-03 22:29:45 +08:00
|
|
|
define i32 @test_x86_tbm_t1mskc_u32(i32 %a) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_t1mskc_u32:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2018-01-12 14:21:36 +08:00
|
|
|
; CHECK-NEXT: t1mskcl %edi, %eax
|
2017-07-03 22:29:45 +08:00
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = xor i32 %a, -1
|
|
|
|
%t1 = add i32 %a, 1
|
|
|
|
%t2 = or i32 %t0, %t1
|
|
|
|
ret i32 %t2
|
|
|
|
}
|
|
|
|
|
2017-08-13 20:16:00 +08:00
|
|
|
define i32 @test_x86_tbm_t1mskc_u32_z(i32 %a, i32 %b) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_t1mskc_u32_z:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2018-01-12 14:21:36 +08:00
|
|
|
; CHECK-NEXT: t1mskcl %edi, %eax
|
[x86] Fix an amazing goof in the handling of sub, or, and xor lowering.
The comment for this code indicated that it should work similar to our
handling of add lowering above: if we see uses of an instruction other
than flag usage and store usage, it tries to avoid the specialized
X86ISD::* nodes that are designed for flag+op modeling and emits an
explicit test.
Problem is, only the add case actually did this. In all the other cases,
the logic was incomplete and inverted. Any time the value was used by
a store, we bailed on the specialized X86ISD node. All of this appears
to have been historical where we had different logic here. =/
Turns out, we have quite a few patterns designed around these nodes. We
should actually form them. I fixed the code to match what we do for add,
and it has quite a positive effect just within some of our test cases.
The only thing close to a regression I see is using:
notl %r
testl %r, %r
instead of:
xorl -1, %r
But we can add a pattern or something to fold that back out. The
improvements seem more than worth this.
I've also worked with Craig to update the comments to no longer be
actively contradicted by the code. =[ Some of this still remains
a mystery to both Craig and myself, but this seems like a large step in
the direction of consistency and slightly more accurate comments.
Many thanks to Craig for help figuring out this nasty stuff.
Differential Revision: https://reviews.llvm.org/D37096
llvm-svn: 311737
2017-08-25 08:34:07 +08:00
|
|
|
; CHECK-NEXT: testl %eax, %eax
|
|
|
|
; CHECK-NEXT: cmovel %esi, %eax
|
2017-08-13 20:16:00 +08:00
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = xor i32 %a, -1
|
|
|
|
%t1 = add i32 %a, 1
|
|
|
|
%t2 = or i32 %t0, %t1
|
|
|
|
%t3 = icmp eq i32 %t2, 0
|
|
|
|
%t4 = select i1 %t3, i32 %b, i32 %t2
|
|
|
|
ret i32 %t4
|
|
|
|
}
|
|
|
|
|
2017-08-26 07:34:55 +08:00
|
|
|
define i32 @test_x86_tbm_t1mskc_u32_z2(i32 %a, i32 %b, i32 %c) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_t1mskc_u32_z2:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2017-08-26 07:34:55 +08:00
|
|
|
; CHECK-NEXT: movl %edi, %eax
|
|
|
|
; CHECK-NEXT: notl %eax
|
|
|
|
; CHECK-NEXT: incl %edi
|
|
|
|
; CHECK-NEXT: orl %eax, %edi
|
|
|
|
; CHECK-NEXT: cmovnel %edx, %esi
|
|
|
|
; CHECK-NEXT: movl %esi, %eax
|
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = xor i32 %a, -1
|
|
|
|
%t1 = add i32 %a, 1
|
|
|
|
%t2 = or i32 %t0, %t1
|
|
|
|
%t3 = icmp eq i32 %t2, 0
|
|
|
|
%t4 = select i1 %t3, i32 %b, i32 %c
|
|
|
|
ret i32 %t4
|
|
|
|
}
|
|
|
|
|
2017-08-13 20:16:00 +08:00
|
|
|
define i64 @test_x86_tbm_t1mskc_u64(i64 %a) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_t1mskc_u64:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2018-01-12 14:21:36 +08:00
|
|
|
; CHECK-NEXT: t1mskcq %rdi, %rax
|
2017-07-03 22:29:45 +08:00
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = xor i64 %a, -1
|
|
|
|
%t1 = add i64 %a, 1
|
|
|
|
%t2 = or i64 %t0, %t1
|
|
|
|
ret i64 %t2
|
|
|
|
}
|
|
|
|
|
2017-08-13 20:16:00 +08:00
|
|
|
define i64 @test_x86_tbm_t1mskc_u64_z(i64 %a, i64 %b) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_t1mskc_u64_z:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2018-01-12 14:21:36 +08:00
|
|
|
; CHECK-NEXT: t1mskcq %rdi, %rax
|
[x86] Fix an amazing goof in the handling of sub, or, and xor lowering.
The comment for this code indicated that it should work similar to our
handling of add lowering above: if we see uses of an instruction other
than flag usage and store usage, it tries to avoid the specialized
X86ISD::* nodes that are designed for flag+op modeling and emits an
explicit test.
Problem is, only the add case actually did this. In all the other cases,
the logic was incomplete and inverted. Any time the value was used by
a store, we bailed on the specialized X86ISD node. All of this appears
to have been historical where we had different logic here. =/
Turns out, we have quite a few patterns designed around these nodes. We
should actually form them. I fixed the code to match what we do for add,
and it has quite a positive effect just within some of our test cases.
The only thing close to a regression I see is using:
notl %r
testl %r, %r
instead of:
xorl -1, %r
But we can add a pattern or something to fold that back out. The
improvements seem more than worth this.
I've also worked with Craig to update the comments to no longer be
actively contradicted by the code. =[ Some of this still remains
a mystery to both Craig and myself, but this seems like a large step in
the direction of consistency and slightly more accurate comments.
Many thanks to Craig for help figuring out this nasty stuff.
Differential Revision: https://reviews.llvm.org/D37096
llvm-svn: 311737
2017-08-25 08:34:07 +08:00
|
|
|
; CHECK-NEXT: testq %rax, %rax
|
|
|
|
; CHECK-NEXT: cmoveq %rsi, %rax
|
2017-08-13 20:16:00 +08:00
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = xor i64 %a, -1
|
|
|
|
%t1 = add i64 %a, 1
|
|
|
|
%t2 = or i64 %t0, %t1
|
|
|
|
%t3 = icmp eq i64 %t2, 0
|
|
|
|
%t4 = select i1 %t3, i64 %b, i64 %t2
|
|
|
|
ret i64 %t4
|
|
|
|
}
|
|
|
|
|
2017-08-26 07:34:55 +08:00
|
|
|
define i64 @test_x86_tbm_t1mskc_u64_z2(i64 %a, i64 %b, i64 %c) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_t1mskc_u64_z2:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2017-08-26 07:34:55 +08:00
|
|
|
; CHECK-NEXT: movq %rdi, %rax
|
|
|
|
; CHECK-NEXT: notq %rax
|
|
|
|
; CHECK-NEXT: incq %rdi
|
|
|
|
; CHECK-NEXT: orq %rax, %rdi
|
|
|
|
; CHECK-NEXT: cmovneq %rdx, %rsi
|
|
|
|
; CHECK-NEXT: movq %rsi, %rax
|
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = xor i64 %a, -1
|
|
|
|
%t1 = add i64 %a, 1
|
|
|
|
%t2 = or i64 %t0, %t1
|
|
|
|
%t3 = icmp eq i64 %t2, 0
|
|
|
|
%t4 = select i1 %t3, i64 %b, i64 %c
|
|
|
|
ret i64 %t4
|
|
|
|
}
|
|
|
|
|
2017-07-03 22:29:45 +08:00
|
|
|
define i32 @test_x86_tbm_tzmsk_u32(i32 %a) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_tzmsk_u32:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2018-01-12 14:21:36 +08:00
|
|
|
; CHECK-NEXT: tzmskl %edi, %eax
|
2017-07-03 22:29:45 +08:00
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = xor i32 %a, -1
|
|
|
|
%t1 = add i32 %a, -1
|
|
|
|
%t2 = and i32 %t0, %t1
|
|
|
|
ret i32 %t2
|
|
|
|
}
|
|
|
|
|
2017-08-13 20:16:00 +08:00
|
|
|
define i32 @test_x86_tbm_tzmsk_u32_z(i32 %a, i32 %b) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_tzmsk_u32_z:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2018-01-12 14:21:36 +08:00
|
|
|
; CHECK-NEXT: tzmskl %edi, %eax
|
[x86] Fix an amazing goof in the handling of sub, or, and xor lowering.
The comment for this code indicated that it should work similar to our
handling of add lowering above: if we see uses of an instruction other
than flag usage and store usage, it tries to avoid the specialized
X86ISD::* nodes that are designed for flag+op modeling and emits an
explicit test.
Problem is, only the add case actually did this. In all the other cases,
the logic was incomplete and inverted. Any time the value was used by
a store, we bailed on the specialized X86ISD node. All of this appears
to have been historical where we had different logic here. =/
Turns out, we have quite a few patterns designed around these nodes. We
should actually form them. I fixed the code to match what we do for add,
and it has quite a positive effect just within some of our test cases.
The only thing close to a regression I see is using:
notl %r
testl %r, %r
instead of:
xorl -1, %r
But we can add a pattern or something to fold that back out. The
improvements seem more than worth this.
I've also worked with Craig to update the comments to no longer be
actively contradicted by the code. =[ Some of this still remains
a mystery to both Craig and myself, but this seems like a large step in
the direction of consistency and slightly more accurate comments.
Many thanks to Craig for help figuring out this nasty stuff.
Differential Revision: https://reviews.llvm.org/D37096
llvm-svn: 311737
2017-08-25 08:34:07 +08:00
|
|
|
; CHECK-NEXT: testl %eax, %eax
|
|
|
|
; CHECK-NEXT: cmovel %esi, %eax
|
2017-08-13 20:16:00 +08:00
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = xor i32 %a, -1
|
|
|
|
%t1 = add i32 %a, -1
|
|
|
|
%t2 = and i32 %t0, %t1
|
|
|
|
%t3 = icmp eq i32 %t2, 0
|
|
|
|
%t4 = select i1 %t3, i32 %b, i32 %t2
|
|
|
|
ret i32 %t4
|
|
|
|
}
|
|
|
|
|
2017-08-26 07:34:55 +08:00
|
|
|
define i32 @test_x86_tbm_tzmsk_u32_z2(i32 %a, i32 %b, i32 %c) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_tzmsk_u32_z2:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2017-08-26 07:34:55 +08:00
|
|
|
; CHECK-NEXT: movl %edi, %eax
|
|
|
|
; CHECK-NEXT: notl %eax
|
|
|
|
; CHECK-NEXT: decl %edi
|
|
|
|
; CHECK-NEXT: testl %edi, %eax
|
|
|
|
; CHECK-NEXT: cmovnel %edx, %esi
|
|
|
|
; CHECK-NEXT: movl %esi, %eax
|
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = xor i32 %a, -1
|
|
|
|
%t1 = add i32 %a, -1
|
|
|
|
%t2 = and i32 %t0, %t1
|
|
|
|
%t3 = icmp eq i32 %t2, 0
|
|
|
|
%t4 = select i1 %t3, i32 %b, i32 %c
|
|
|
|
ret i32 %t4
|
|
|
|
}
|
|
|
|
|
2017-07-03 22:29:45 +08:00
|
|
|
define i64 @test_x86_tbm_tzmsk_u64(i64 %a) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_tzmsk_u64:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2018-01-12 14:21:36 +08:00
|
|
|
; CHECK-NEXT: tzmskq %rdi, %rax
|
2017-07-03 22:29:45 +08:00
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = xor i64 %a, -1
|
|
|
|
%t1 = add i64 %a, -1
|
|
|
|
%t2 = and i64 %t0, %t1
|
|
|
|
ret i64 %t2
|
2013-10-03 12:16:45 +08:00
|
|
|
}
|
2017-07-03 22:29:45 +08:00
|
|
|
|
2017-08-13 20:16:00 +08:00
|
|
|
define i64 @test_x86_tbm_tzmsk_u64_z(i64 %a, i64 %b) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_tzmsk_u64_z:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2018-01-12 14:21:36 +08:00
|
|
|
; CHECK-NEXT: tzmskq %rdi, %rax
|
[x86] Fix an amazing goof in the handling of sub, or, and xor lowering.
The comment for this code indicated that it should work similar to our
handling of add lowering above: if we see uses of an instruction other
than flag usage and store usage, it tries to avoid the specialized
X86ISD::* nodes that are designed for flag+op modeling and emits an
explicit test.
Problem is, only the add case actually did this. In all the other cases,
the logic was incomplete and inverted. Any time the value was used by
a store, we bailed on the specialized X86ISD node. All of this appears
to have been historical where we had different logic here. =/
Turns out, we have quite a few patterns designed around these nodes. We
should actually form them. I fixed the code to match what we do for add,
and it has quite a positive effect just within some of our test cases.
The only thing close to a regression I see is using:
notl %r
testl %r, %r
instead of:
xorl -1, %r
But we can add a pattern or something to fold that back out. The
improvements seem more than worth this.
I've also worked with Craig to update the comments to no longer be
actively contradicted by the code. =[ Some of this still remains
a mystery to both Craig and myself, but this seems like a large step in
the direction of consistency and slightly more accurate comments.
Many thanks to Craig for help figuring out this nasty stuff.
Differential Revision: https://reviews.llvm.org/D37096
llvm-svn: 311737
2017-08-25 08:34:07 +08:00
|
|
|
; CHECK-NEXT: testq %rax, %rax
|
|
|
|
; CHECK-NEXT: cmoveq %rsi, %rax
|
2017-08-13 20:16:00 +08:00
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = xor i64 %a, -1
|
|
|
|
%t1 = add i64 %a, -1
|
|
|
|
%t2 = and i64 %t0, %t1
|
|
|
|
%t3 = icmp eq i64 %t2, 0
|
|
|
|
%t4 = select i1 %t3, i64 %b, i64 %t2
|
|
|
|
ret i64 %t4
|
|
|
|
}
|
|
|
|
|
2017-08-26 07:34:55 +08:00
|
|
|
define i64 @test_x86_tbm_tzmsk_u64_z2(i64 %a, i64 %b, i64 %c) nounwind {
|
|
|
|
; CHECK-LABEL: test_x86_tbm_tzmsk_u64_z2:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0:
|
2017-08-26 07:34:55 +08:00
|
|
|
; CHECK-NEXT: movq %rdi, %rax
|
|
|
|
; CHECK-NEXT: notq %rax
|
|
|
|
; CHECK-NEXT: decq %rdi
|
|
|
|
; CHECK-NEXT: testq %rdi, %rax
|
|
|
|
; CHECK-NEXT: cmovneq %rdx, %rsi
|
|
|
|
; CHECK-NEXT: movq %rsi, %rax
|
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%t0 = xor i64 %a, -1
|
|
|
|
%t1 = add i64 %a, -1
|
|
|
|
%t2 = and i64 %t0, %t1
|
|
|
|
%t3 = icmp eq i64 %t2, 0
|
|
|
|
%t4 = select i1 %t3, i64 %b, i64 %c
|
|
|
|
ret i64 %t4
|
|
|
|
}
|
|
|
|
|
2017-08-02 01:18:14 +08:00
|
|
|
define i64 @test_and_large_constant_mask(i64 %x) {
|
|
|
|
; CHECK-LABEL: test_and_large_constant_mask:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0: # %entry
|
2018-01-12 14:21:36 +08:00
|
|
|
; CHECK-NEXT: bextrq $15872, %rdi, %rax # imm = 0x3E00
|
2017-08-02 01:18:14 +08:00
|
|
|
; CHECK-NEXT: retq
|
|
|
|
entry:
|
|
|
|
%and = and i64 %x, 4611686018427387903
|
|
|
|
ret i64 %and
|
|
|
|
}
|
|
|
|
|
|
|
|
define i64 @test_and_large_constant_mask_load(i64* %x) {
|
|
|
|
; CHECK-LABEL: test_and_large_constant_mask_load:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: # %bb.0: # %entry
|
2018-01-12 14:21:36 +08:00
|
|
|
; CHECK-NEXT: bextrq $15872, (%rdi), %rax # imm = 0x3E00
|
2017-08-02 01:18:14 +08:00
|
|
|
; CHECK-NEXT: retq
|
|
|
|
entry:
|
|
|
|
%x1 = load i64, i64* %x
|
|
|
|
%and = and i64 %x1, 4611686018427387903
|
|
|
|
ret i64 %and
|
|
|
|
}
|
2018-02-13 15:28:28 +08:00
|
|
|
|
2018-02-14 00:25:27 +08:00
|
|
|
; Make sure the mask doesn't break our matching of blcic
|
2018-02-13 15:28:28 +08:00
|
|
|
define i64 @masked_blcic(i64) {
|
|
|
|
; CHECK-LABEL: masked_blcic:
|
|
|
|
; CHECK: # %bb.0:
|
|
|
|
; CHECK-NEXT: movzwl %di, %eax
|
2018-02-14 00:25:27 +08:00
|
|
|
; CHECK-NEXT: blcicl %eax, %eax
|
2018-02-13 15:28:28 +08:00
|
|
|
; CHECK-NEXT: retq
|
|
|
|
%2 = and i64 %0, 65535
|
|
|
|
%3 = xor i64 %2, -1
|
|
|
|
%4 = add nuw nsw i64 %2, 1
|
|
|
|
%5 = and i64 %4, %3
|
|
|
|
ret i64 %5
|
|
|
|
}
|