mirror of
https://github.com/c64scene-ar/llvm-6502.git
synced 2025-01-14 00:32:55 +00:00
fef8e383eb
If we know that a particular 64-bit constant has all high bits zero, then we can rely on the fact that 32-bit ARM64 instructions automatically zero out the high bits of an x-register. This gives the expansion logic less constraints to satisfy and so sometimes allows it to pick better sequences. Came up while porting test/CodeGen/AArch64/movw-consts.ll: this will allow a 32-bit MOVN to be used in @test8 soon. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@206379 91177308-0d34-0410-b5e6-96231b3b80d8
223 lines
5.4 KiB
LLVM
223 lines
5.4 KiB
LLVM
; RUN: llc -O3 < %s | FileCheck %s
|
|
target datalayout = "e-p:64:64:64-i1:8:8-i8:8:8-i16:16:16-i32:32:32-i64:64:64-f32:32:32-f64:64:64-v64:64:64-v128:128:128-a0:0:64-n32:64"
|
|
target triple = "arm64-unknown-unknown"
|
|
|
|
; CHECK: foo1
|
|
; CHECK: csinc w{{[0-9]+}}, w[[REG:[0-9]+]],
|
|
; CHECK: w[[REG]], eq
|
|
define i32 @foo1(i32 %b, i32 %c) nounwind readnone ssp {
|
|
entry:
|
|
%not.tobool = icmp ne i32 %c, 0
|
|
%add = zext i1 %not.tobool to i32
|
|
%b.add = add i32 %c, %b
|
|
%add1 = add i32 %b.add, %add
|
|
ret i32 %add1
|
|
}
|
|
|
|
; CHECK: foo2
|
|
; CHECK: csneg w{{[0-9]+}}, w[[REG:[0-9]+]],
|
|
; CHECK: w[[REG]], eq
|
|
define i32 @foo2(i32 %b, i32 %c) nounwind readnone ssp {
|
|
entry:
|
|
%mul = sub i32 0, %b
|
|
%tobool = icmp eq i32 %c, 0
|
|
%b.mul = select i1 %tobool, i32 %b, i32 %mul
|
|
%add = add nsw i32 %b.mul, %c
|
|
ret i32 %add
|
|
}
|
|
|
|
; CHECK: foo3
|
|
; CHECK: csinv w{{[0-9]+}}, w[[REG:[0-9]+]],
|
|
; CHECK: w[[REG]], eq
|
|
define i32 @foo3(i32 %b, i32 %c) nounwind readnone ssp {
|
|
entry:
|
|
%not.tobool = icmp ne i32 %c, 0
|
|
%xor = sext i1 %not.tobool to i32
|
|
%b.xor = xor i32 %xor, %b
|
|
%add = add nsw i32 %b.xor, %c
|
|
ret i32 %add
|
|
}
|
|
|
|
; rdar://11632325
|
|
define i32@foo4(i32 %a) nounwind ssp {
|
|
; CHECK: foo4
|
|
; CHECK: csneg
|
|
; CHECK-NEXT: ret
|
|
%cmp = icmp sgt i32 %a, -1
|
|
%neg = sub nsw i32 0, %a
|
|
%cond = select i1 %cmp, i32 %a, i32 %neg
|
|
ret i32 %cond
|
|
}
|
|
|
|
define i32@foo5(i32 %a, i32 %b) nounwind ssp {
|
|
entry:
|
|
; CHECK: foo5
|
|
; CHECK: subs
|
|
; CHECK-NEXT: csneg
|
|
; CHECK-NEXT: ret
|
|
%sub = sub nsw i32 %a, %b
|
|
%cmp = icmp sgt i32 %sub, -1
|
|
%sub3 = sub nsw i32 0, %sub
|
|
%cond = select i1 %cmp, i32 %sub, i32 %sub3
|
|
ret i32 %cond
|
|
}
|
|
|
|
; make sure we can handle branch instruction in optimizeCompare.
|
|
define i32@foo6(i32 %a, i32 %b) nounwind ssp {
|
|
; CHECK: foo6
|
|
; CHECK: b
|
|
%sub = sub nsw i32 %a, %b
|
|
%cmp = icmp sgt i32 %sub, 0
|
|
br i1 %cmp, label %l.if, label %l.else
|
|
|
|
l.if:
|
|
ret i32 1
|
|
|
|
l.else:
|
|
ret i32 %sub
|
|
}
|
|
|
|
; If CPSR is used multiple times and V flag is used, we don't remove cmp.
|
|
define i32 @foo7(i32 %a, i32 %b) nounwind {
|
|
entry:
|
|
; CHECK-LABEL: foo7:
|
|
; CHECK: sub
|
|
; CHECK-next: adds
|
|
; CHECK-next: csneg
|
|
; CHECK-next: b
|
|
%sub = sub nsw i32 %a, %b
|
|
%cmp = icmp sgt i32 %sub, -1
|
|
%sub3 = sub nsw i32 0, %sub
|
|
%cond = select i1 %cmp, i32 %sub, i32 %sub3
|
|
br i1 %cmp, label %if.then, label %if.else
|
|
|
|
if.then:
|
|
%cmp2 = icmp slt i32 %sub, -1
|
|
%sel = select i1 %cmp2, i32 %cond, i32 %a
|
|
ret i32 %sel
|
|
|
|
if.else:
|
|
ret i32 %cond
|
|
}
|
|
|
|
define i32 @foo8(i32 %v, i32 %a, i32 %b) nounwind readnone ssp {
|
|
entry:
|
|
; CHECK-LABEL: foo8:
|
|
; CHECK: cmp w0, #0
|
|
; CHECK: csinv w0, w1, w2, ne
|
|
%tobool = icmp eq i32 %v, 0
|
|
%neg = xor i32 -1, %b
|
|
%cond = select i1 %tobool, i32 %neg, i32 %a
|
|
ret i32 %cond
|
|
}
|
|
|
|
define i32 @foo9(i32 %v) nounwind readnone optsize ssp {
|
|
entry:
|
|
; CHECK-LABEL: foo9:
|
|
; CHECK: cmp w0, #0
|
|
; CHECK: orr w[[REG:[0-9]+]], wzr, #0x4
|
|
; CHECK: csinv w0, w[[REG]], w[[REG]], ne
|
|
%tobool = icmp ne i32 %v, 0
|
|
%cond = select i1 %tobool, i32 4, i32 -5
|
|
ret i32 %cond
|
|
}
|
|
|
|
define i64 @foo10(i64 %v) nounwind readnone optsize ssp {
|
|
entry:
|
|
; CHECK-LABEL: foo10:
|
|
; CHECK: cmp x0, #0
|
|
; CHECK: orr w[[REG:[0-9]+]], wzr, #0x4
|
|
; CHECK: csinv x0, x[[REG]], x[[REG]], ne
|
|
%tobool = icmp ne i64 %v, 0
|
|
%cond = select i1 %tobool, i64 4, i64 -5
|
|
ret i64 %cond
|
|
}
|
|
|
|
define i32 @foo11(i32 %v) nounwind readnone optsize ssp {
|
|
entry:
|
|
; CHECK-LABEL: foo11:
|
|
; CHECK: cmp w0, #0
|
|
; CHECK: orr w[[REG:[0-9]+]], wzr, #0x4
|
|
; CHECK: csneg w0, w[[REG]], w[[REG]], ne
|
|
%tobool = icmp ne i32 %v, 0
|
|
%cond = select i1 %tobool, i32 4, i32 -4
|
|
ret i32 %cond
|
|
}
|
|
|
|
define i64 @foo12(i64 %v) nounwind readnone optsize ssp {
|
|
entry:
|
|
; CHECK-LABEL: foo12:
|
|
; CHECK: cmp x0, #0
|
|
; CHECK: orr w[[REG:[0-9]+]], wzr, #0x4
|
|
; CHECK: csneg x0, x[[REG]], x[[REG]], ne
|
|
%tobool = icmp ne i64 %v, 0
|
|
%cond = select i1 %tobool, i64 4, i64 -4
|
|
ret i64 %cond
|
|
}
|
|
|
|
define i32 @foo13(i32 %v, i32 %a, i32 %b) nounwind readnone optsize ssp {
|
|
entry:
|
|
; CHECK-LABEL: foo13:
|
|
; CHECK: cmp w0, #0
|
|
; CHECK: csneg w0, w1, w2, ne
|
|
%tobool = icmp eq i32 %v, 0
|
|
%sub = sub i32 0, %b
|
|
%cond = select i1 %tobool, i32 %sub, i32 %a
|
|
ret i32 %cond
|
|
}
|
|
|
|
define i64 @foo14(i64 %v, i64 %a, i64 %b) nounwind readnone optsize ssp {
|
|
entry:
|
|
; CHECK-LABEL: foo14:
|
|
; CHECK: cmp x0, #0
|
|
; CHECK: csneg x0, x1, x2, ne
|
|
%tobool = icmp eq i64 %v, 0
|
|
%sub = sub i64 0, %b
|
|
%cond = select i1 %tobool, i64 %sub, i64 %a
|
|
ret i64 %cond
|
|
}
|
|
|
|
define i32 @foo15(i32 %a, i32 %b) nounwind readnone optsize ssp {
|
|
entry:
|
|
; CHECK-LABEL: foo15:
|
|
; CHECK: cmp w0, w1
|
|
; CHECK: orr w[[REG:[0-9]+]], wzr, #0x1
|
|
; CHECK: csinc w0, w[[REG]], w[[REG]], le
|
|
%cmp = icmp sgt i32 %a, %b
|
|
%. = select i1 %cmp, i32 2, i32 1
|
|
ret i32 %.
|
|
}
|
|
|
|
define i32 @foo16(i32 %a, i32 %b) nounwind readnone optsize ssp {
|
|
entry:
|
|
; CHECK-LABEL: foo16:
|
|
; CHECK: cmp w0, w1
|
|
; CHECK: orr w[[REG:[0-9]+]], wzr, #0x1
|
|
; CHECK: csinc w0, w[[REG]], w[[REG]], gt
|
|
%cmp = icmp sgt i32 %a, %b
|
|
%. = select i1 %cmp, i32 1, i32 2
|
|
ret i32 %.
|
|
}
|
|
|
|
define i64 @foo17(i64 %a, i64 %b) nounwind readnone optsize ssp {
|
|
entry:
|
|
; CHECK-LABEL: foo17:
|
|
; CHECK: cmp x0, x1
|
|
; CHECK: orr w[[REG:[0-9]+]], wzr, #0x1
|
|
; CHECK: csinc x0, x[[REG]], x[[REG]], le
|
|
%cmp = icmp sgt i64 %a, %b
|
|
%. = select i1 %cmp, i64 2, i64 1
|
|
ret i64 %.
|
|
}
|
|
|
|
define i64 @foo18(i64 %a, i64 %b) nounwind readnone optsize ssp {
|
|
entry:
|
|
; CHECK-LABEL: foo18:
|
|
; CHECK: cmp x0, x1
|
|
; CHECK: orr w[[REG:[0-9]+]], wzr, #0x1
|
|
; CHECK: csinc x0, x[[REG]], x[[REG]], gt
|
|
%cmp = icmp sgt i64 %a, %b
|
|
%. = select i1 %cmp, i64 1, i64 2
|
|
ret i64 %.
|
|
}
|