blob: 68bd05a0737b6a95d1ac8c4927dac00cad6deab2 [file] [log] [blame]
; NOTE: Assertions have been autogenerated by utils/update_llc_test_checks.py UTC_ARGS: --version 5
; RUN: llc < %s -mtriple=x86_64 -mattr=+nf -verify-machineinstrs | FileCheck %s
define void @convertToThreeAddress(ptr %arg, ptr %arg1) {
; CHECK-LABEL: convertToThreeAddress:
; CHECK: # %bb.0: # %entry
; CHECK-NEXT: movslq (%rdi), %rax
; CHECK-NEXT: movslq (%rsi), %rcx
; CHECK-NEXT: subq %rax, %rcx
; CHECK-NEXT: leaq 1(%rcx), %rax
; CHECK-NEXT: js .LBB0_1
; CHECK-NEXT: .p2align 4
; CHECK-NEXT: .LBB0_6: # %bb
; CHECK-NEXT: # =>This Inner Loop Header: Depth=1
; CHECK-NEXT: cmpq $1, %rax
; CHECK-NEXT: jg .LBB0_6
; CHECK-NEXT: .LBB0_5: # %bb16
; CHECK-NEXT: retq
; CHECK-NEXT: .LBB0_1:
; CHECK-NEXT: xorl %edx, %edx
; CHECK-NEXT: .p2align 4
; CHECK-NEXT: .LBB0_2: # %bb10
; CHECK-NEXT: # =>This Inner Loop Header: Depth=1
; CHECK-NEXT: testb %dl, %dl
; CHECK-NEXT: je .LBB0_3
; CHECK-NEXT: # %bb.7: # %bb11
; CHECK-NEXT: # in Loop: Header=BB0_2 Depth=1
; CHECK-NEXT: testq %rcx, %rcx
; CHECK-NEXT: jns .LBB0_2
; CHECK-NEXT: jmp .LBB0_5
; CHECK-NEXT: .LBB0_3: # %bb10
; CHECK-NEXT: xorl %ecx, %ecx
; CHECK-NEXT: testb %cl, %cl
; CHECK-NEXT: jne .LBB0_5
; CHECK-NEXT: .p2align 4
; CHECK-NEXT: .LBB0_4: # %bb12
; CHECK-NEXT: # =>This Inner Loop Header: Depth=1
; CHECK-NEXT: cmpq $1, %rax
; CHECK-NEXT: jg .LBB0_4
; CHECK-NEXT: jmp .LBB0_5
entry:
%i = load i32, ptr %arg, align 4
%i2 = sext i32 %i to i64
%i3 = load i32, ptr %arg1, align 4
%i4 = sext i32 %i3 to i64
%i5 = sub nsw i64 %i4, %i2
%i6 = add nsw i64 %i5, 1
%i7 = icmp sgt i64 %i5, -1
br i1 %i7, label %bb, label %bb10
bb: ; preds = %bb, %entry
%i8 = phi i64 [ %i6, %entry ], [ poison, %bb ]
%i9 = icmp sgt i64 %i8, 1
br i1 %i9, label %bb, label %bb16
bb10: ; preds = %bb11, %entry
switch i32 poison, label %bb16 [
i32 1, label %bb11
i32 2, label %bb12
]
bb11: ; preds = %bb10
br i1 %i7, label %bb10, label %bb16
bb12: ; preds = %bb14, %bb10
%i13 = phi i64 [ poison, %bb14 ], [ %i6, %bb10 ]
br label %bb14
bb14: ; preds = %bb12
%i15 = icmp sgt i64 %i13, 1
br i1 %i15, label %bb12, label %bb16
bb16: ; preds = %bb14, %bb11, %bb10, %bb
ret void
}
; We must not try to replace CMP with AND_NF as it sets no flags
define void @cmp_peephole_and_nf(i64 %arg0, ptr %ptr1, ptr %ptr2) {
; CHECK-LABEL: cmp_peephole_and_nf:
; CHECK: # %bb.0: # %entry
; CHECK-NEXT: negq %rdi
; CHECK-NEXT: movl %edi, %eax
; CHECK-NEXT: {nf} andl $1, %eax
; CHECK-NEXT: jb .LBB1_2
; CHECK-NEXT: # %bb.1: # %true
; CHECK-NEXT: testq %rax, %rax
; CHECK-NEXT: sete (%rsi)
; CHECK-NEXT: retq
; CHECK-NEXT: .LBB1_2: # %false
; CHECK-NEXT: movq %rdi, (%rsi)
; CHECK-NEXT: movq %rax, (%rdx)
; CHECK-NEXT: retq
entry:
%sub_flag = sub i64 0, %arg0
%and_nf = and i64 %sub_flag, 1
%elim = icmp eq i64 0, %arg0
br i1 %elim, label %true, label %false
true:
%8 = icmp eq i64 %and_nf, 0
store i1 %8, ptr %ptr1
ret void
false:
store i64 %sub_flag, ptr %ptr1
store i64 %and_nf, ptr %ptr2
ret void
}