llvm/test/CodeGen/X86/wide-integer-cmp.ll
Hans Wennborg 086b179985 X86: More efficient legalization of wide integer compares
In particular, this makes the code for 64-bit compares on 32-bit targets
much more efficient.

Example:

  define i32 @test_slt(i64 %a, i64 %b) {
  entry:
    %cmp = icmp slt i64 %a, %b
    br i1 %cmp, label %bb1, label %bb2
  bb1:
    ret i32 1
  bb2:
    ret i32 2
  }

Before this patch:

  test_slt:
          movl    4(%esp), %eax
          movl    8(%esp), %ecx
          cmpl    12(%esp), %eax
          setae   %al
          cmpl    16(%esp), %ecx
          setge   %cl
          je      .LBB2_2
          movb    %cl, %al
  .LBB2_2:
          testb   %al, %al
          jne     .LBB2_4
          movl    $1, %eax
          retl
  .LBB2_4:
          movl    $2, %eax
          retl

After this patch:

  test_slt:
          movl    4(%esp), %eax
          movl    8(%esp), %ecx
          cmpl    12(%esp), %eax
          sbbl    16(%esp), %ecx
          jge     .LBB1_2
          movl    $1, %eax
          retl
  .LBB1_2:
          movl    $2, %eax
          retl

Differential Revision: http://reviews.llvm.org/D14496

git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@253572 91177308-0d34-0410-b5e6-96231b3b80d8
2015-11-19 16:35:08 +00:00

131 lines
2.7 KiB
LLVM

; RUN: llc -mtriple=i686-linux-gnu %s -o - | FileCheck %s
define i32 @branch_eq(i64 %a, i64 %b) {
entry:
%cmp = icmp eq i64 %a, %b
br i1 %cmp, label %bb1, label %bb2
bb1:
ret i32 1
bb2:
ret i32 2
; CHECK-LABEL: branch_eq:
; CHECK: movl 4(%esp), [[LHSLo:%[a-z]+]]
; CHECK: movl 8(%esp), [[LHSHi:%[a-z]+]]
; CHECK: xorl 16(%esp), [[LHSHi]]
; CHECK: xorl 12(%esp), [[LHSLo]]
; CHECK: orl [[LHSHi]], [[LHSLo]]
; CHECK: jne [[FALSE:.LBB[0-9_]+]]
; CHECK: movl $1, %eax
; CHECK: retl
; CHECK: [[FALSE]]:
; CHECK: movl $2, %eax
; CHECK: retl
}
define i32 @branch_slt(i64 %a, i64 %b) {
entry:
%cmp = icmp slt i64 %a, %b
br i1 %cmp, label %bb1, label %bb2
bb1:
ret i32 1
bb2:
ret i32 2
; CHECK-LABEL: branch_slt:
; CHECK: movl 4(%esp), [[LHSLo:%[a-z]+]]
; CHECK: movl 8(%esp), [[LHSHi:%[a-z]+]]
; CHECK: cmpl 12(%esp), [[LHSLo]]
; CHECK: sbbl 16(%esp), [[LHSHi]]
; CHECK: jge [[FALSE:.LBB[0-9_]+]]
; CHECK: movl $1, %eax
; CHECK: retl
; CHECK: [[FALSE]]:
; CHECK: movl $2, %eax
; CHECK: retl
}
define i32 @branch_ule(i64 %a, i64 %b) {
entry:
%cmp = icmp ule i64 %a, %b
br i1 %cmp, label %bb1, label %bb2
bb1:
ret i32 1
bb2:
ret i32 2
; CHECK-LABEL: branch_ule:
; CHECK: movl 12(%esp), [[RHSLo:%[a-z]+]]
; CHECK: movl 16(%esp), [[RHSHi:%[a-z]+]]
; CHECK: cmpl 4(%esp), [[RHSLo]]
; CHECK: sbbl 8(%esp), [[RHSHi]]
; CHECK: jb [[FALSE:.LBB[0-9_]+]]
; CHECK: movl $1, %eax
; CHECK: retl
; CHECK: [[FALSE]]:
; CHECK: movl $2, %eax
; CHECK: retl
}
define i32 @set_gt(i64 %a, i64 %b) {
entry:
%cmp = icmp sgt i64 %a, %b
%res = select i1 %cmp, i32 1, i32 0
ret i32 %res
; CHECK-LABEL: set_gt:
; CHECK: movl 12(%esp), [[RHSLo:%[a-z]+]]
; CHECK: movl 16(%esp), [[RHSHi:%[a-z]+]]
; CHECK: cmpl 4(%esp), [[RHSLo]]
; CHECK: sbbl 8(%esp), [[RHSHi]]
; CHECK: setl %al
; CHECK: retl
}
define i32 @test_wide(i128 %a, i128 %b) {
entry:
%cmp = icmp slt i128 %a, %b
br i1 %cmp, label %bb1, label %bb2
bb1:
ret i32 1
bb2:
ret i32 2
; CHECK-LABEL: test_wide:
; CHECK: cmpl 24(%esp)
; CHECK: sbbl 28(%esp)
; CHECK: sbbl 32(%esp)
; CHECK: sbbl 36(%esp)
; CHECK: jge [[FALSE:.LBB[0-9_]+]]
; CHECK: movl $1, %eax
; CHECK: retl
; CHECK: [[FALSE]]:
; CHECK: movl $2, %eax
; CHECK: retl
}
define i32 @test_carry_false(i64 %a, i64 %b) {
entry:
%x = and i64 %a, -4294967296 ;0xffffffff00000000
%y = and i64 %b, -4294967296
%cmp = icmp slt i64 %x, %y
br i1 %cmp, label %bb1, label %bb2
bb1:
ret i32 1
bb2:
ret i32 2
; The comparison of the low bits will be folded to a CARRY_FALSE node. Make
; sure the code can handle that.
; CHECK-LABEL: carry_false:
; CHECK: movl 8(%esp), [[LHSHi:%[a-z]+]]
; CHECK: cmpl 16(%esp), [[LHSHi]]
; CHECK: jge [[FALSE:.LBB[0-9_]+]]
; CHECK: movl $1, %eax
; CHECK: retl
; CHECK: [[FALSE]]:
; CHECK: movl $2, %eax
; CHECK: retl
}