2016-12-06 06:40:13 +08:00
|
|
|
; RUN: llc -O0 -aarch64-enable-atomic-cfg-tidy=0 -stop-after=irtranslator -global-isel -verify-machineinstrs %s -o - 2>&1 | FileCheck %s
|
2019-01-29 03:22:29 +08:00
|
|
|
; RUN: llc -O3 -aarch64-enable-atomic-cfg-tidy=0 -stop-after=irtranslator -global-isel -verify-machineinstrs %s -o - 2>&1 | FileCheck %s --check-prefix=O3
|
2016-08-03 03:04:29 +08:00
|
|
|
|
2016-03-08 09:48:08 +08:00
|
|
|
; This file checks that the translation from llvm IR to generic MachineInstr
|
|
|
|
; is correct.
|
|
|
|
target datalayout = "e-m:o-i64:64-i128:128-n32:64-S128"
|
2016-10-14 18:19:40 +08:00
|
|
|
target triple = "aarch64--"
|
2016-03-08 09:48:08 +08:00
|
|
|
|
|
|
|
; Tests for add.
|
2016-07-26 05:01:29 +08:00
|
|
|
; CHECK-LABEL: name: addi64
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[ARG1:%[0-9]+]]:_(s64) = COPY $x0
|
|
|
|
; CHECK-NEXT: [[ARG2:%[0-9]+]]:_(s64) = COPY $x1
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK-NEXT: [[RES:%[0-9]+]]:_(s64) = G_ADD [[ARG1]], [[ARG2]]
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK-NEXT: $x0 = COPY [[RES]]
|
|
|
|
; CHECK-NEXT: RET_ReallyLR implicit $x0
|
2016-03-08 09:48:08 +08:00
|
|
|
define i64 @addi64(i64 %arg1, i64 %arg2) {
|
|
|
|
%res = add i64 %arg1, %arg2
|
|
|
|
ret i64 %res
|
|
|
|
}
|
2016-03-12 01:28:03 +08:00
|
|
|
|
2016-08-05 05:39:44 +08:00
|
|
|
; CHECK-LABEL: name: muli64
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[ARG1:%[0-9]+]]:_(s64) = COPY $x0
|
|
|
|
; CHECK-NEXT: [[ARG2:%[0-9]+]]:_(s64) = COPY $x1
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK-NEXT: [[RES:%[0-9]+]]:_(s64) = G_MUL [[ARG1]], [[ARG2]]
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK-NEXT: $x0 = COPY [[RES]]
|
|
|
|
; CHECK-NEXT: RET_ReallyLR implicit $x0
|
2016-08-05 05:39:44 +08:00
|
|
|
define i64 @muli64(i64 %arg1, i64 %arg2) {
|
|
|
|
%res = mul i64 %arg1, %arg2
|
|
|
|
ret i64 %res
|
|
|
|
}
|
|
|
|
|
2016-07-23 00:59:52 +08:00
|
|
|
; Tests for alloca
|
2016-07-26 05:01:29 +08:00
|
|
|
; CHECK-LABEL: name: allocai64
|
2016-07-23 00:59:52 +08:00
|
|
|
; CHECK: stack:
|
2017-06-06 16:16:19 +08:00
|
|
|
; CHECK-NEXT: - { id: 0, name: ptr1, type: default, offset: 0, size: 8, alignment: 8,
|
2017-09-29 02:52:14 +08:00
|
|
|
; CHECK-NEXT: stack-id: 0, callee-saved-register: '', callee-saved-restored: true,
|
2018-04-26 02:58:06 +08:00
|
|
|
; CHECK-NEXT: debug-info-variable: '', debug-info-expression: '', debug-info-location: '' }
|
2017-06-06 16:16:19 +08:00
|
|
|
; CHECK-NEXT: - { id: 1, name: ptr2, type: default, offset: 0, size: 8, alignment: 1,
|
2017-09-29 02:52:14 +08:00
|
|
|
; CHECK-NEXT: stack-id: 0, callee-saved-register: '', callee-saved-restored: true,
|
2018-04-26 02:58:06 +08:00
|
|
|
; CHECK-NEXT: debug-info-variable: '', debug-info-expression: '', debug-info-location: '' }
|
2017-06-06 16:16:19 +08:00
|
|
|
; CHECK-NEXT: - { id: 2, name: ptr3, type: default, offset: 0, size: 128, alignment: 8,
|
2017-09-29 02:52:14 +08:00
|
|
|
; CHECK-NEXT: stack-id: 0, callee-saved-register: '', callee-saved-restored: true,
|
2018-04-26 02:58:06 +08:00
|
|
|
; CHECK-NEXT: debug-info-variable: '', debug-info-expression: '', debug-info-location: '' }
|
2017-06-06 16:16:19 +08:00
|
|
|
; CHECK-NEXT: - { id: 3, name: ptr4, type: default, offset: 0, size: 1, alignment: 8,
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: %{{[0-9]+}}:_(p0) = G_FRAME_INDEX %stack.0.ptr1
|
|
|
|
; CHECK: %{{[0-9]+}}:_(p0) = G_FRAME_INDEX %stack.1.ptr2
|
|
|
|
; CHECK: %{{[0-9]+}}:_(p0) = G_FRAME_INDEX %stack.2.ptr3
|
|
|
|
; CHECK: %{{[0-9]+}}:_(p0) = G_FRAME_INDEX %stack.3.ptr4
|
2016-07-23 00:59:52 +08:00
|
|
|
define void @allocai64() {
|
|
|
|
%ptr1 = alloca i64
|
|
|
|
%ptr2 = alloca i64, align 1
|
|
|
|
%ptr3 = alloca i64, i32 16
|
2016-07-28 01:47:54 +08:00
|
|
|
%ptr4 = alloca [0 x i64]
|
2016-07-23 00:59:52 +08:00
|
|
|
ret void
|
|
|
|
}
|
|
|
|
|
2016-03-12 01:28:03 +08:00
|
|
|
; Tests for br.
|
2016-07-26 05:01:29 +08:00
|
|
|
; CHECK-LABEL: name: uncondbr
|
2016-03-12 01:28:03 +08:00
|
|
|
; CHECK: body:
|
|
|
|
;
|
2016-12-16 07:32:25 +08:00
|
|
|
; ABI/constant lowering and IR-level entry basic block.
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: bb.{{[0-9]+}}.{{[a-zA-Z0-9.]+}}:
|
2016-03-12 01:28:03 +08:00
|
|
|
;
|
|
|
|
; Make sure we have one successor and only one.
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK-NEXT: successors: %[[BB2:bb.[0-9]+]](0x80000000)
|
2016-03-12 01:28:03 +08:00
|
|
|
;
|
|
|
|
; Check that we emit the correct branch.
|
2017-03-22 07:42:50 +08:00
|
|
|
; CHECK: G_BR %[[BB2]]
|
2016-03-12 01:28:03 +08:00
|
|
|
;
|
|
|
|
; Check that end contains the return instruction.
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: [[END:bb.[0-9]+]].{{[a-zA-Z0-9.]+}}:
|
2016-03-12 01:28:03 +08:00
|
|
|
; CHECK-NEXT: RET_ReallyLR
|
2017-03-22 07:42:50 +08:00
|
|
|
;
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: bb.{{[0-9]+}}.{{[a-zA-Z0-9.]+}}:
|
2017-03-22 07:42:50 +08:00
|
|
|
; CHECK-NEXT: successors: %[[END]](0x80000000)
|
|
|
|
; CHECK: G_BR %[[END]]
|
2016-03-12 01:28:03 +08:00
|
|
|
define void @uncondbr() {
|
2017-03-22 07:42:50 +08:00
|
|
|
entry:
|
|
|
|
br label %bb2
|
|
|
|
end:
|
|
|
|
ret void
|
|
|
|
bb2:
|
|
|
|
br label %end
|
|
|
|
}
|
|
|
|
|
|
|
|
; CHECK-LABEL: name: uncondbr_fallthrough
|
|
|
|
; CHECK: body:
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: bb.{{[0-9]+}}.{{[a-zA-Z0-9.]+}}:
|
|
|
|
; CHECK-NEXT: successors: %[[END:bb.[0-9]+]](0x80000000)
|
2017-03-22 07:42:50 +08:00
|
|
|
; We don't emit a branch here, as we can fallthrough to the successor.
|
|
|
|
; CHECK-NOT: G_BR
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: [[END]].{{[a-zA-Z0-9.]+}}:
|
2017-03-22 07:42:50 +08:00
|
|
|
; CHECK-NEXT: RET_ReallyLR
|
|
|
|
define void @uncondbr_fallthrough() {
|
|
|
|
entry:
|
2016-03-12 01:28:03 +08:00
|
|
|
br label %end
|
|
|
|
end:
|
|
|
|
ret void
|
|
|
|
}
|
2016-06-11 04:50:35 +08:00
|
|
|
|
2016-07-30 01:58:00 +08:00
|
|
|
; Tests for conditional br.
|
|
|
|
; CHECK-LABEL: name: condbr
|
|
|
|
; CHECK: body:
|
|
|
|
;
|
2016-12-16 07:32:25 +08:00
|
|
|
; ABI/constant lowering and IR-level entry basic block.
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: bb.{{[0-9]+}} (%ir-block.{{[0-9]+}}):
|
2016-07-30 01:58:00 +08:00
|
|
|
; Make sure we have two successors
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK-NEXT: successors: %[[TRUE:bb.[0-9]+]](0x40000000),
|
|
|
|
; CHECK: %[[FALSE:bb.[0-9]+]](0x40000000)
|
2016-07-30 01:58:00 +08:00
|
|
|
;
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[ADDR:%.*]]:_(p0) = COPY $x0
|
2016-12-16 07:32:25 +08:00
|
|
|
;
|
2016-07-30 01:58:00 +08:00
|
|
|
; Check that we emit the correct branch.
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[TST:%.*]]:_(s1) = G_LOAD [[ADDR]](p0)
|
2016-09-12 19:20:10 +08:00
|
|
|
; CHECK: G_BRCOND [[TST]](s1), %[[TRUE]]
|
2016-09-09 19:46:34 +08:00
|
|
|
; CHECK: G_BR %[[FALSE]]
|
2016-07-30 01:58:00 +08:00
|
|
|
;
|
|
|
|
; Check that each successor contains the return instruction.
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: [[TRUE]].{{[a-zA-Z0-9.]+}}:
|
2016-07-30 01:58:00 +08:00
|
|
|
; CHECK-NEXT: RET_ReallyLR
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: [[FALSE]].{{[a-zA-Z0-9.]+}}:
|
2016-07-30 01:58:00 +08:00
|
|
|
; CHECK-NEXT: RET_ReallyLR
|
|
|
|
define void @condbr(i1* %tstaddr) {
|
|
|
|
%tst = load i1, i1* %tstaddr
|
|
|
|
br i1 %tst, label %true, label %false
|
|
|
|
true:
|
|
|
|
ret void
|
|
|
|
false:
|
|
|
|
ret void
|
|
|
|
}
|
|
|
|
|
2017-01-05 19:28:51 +08:00
|
|
|
; Tests for switch.
|
|
|
|
; This gets lowered to a very straightforward sequence of comparisons for now.
|
|
|
|
; CHECK-LABEL: name: switch
|
|
|
|
; CHECK: body:
|
|
|
|
;
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: bb.{{[a-zA-Z0-9.]+}}:
|
|
|
|
; CHECK-NEXT: successors: %[[BB_CASE100:bb.[0-9]+]](0x40000000), %[[BB_NOTCASE100_CHECKNEXT:bb.[0-9]+]](0x40000000)
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: %0:_(s32) = COPY $w0
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: %[[reg100:[0-9]+]]:_(s32) = G_CONSTANT i32 100
|
|
|
|
; CHECK: %[[reg200:[0-9]+]]:_(s32) = G_CONSTANT i32 200
|
|
|
|
; CHECK: %[[reg2:[0-9]+]]:_(s32) = G_CONSTANT i32 2
|
2018-08-01 10:17:42 +08:00
|
|
|
; CHECK: %[[reg1:[0-9]+]]:_(s32) = G_CONSTANT i32 1
|
|
|
|
; CHECK: %[[reg0:[0-9]+]]:_(s32) = G_CONSTANT i32 0
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: %[[regicmp100:[0-9]+]]:_(s1) = G_ICMP intpred(eq), %[[reg100]](s32), %0
|
2017-01-05 19:28:51 +08:00
|
|
|
; CHECK: G_BRCOND %[[regicmp100]](s1), %[[BB_CASE100]]
|
|
|
|
; CHECK: G_BR %[[BB_NOTCASE100_CHECKNEXT]]
|
|
|
|
;
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: [[BB_NOTCASE100_CHECKNEXT]].{{[a-zA-Z0-9.]+}}:
|
|
|
|
; CHECK-NEXT: successors: %[[BB_CASE200:bb.[0-9]+]](0x40000000), %[[BB_NOTCASE200_CHECKNEXT:bb.[0-9]+]](0x40000000)
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: %[[regicmp200:[0-9]+]]:_(s1) = G_ICMP intpred(eq), %[[reg200]](s32), %0
|
2017-03-16 02:22:37 +08:00
|
|
|
; CHECK: G_BRCOND %[[regicmp200]](s1), %[[BB_CASE200]]
|
|
|
|
; CHECK: G_BR %[[BB_NOTCASE200_CHECKNEXT]]
|
|
|
|
;
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: [[BB_NOTCASE200_CHECKNEXT]].{{[a-zA-Z0-9.]+}}:
|
|
|
|
; CHECK-NEXT: successors: %[[BB_DEFAULT:bb.[0-9]+]](0x80000000)
|
2017-03-16 02:22:37 +08:00
|
|
|
; CHECK: G_BR %[[BB_DEFAULT]]
|
|
|
|
;
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: [[BB_DEFAULT]].{{[a-zA-Z0-9.]+}}:
|
|
|
|
; CHECK-NEXT: successors: %[[BB_RET:bb.[0-9]+]](0x80000000)
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: %[[regretdefault:[0-9]+]]:_(s32) = G_ADD %0, %[[reg0]]
|
2017-01-05 19:28:51 +08:00
|
|
|
; CHECK: G_BR %[[BB_RET]]
|
|
|
|
;
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: [[BB_CASE100]].{{[a-zA-Z0-9.]+}}:
|
|
|
|
; CHECK-NEXT: successors: %[[BB_RET:bb.[0-9]+]](0x80000000)
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: %[[regretc100:[0-9]+]]:_(s32) = G_ADD %0, %[[reg1]]
|
2017-01-05 19:28:51 +08:00
|
|
|
; CHECK: G_BR %[[BB_RET]]
|
|
|
|
;
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: [[BB_CASE200]].{{[a-zA-Z0-9.]+}}:
|
2017-01-05 19:28:51 +08:00
|
|
|
; CHECK-NEXT: successors: %[[BB_RET]](0x80000000)
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: %[[regretc200:[0-9]+]]:_(s32) = G_ADD %0, %[[reg2]]
|
2017-01-05 19:28:51 +08:00
|
|
|
;
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: [[BB_RET]].{{[a-zA-Z0-9.]+}}:
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK-NEXT: %[[regret:[0-9]+]]:_(s32) = G_PHI %[[regretdefault]](s32), %[[BB_DEFAULT]], %[[regretc100]](s32), %[[BB_CASE100]]
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: $w0 = COPY %[[regret]](s32)
|
|
|
|
; CHECK: RET_ReallyLR implicit $w0
|
2017-03-16 02:22:33 +08:00
|
|
|
;
|
2017-01-05 19:28:51 +08:00
|
|
|
define i32 @switch(i32 %argc) {
|
|
|
|
entry:
|
|
|
|
switch i32 %argc, label %default [
|
|
|
|
i32 100, label %case100
|
|
|
|
i32 200, label %case200
|
|
|
|
]
|
|
|
|
|
|
|
|
default:
|
|
|
|
%tmp0 = add i32 %argc, 0
|
|
|
|
br label %return
|
|
|
|
|
|
|
|
case100:
|
|
|
|
%tmp1 = add i32 %argc, 1
|
|
|
|
br label %return
|
|
|
|
|
|
|
|
case200:
|
|
|
|
%tmp2 = add i32 %argc, 2
|
|
|
|
br label %return
|
|
|
|
|
|
|
|
return:
|
|
|
|
%res = phi i32 [ %tmp0, %default ], [ %tmp1, %case100 ], [ %tmp2, %case200 ]
|
|
|
|
ret i32 %res
|
|
|
|
}
|
|
|
|
|
2017-01-18 06:13:50 +08:00
|
|
|
; The switch lowering code changes the CFG, which means that the original
|
|
|
|
; %entry block is no longer a predecessor for the phi instruction. We need to
|
|
|
|
; use the correct lowered MachineBasicBlock instead.
|
|
|
|
; CHECK-LABEL: name: test_cfg_remap
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: bb.{{[0-9]+.[a-zA-Z0-9.]+}}:
|
|
|
|
; CHECK-NEXT: successors: %{{bb.[0-9]+}}(0x40000000), %[[NOTCASE1_BLOCK:bb.[0-9]+]](0x40000000)
|
|
|
|
; CHECK: [[NOTCASE1_BLOCK]].{{[a-zA-Z0-9.]+}}:
|
|
|
|
; CHECK-NEXT: successors: %{{bb.[0-9]+}}(0x40000000), %[[NOTCASE57_BLOCK:bb.[0-9]+]](0x40000000)
|
|
|
|
; CHECK: [[NOTCASE57_BLOCK]].{{[a-zA-Z0-9.]+}}:
|
|
|
|
; CHECK-NEXT: successors: %[[PHI_BLOCK:bb.[0-9]+]](0x80000000)
|
2017-01-18 06:13:50 +08:00
|
|
|
; CHECK: G_BR %[[PHI_BLOCK]]
|
2017-03-16 02:22:37 +08:00
|
|
|
;
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: [[PHI_BLOCK]].{{[a-zA-Z0-9.]+}}:
|
|
|
|
; CHECK-NEXT: G_PHI %{{.*}}(s32), %[[NOTCASE57_BLOCK:bb.[0-9]+]], %{{.*}}(s32),
|
2017-03-16 02:22:37 +08:00
|
|
|
;
|
2017-01-18 06:13:50 +08:00
|
|
|
define i32 @test_cfg_remap(i32 %in) {
|
|
|
|
entry:
|
|
|
|
switch i32 %in, label %phi.block [i32 1, label %next
|
|
|
|
i32 57, label %other]
|
|
|
|
|
|
|
|
next:
|
|
|
|
br label %phi.block
|
|
|
|
|
|
|
|
other:
|
|
|
|
ret i32 undef
|
|
|
|
|
|
|
|
phi.block:
|
|
|
|
%res = phi i32 [1, %entry], [42, %next]
|
|
|
|
ret i32 %res
|
|
|
|
}
|
|
|
|
|
|
|
|
; CHECK-LABEL: name: test_cfg_remap_multiple_preds
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: G_PHI [[ENTRY:%.*]](s32), %bb.{{[0-9]+}}, [[ENTRY]](s32), %bb.{{[0-9]+}}
|
2017-01-18 06:13:50 +08:00
|
|
|
define i32 @test_cfg_remap_multiple_preds(i32 %in) {
|
|
|
|
entry:
|
|
|
|
switch i32 %in, label %odd [i32 1, label %next
|
|
|
|
i32 57, label %other
|
|
|
|
i32 128, label %phi.block
|
|
|
|
i32 256, label %phi.block]
|
|
|
|
odd:
|
|
|
|
unreachable
|
|
|
|
|
|
|
|
next:
|
|
|
|
br label %phi.block
|
|
|
|
|
|
|
|
other:
|
|
|
|
ret i32 undef
|
|
|
|
|
|
|
|
phi.block:
|
|
|
|
%res = phi i32 [1, %entry], [1, %entry], [42, %next]
|
|
|
|
ret i32 12
|
|
|
|
}
|
|
|
|
|
2017-01-30 17:13:18 +08:00
|
|
|
; Tests for indirect br.
|
|
|
|
; CHECK-LABEL: name: indirectbr
|
|
|
|
; CHECK: body:
|
|
|
|
;
|
|
|
|
; ABI/constant lowering and IR-level entry basic block.
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: bb.{{[0-9]+.[a-zA-Z0-9.]+}}:
|
2017-01-30 17:13:18 +08:00
|
|
|
; Make sure we have one successor
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK-NEXT: successors: %[[BB_L1:bb.[0-9]+]](0x80000000)
|
2017-03-22 07:42:50 +08:00
|
|
|
; CHECK-NOT: G_BR
|
2017-01-30 17:13:18 +08:00
|
|
|
;
|
|
|
|
; Check basic block L1 has 2 successors: BBL1 and BBL2
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: [[BB_L1]].{{[a-zA-Z0-9.]+}} (address-taken):
|
2017-01-30 17:13:18 +08:00
|
|
|
; CHECK-NEXT: successors: %[[BB_L1]](0x40000000),
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: %[[BB_L2:bb.[0-9]+]](0x40000000)
|
2017-01-30 17:13:18 +08:00
|
|
|
; CHECK: G_BRINDIRECT %{{[0-9]+}}(p0)
|
|
|
|
;
|
|
|
|
; Check basic block L2 is the return basic block
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: [[BB_L2]].{{[a-zA-Z0-9.]+}} (address-taken):
|
2017-01-30 17:13:18 +08:00
|
|
|
; CHECK-NEXT: RET_ReallyLR
|
|
|
|
|
|
|
|
@indirectbr.L = internal unnamed_addr constant [3 x i8*] [i8* blockaddress(@indirectbr, %L1), i8* blockaddress(@indirectbr, %L2), i8* null], align 8
|
|
|
|
|
|
|
|
define void @indirectbr() {
|
|
|
|
entry:
|
|
|
|
br label %L1
|
|
|
|
L1: ; preds = %entry, %L1
|
|
|
|
%i = phi i32 [ 0, %entry ], [ %inc, %L1 ]
|
|
|
|
%inc = add i32 %i, 1
|
|
|
|
%idxprom = zext i32 %i to i64
|
|
|
|
%arrayidx = getelementptr inbounds [3 x i8*], [3 x i8*]* @indirectbr.L, i64 0, i64 %idxprom
|
|
|
|
%brtarget = load i8*, i8** %arrayidx, align 8
|
|
|
|
indirectbr i8* %brtarget, [label %L1, label %L2]
|
|
|
|
L2: ; preds = %L1
|
|
|
|
ret void
|
|
|
|
}
|
|
|
|
|
2016-06-11 04:50:35 +08:00
|
|
|
; Tests for or.
|
2016-07-26 05:01:29 +08:00
|
|
|
; CHECK-LABEL: name: ori64
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[ARG1:%[0-9]+]]:_(s64) = COPY $x0
|
|
|
|
; CHECK-NEXT: [[ARG2:%[0-9]+]]:_(s64) = COPY $x1
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK-NEXT: [[RES:%[0-9]+]]:_(s64) = G_OR [[ARG1]], [[ARG2]]
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK-NEXT: $x0 = COPY [[RES]]
|
|
|
|
; CHECK-NEXT: RET_ReallyLR implicit $x0
|
2016-06-11 04:50:35 +08:00
|
|
|
define i64 @ori64(i64 %arg1, i64 %arg2) {
|
|
|
|
%res = or i64 %arg1, %arg2
|
|
|
|
ret i64 %res
|
|
|
|
}
|
|
|
|
|
2016-07-26 05:01:29 +08:00
|
|
|
; CHECK-LABEL: name: ori32
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[ARG1:%[0-9]+]]:_(s32) = COPY $w0
|
|
|
|
; CHECK-NEXT: [[ARG2:%[0-9]+]]:_(s32) = COPY $w1
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK-NEXT: [[RES:%[0-9]+]]:_(s32) = G_OR [[ARG1]], [[ARG2]]
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK-NEXT: $w0 = COPY [[RES]]
|
|
|
|
; CHECK-NEXT: RET_ReallyLR implicit $w0
|
2016-06-11 04:50:35 +08:00
|
|
|
define i32 @ori32(i32 %arg1, i32 %arg2) {
|
|
|
|
%res = or i32 %arg1, %arg2
|
|
|
|
ret i32 %res
|
|
|
|
}
|
2016-07-21 23:50:42 +08:00
|
|
|
|
2016-07-30 00:56:20 +08:00
|
|
|
; Tests for xor.
|
|
|
|
; CHECK-LABEL: name: xori64
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[ARG1:%[0-9]+]]:_(s64) = COPY $x0
|
|
|
|
; CHECK-NEXT: [[ARG2:%[0-9]+]]:_(s64) = COPY $x1
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK-NEXT: [[RES:%[0-9]+]]:_(s64) = G_XOR [[ARG1]], [[ARG2]]
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK-NEXT: $x0 = COPY [[RES]]
|
|
|
|
; CHECK-NEXT: RET_ReallyLR implicit $x0
|
2016-07-30 00:56:20 +08:00
|
|
|
define i64 @xori64(i64 %arg1, i64 %arg2) {
|
|
|
|
%res = xor i64 %arg1, %arg2
|
|
|
|
ret i64 %res
|
|
|
|
}
|
|
|
|
|
|
|
|
; CHECK-LABEL: name: xori32
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[ARG1:%[0-9]+]]:_(s32) = COPY $w0
|
|
|
|
; CHECK-NEXT: [[ARG2:%[0-9]+]]:_(s32) = COPY $w1
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK-NEXT: [[RES:%[0-9]+]]:_(s32) = G_XOR [[ARG1]], [[ARG2]]
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK-NEXT: $w0 = COPY [[RES]]
|
|
|
|
; CHECK-NEXT: RET_ReallyLR implicit $w0
|
2016-07-30 00:56:20 +08:00
|
|
|
define i32 @xori32(i32 %arg1, i32 %arg2) {
|
|
|
|
%res = xor i32 %arg1, %arg2
|
|
|
|
ret i32 %res
|
|
|
|
}
|
|
|
|
|
2016-07-21 23:50:42 +08:00
|
|
|
; Tests for and.
|
2016-07-26 05:01:29 +08:00
|
|
|
; CHECK-LABEL: name: andi64
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[ARG1:%[0-9]+]]:_(s64) = COPY $x0
|
|
|
|
; CHECK-NEXT: [[ARG2:%[0-9]+]]:_(s64) = COPY $x1
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK-NEXT: [[RES:%[0-9]+]]:_(s64) = G_AND [[ARG1]], [[ARG2]]
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK-NEXT: $x0 = COPY [[RES]]
|
|
|
|
; CHECK-NEXT: RET_ReallyLR implicit $x0
|
2016-07-21 23:50:42 +08:00
|
|
|
define i64 @andi64(i64 %arg1, i64 %arg2) {
|
|
|
|
%res = and i64 %arg1, %arg2
|
|
|
|
ret i64 %res
|
|
|
|
}
|
|
|
|
|
2016-07-26 05:01:29 +08:00
|
|
|
; CHECK-LABEL: name: andi32
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[ARG1:%[0-9]+]]:_(s32) = COPY $w0
|
|
|
|
; CHECK-NEXT: [[ARG2:%[0-9]+]]:_(s32) = COPY $w1
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK-NEXT: [[RES:%[0-9]+]]:_(s32) = G_AND [[ARG1]], [[ARG2]]
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK-NEXT: $w0 = COPY [[RES]]
|
|
|
|
; CHECK-NEXT: RET_ReallyLR implicit $w0
|
2016-07-21 23:50:42 +08:00
|
|
|
define i32 @andi32(i32 %arg1, i32 %arg2) {
|
|
|
|
%res = and i32 %arg1, %arg2
|
|
|
|
ret i32 %res
|
|
|
|
}
|
2016-07-22 01:26:50 +08:00
|
|
|
|
|
|
|
; Tests for sub.
|
2016-07-26 05:01:29 +08:00
|
|
|
; CHECK-LABEL: name: subi64
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[ARG1:%[0-9]+]]:_(s64) = COPY $x0
|
|
|
|
; CHECK-NEXT: [[ARG2:%[0-9]+]]:_(s64) = COPY $x1
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK-NEXT: [[RES:%[0-9]+]]:_(s64) = G_SUB [[ARG1]], [[ARG2]]
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK-NEXT: $x0 = COPY [[RES]]
|
|
|
|
; CHECK-NEXT: RET_ReallyLR implicit $x0
|
2016-07-22 01:26:50 +08:00
|
|
|
define i64 @subi64(i64 %arg1, i64 %arg2) {
|
|
|
|
%res = sub i64 %arg1, %arg2
|
|
|
|
ret i64 %res
|
|
|
|
}
|
|
|
|
|
2016-07-26 05:01:29 +08:00
|
|
|
; CHECK-LABEL: name: subi32
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[ARG1:%[0-9]+]]:_(s32) = COPY $w0
|
|
|
|
; CHECK-NEXT: [[ARG2:%[0-9]+]]:_(s32) = COPY $w1
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK-NEXT: [[RES:%[0-9]+]]:_(s32) = G_SUB [[ARG1]], [[ARG2]]
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK-NEXT: $w0 = COPY [[RES]]
|
|
|
|
; CHECK-NEXT: RET_ReallyLR implicit $w0
|
2016-07-22 01:26:50 +08:00
|
|
|
define i32 @subi32(i32 %arg1, i32 %arg2) {
|
|
|
|
%res = sub i32 %arg1, %arg2
|
|
|
|
ret i32 %res
|
|
|
|
}
|
2016-07-26 05:01:29 +08:00
|
|
|
|
|
|
|
; CHECK-LABEL: name: ptrtoint
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[ARG1:%[0-9]+]]:_(p0) = COPY $x0
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[RES:%[0-9]+]]:_(s64) = G_PTRTOINT [[ARG1]]
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: $x0 = COPY [[RES]]
|
|
|
|
; CHECK: RET_ReallyLR implicit $x0
|
2016-07-26 05:01:29 +08:00
|
|
|
define i64 @ptrtoint(i64* %a) {
|
|
|
|
%val = ptrtoint i64* %a to i64
|
|
|
|
ret i64 %val
|
|
|
|
}
|
|
|
|
|
|
|
|
; CHECK-LABEL: name: inttoptr
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[ARG1:%[0-9]+]]:_(s64) = COPY $x0
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[RES:%[0-9]+]]:_(p0) = G_INTTOPTR [[ARG1]]
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: $x0 = COPY [[RES]]
|
|
|
|
; CHECK: RET_ReallyLR implicit $x0
|
2016-07-26 05:01:29 +08:00
|
|
|
define i64* @inttoptr(i64 %a) {
|
|
|
|
%val = inttoptr i64 %a to i64*
|
|
|
|
ret i64* %val
|
|
|
|
}
|
|
|
|
|
|
|
|
; CHECK-LABEL: name: trivial_bitcast
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[ARG1:%[0-9]+]]:_(p0) = COPY $x0
|
|
|
|
; CHECK: $x0 = COPY [[ARG1]]
|
|
|
|
; CHECK: RET_ReallyLR implicit $x0
|
2016-07-26 05:01:29 +08:00
|
|
|
define i64* @trivial_bitcast(i8* %a) {
|
|
|
|
%val = bitcast i8* %a to i64*
|
|
|
|
ret i64* %val
|
|
|
|
}
|
|
|
|
|
2016-08-11 00:51:14 +08:00
|
|
|
; CHECK-LABEL: name: trivial_bitcast_with_copy
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[A:%[0-9]+]]:_(p0) = COPY $x0
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: G_BR %[[CAST:bb\.[0-9]+]]
|
2016-08-11 00:51:14 +08:00
|
|
|
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: [[END:bb\.[0-9]+]].{{[a-zA-Z0-9.]+}}:
|
2018-08-01 10:17:42 +08:00
|
|
|
; CHECK: $x0 = COPY [[A]]
|
2017-03-16 02:22:33 +08:00
|
|
|
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: [[CAST]].{{[a-zA-Z0-9.]+}}:
|
2017-03-16 02:22:33 +08:00
|
|
|
; CHECK: G_BR %[[END]]
|
2016-08-11 00:51:14 +08:00
|
|
|
define i64* @trivial_bitcast_with_copy(i8* %a) {
|
|
|
|
br label %cast
|
|
|
|
|
|
|
|
end:
|
|
|
|
ret i64* %val
|
|
|
|
|
|
|
|
cast:
|
|
|
|
%val = bitcast i8* %a to i64*
|
|
|
|
br label %end
|
|
|
|
}
|
|
|
|
|
2016-07-26 05:01:29 +08:00
|
|
|
; CHECK-LABEL: name: bitcast
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[ARG1:%[0-9]+]]:_(s64) = COPY $x0
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[RES1:%[0-9]+]]:_(<2 x s32>) = G_BITCAST [[ARG1]]
|
|
|
|
; CHECK: [[RES2:%[0-9]+]]:_(s64) = G_BITCAST [[RES1]]
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: $x0 = COPY [[RES2]]
|
|
|
|
; CHECK: RET_ReallyLR implicit $x0
|
2016-07-26 05:01:29 +08:00
|
|
|
define i64 @bitcast(i64 %a) {
|
|
|
|
%res1 = bitcast i64 %a to <2 x i32>
|
|
|
|
%res2 = bitcast <2 x i32> %res1 to i64
|
|
|
|
ret i64 %res2
|
|
|
|
}
|
2016-07-27 04:23:26 +08:00
|
|
|
|
2018-06-23 04:58:51 +08:00
|
|
|
; CHECK-LABEL: name: addrspacecast
|
|
|
|
; CHECK: [[ARG1:%[0-9]+]]:_(p1) = COPY $x0
|
|
|
|
; CHECK: [[RES1:%[0-9]+]]:_(p2) = G_ADDRSPACE_CAST [[ARG1]]
|
|
|
|
; CHECK: [[RES2:%[0-9]+]]:_(p0) = G_ADDRSPACE_CAST [[RES1]]
|
|
|
|
; CHECK: $x0 = COPY [[RES2]]
|
|
|
|
; CHECK: RET_ReallyLR implicit $x0
|
|
|
|
define i64* @addrspacecast(i32 addrspace(1)* %a) {
|
|
|
|
%res1 = addrspacecast i32 addrspace(1)* %a to i64 addrspace(2)*
|
|
|
|
%res2 = addrspacecast i64 addrspace(2)* %res1 to i64*
|
|
|
|
ret i64* %res2
|
|
|
|
}
|
|
|
|
|
2016-08-05 02:35:17 +08:00
|
|
|
; CHECK-LABEL: name: trunc
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[ARG1:%[0-9]+]]:_(s64) = COPY $x0
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[VEC:%[0-9]+]]:_(<4 x s32>) = G_LOAD
|
|
|
|
; CHECK: [[RES1:%[0-9]+]]:_(s8) = G_TRUNC [[ARG1]]
|
|
|
|
; CHECK: [[RES2:%[0-9]+]]:_(<4 x s16>) = G_TRUNC [[VEC]]
|
2016-08-05 02:35:17 +08:00
|
|
|
define void @trunc(i64 %a) {
|
|
|
|
%vecptr = alloca <4 x i32>
|
|
|
|
%vec = load <4 x i32>, <4 x i32>* %vecptr
|
|
|
|
%res1 = trunc i64 %a to i8
|
|
|
|
%res2 = trunc <4 x i32> %vec to <4 x i16>
|
|
|
|
ret void
|
|
|
|
}
|
|
|
|
|
2016-07-27 04:23:26 +08:00
|
|
|
; CHECK-LABEL: name: load
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[ADDR:%[0-9]+]]:_(p0) = COPY $x0
|
|
|
|
; CHECK: [[ADDR42:%[0-9]+]]:_(p42) = COPY $x1
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[VAL1:%[0-9]+]]:_(s64) = G_LOAD [[ADDR]](p0) :: (load 8 from %ir.addr, align 16)
|
2018-01-26 19:47:28 +08:00
|
|
|
; CHECK: [[VAL2:%[0-9]+]]:_(s64) = G_LOAD [[ADDR42]](p42) :: (load 8 from %ir.addr42, addrspace 42)
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[SUM2:%.*]]:_(s64) = G_ADD [[VAL1]], [[VAL2]]
|
|
|
|
; CHECK: [[VAL3:%[0-9]+]]:_(s64) = G_LOAD [[ADDR]](p0) :: (volatile load 8 from %ir.addr)
|
|
|
|
; CHECK: [[SUM3:%[0-9]+]]:_(s64) = G_ADD [[SUM2]], [[VAL3]]
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: $x0 = COPY [[SUM3]]
|
|
|
|
; CHECK: RET_ReallyLR implicit $x0
|
2016-07-27 04:23:26 +08:00
|
|
|
define i64 @load(i64* %addr, i64 addrspace(42)* %addr42) {
|
|
|
|
%val1 = load i64, i64* %addr, align 16
|
2016-10-19 23:55:06 +08:00
|
|
|
|
2016-07-27 04:23:26 +08:00
|
|
|
%val2 = load i64, i64 addrspace(42)* %addr42
|
2016-10-19 23:55:06 +08:00
|
|
|
%sum2 = add i64 %val1, %val2
|
|
|
|
|
|
|
|
%val3 = load volatile i64, i64* %addr
|
|
|
|
%sum3 = add i64 %sum2, %val3
|
|
|
|
ret i64 %sum3
|
2016-07-27 04:23:26 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
; CHECK-LABEL: name: store
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[ADDR:%[0-9]+]]:_(p0) = COPY $x0
|
|
|
|
; CHECK: [[ADDR42:%[0-9]+]]:_(p42) = COPY $x1
|
|
|
|
; CHECK: [[VAL1:%[0-9]+]]:_(s64) = COPY $x2
|
|
|
|
; CHECK: [[VAL2:%[0-9]+]]:_(s64) = COPY $x3
|
2016-09-12 19:20:10 +08:00
|
|
|
; CHECK: G_STORE [[VAL1]](s64), [[ADDR]](p0) :: (store 8 into %ir.addr, align 16)
|
2018-01-26 19:47:28 +08:00
|
|
|
; CHECK: G_STORE [[VAL2]](s64), [[ADDR42]](p42) :: (store 8 into %ir.addr42, addrspace 42)
|
2016-10-19 23:55:06 +08:00
|
|
|
; CHECK: G_STORE [[VAL1]](s64), [[ADDR]](p0) :: (volatile store 8 into %ir.addr)
|
2016-07-27 04:23:26 +08:00
|
|
|
; CHECK: RET_ReallyLR
|
|
|
|
define void @store(i64* %addr, i64 addrspace(42)* %addr42, i64 %val1, i64 %val2) {
|
|
|
|
store i64 %val1, i64* %addr, align 16
|
|
|
|
store i64 %val2, i64 addrspace(42)* %addr42
|
2016-10-19 23:55:06 +08:00
|
|
|
store volatile i64 %val1, i64* %addr
|
2016-07-27 04:23:26 +08:00
|
|
|
%sum = add i64 %val1, %val2
|
|
|
|
ret void
|
|
|
|
}
|
2016-07-30 06:32:36 +08:00
|
|
|
|
|
|
|
; CHECK-LABEL: name: intrinsics
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[CUR:%[0-9]+]]:_(s32) = COPY $w0
|
|
|
|
; CHECK: [[BITS:%[0-9]+]]:_(s32) = COPY $w1
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[CREG:%[0-9]+]]:_(s32) = G_CONSTANT i32 0
|
|
|
|
; CHECK: [[PTR:%[0-9]+]]:_(p0) = G_INTRINSIC intrinsic(@llvm.returnaddress), [[CREG]]
|
|
|
|
; CHECK: [[PTR_VEC:%[0-9]+]]:_(p0) = G_FRAME_INDEX %stack.0.ptr.vec
|
|
|
|
; CHECK: [[VEC:%[0-9]+]]:_(<8 x s8>) = G_LOAD [[PTR_VEC]]
|
2016-09-12 19:20:10 +08:00
|
|
|
; CHECK: G_INTRINSIC_W_SIDE_EFFECTS intrinsic(@llvm.aarch64.neon.st2), [[VEC]](<8 x s8>), [[VEC]](<8 x s8>), [[PTR]](p0)
|
2016-07-30 06:32:36 +08:00
|
|
|
; CHECK: RET_ReallyLR
|
|
|
|
declare i8* @llvm.returnaddress(i32)
|
|
|
|
declare void @llvm.aarch64.neon.st2.v8i8.p0i8(<8 x i8>, <8 x i8>, i8*)
|
|
|
|
declare { <8 x i8>, <8 x i8> } @llvm.aarch64.neon.ld2.v8i8.p0v8i8(<8 x i8>*)
|
|
|
|
define void @intrinsics(i32 %cur, i32 %bits) {
|
|
|
|
%ptr = call i8* @llvm.returnaddress(i32 0)
|
|
|
|
%ptr.vec = alloca <8 x i8>
|
|
|
|
%vec = load <8 x i8>, <8 x i8>* %ptr.vec
|
|
|
|
call void @llvm.aarch64.neon.st2.v8i8.p0i8(<8 x i8> %vec, <8 x i8> %vec, i8* %ptr)
|
|
|
|
ret void
|
|
|
|
}
|
2016-07-30 06:41:55 +08:00
|
|
|
|
2016-08-06 01:16:40 +08:00
|
|
|
; CHECK-LABEL: name: test_phi
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: G_BRCOND {{%.*}}, %[[TRUE:bb\.[0-9]+]]
|
|
|
|
; CHECK: G_BR %[[FALSE:bb\.[0-9]+]]
|
2016-08-06 01:16:40 +08:00
|
|
|
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: [[TRUE]].{{[a-zA-Z0-9.]+}}:
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[RES1:%[0-9]+]]:_(s32) = G_LOAD
|
2016-08-06 01:16:40 +08:00
|
|
|
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: [[FALSE]].{{[a-zA-Z0-9.]+}}:
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[RES2:%[0-9]+]]:_(s32) = G_LOAD
|
2016-08-06 01:16:40 +08:00
|
|
|
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[RES:%[0-9]+]]:_(s32) = G_PHI [[RES1]](s32), %[[TRUE]], [[RES2]](s32), %[[FALSE]]
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: $w0 = COPY [[RES]]
|
2016-08-06 01:16:40 +08:00
|
|
|
define i32 @test_phi(i32* %addr1, i32* %addr2, i1 %tst) {
|
|
|
|
br i1 %tst, label %true, label %false
|
|
|
|
|
|
|
|
true:
|
|
|
|
%res1 = load i32, i32* %addr1
|
|
|
|
br label %end
|
|
|
|
|
|
|
|
false:
|
|
|
|
%res2 = load i32, i32* %addr2
|
|
|
|
br label %end
|
|
|
|
|
|
|
|
end:
|
|
|
|
%res = phi i32 [%res1, %true], [%res2, %false]
|
|
|
|
ret i32 %res
|
|
|
|
}
|
2016-08-06 01:50:36 +08:00
|
|
|
|
|
|
|
; CHECK-LABEL: name: unreachable
|
|
|
|
; CHECK: G_ADD
|
|
|
|
; CHECK-NEXT: {{^$}}
|
|
|
|
; CHECK-NEXT: ...
|
|
|
|
define void @unreachable(i32 %a) {
|
|
|
|
%sum = add i32 %a, %a
|
|
|
|
unreachable
|
|
|
|
}
|
2016-08-10 05:28:04 +08:00
|
|
|
|
|
|
|
; It's important that constants are after argument passing, but before the
|
|
|
|
; rest of the entry block.
|
|
|
|
; CHECK-LABEL: name: constant_int
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[IN:%[0-9]+]]:_(s32) = COPY $w0
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[ONE:%[0-9]+]]:_(s32) = G_CONSTANT i32 1
|
2016-08-10 05:28:04 +08:00
|
|
|
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: bb.{{[0-9]+}}.{{[a-zA-Z0-9.]+}}:
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[SUM1:%[0-9]+]]:_(s32) = G_ADD [[IN]], [[ONE]]
|
|
|
|
; CHECK: [[SUM2:%[0-9]+]]:_(s32) = G_ADD [[IN]], [[ONE]]
|
|
|
|
; CHECK: [[RES:%[0-9]+]]:_(s32) = G_ADD [[SUM1]], [[SUM2]]
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: $w0 = COPY [[RES]]
|
2016-08-10 05:28:04 +08:00
|
|
|
|
|
|
|
define i32 @constant_int(i32 %in) {
|
|
|
|
br label %next
|
|
|
|
|
|
|
|
next:
|
|
|
|
%sum1 = add i32 %in, 1
|
|
|
|
%sum2 = add i32 %in, 1
|
|
|
|
%res = add i32 %sum1, %sum2
|
|
|
|
ret i32 %res
|
|
|
|
}
|
|
|
|
|
|
|
|
; CHECK-LABEL: name: constant_int_start
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[TWO:%[0-9]+]]:_(s32) = G_CONSTANT i32 2
|
|
|
|
; CHECK: [[ANSWER:%[0-9]+]]:_(s32) = G_CONSTANT i32 42
|
|
|
|
; CHECK: [[RES:%[0-9]+]]:_(s32) = G_ADD [[TWO]], [[ANSWER]]
|
2016-08-10 05:28:04 +08:00
|
|
|
define i32 @constant_int_start() {
|
|
|
|
%res = add i32 2, 42
|
|
|
|
ret i32 %res
|
|
|
|
}
|
2016-08-10 07:01:30 +08:00
|
|
|
|
|
|
|
; CHECK-LABEL: name: test_undef
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[UNDEF:%[0-9]+]]:_(s32) = G_IMPLICIT_DEF
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: $w0 = COPY [[UNDEF]]
|
2016-08-10 07:01:30 +08:00
|
|
|
define i32 @test_undef() {
|
|
|
|
ret i32 undef
|
|
|
|
}
|
2016-08-11 07:02:41 +08:00
|
|
|
|
|
|
|
; CHECK-LABEL: name: test_constant_inttoptr
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[ONE:%[0-9]+]]:_(s64) = G_CONSTANT i64 1
|
|
|
|
; CHECK: [[PTR:%[0-9]+]]:_(p0) = G_INTTOPTR [[ONE]]
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: $x0 = COPY [[PTR]]
|
2016-08-11 07:02:41 +08:00
|
|
|
define i8* @test_constant_inttoptr() {
|
|
|
|
ret i8* inttoptr(i64 1 to i8*)
|
|
|
|
}
|
2016-08-12 00:21:29 +08:00
|
|
|
|
|
|
|
; This failed purely because the Constant -> VReg map was kept across
|
|
|
|
; functions, so reuse the "i64 1" from above.
|
|
|
|
; CHECK-LABEL: name: test_reused_constant
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[ONE:%[0-9]+]]:_(s64) = G_CONSTANT i64 1
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: $x0 = COPY [[ONE]]
|
2016-08-12 00:21:29 +08:00
|
|
|
define i64 @test_reused_constant() {
|
|
|
|
ret i64 1
|
|
|
|
}
|
2016-08-12 05:01:10 +08:00
|
|
|
|
|
|
|
; CHECK-LABEL: name: test_sext
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[IN:%[0-9]+]]:_(s32) = COPY $w0
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[RES:%[0-9]+]]:_(s64) = G_SEXT [[IN]]
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: $x0 = COPY [[RES]]
|
2016-08-12 05:01:10 +08:00
|
|
|
define i64 @test_sext(i32 %in) {
|
|
|
|
%res = sext i32 %in to i64
|
|
|
|
ret i64 %res
|
|
|
|
}
|
|
|
|
|
|
|
|
; CHECK-LABEL: name: test_zext
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[IN:%[0-9]+]]:_(s32) = COPY $w0
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[RES:%[0-9]+]]:_(s64) = G_ZEXT [[IN]]
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: $x0 = COPY [[RES]]
|
2016-08-12 05:01:10 +08:00
|
|
|
define i64 @test_zext(i32 %in) {
|
|
|
|
%res = zext i32 %in to i64
|
|
|
|
ret i64 %res
|
|
|
|
}
|
2016-08-12 05:01:13 +08:00
|
|
|
|
|
|
|
; CHECK-LABEL: name: test_shl
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[ARG1:%[0-9]+]]:_(s32) = COPY $w0
|
|
|
|
; CHECK-NEXT: [[ARG2:%[0-9]+]]:_(s32) = COPY $w1
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK-NEXT: [[RES:%[0-9]+]]:_(s32) = G_SHL [[ARG1]], [[ARG2]]
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK-NEXT: $w0 = COPY [[RES]]
|
|
|
|
; CHECK-NEXT: RET_ReallyLR implicit $w0
|
2016-08-12 05:01:13 +08:00
|
|
|
define i32 @test_shl(i32 %arg1, i32 %arg2) {
|
|
|
|
%res = shl i32 %arg1, %arg2
|
|
|
|
ret i32 %res
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
; CHECK-LABEL: name: test_lshr
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[ARG1:%[0-9]+]]:_(s32) = COPY $w0
|
|
|
|
; CHECK-NEXT: [[ARG2:%[0-9]+]]:_(s32) = COPY $w1
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK-NEXT: [[RES:%[0-9]+]]:_(s32) = G_LSHR [[ARG1]], [[ARG2]]
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK-NEXT: $w0 = COPY [[RES]]
|
|
|
|
; CHECK-NEXT: RET_ReallyLR implicit $w0
|
2016-08-12 05:01:13 +08:00
|
|
|
define i32 @test_lshr(i32 %arg1, i32 %arg2) {
|
|
|
|
%res = lshr i32 %arg1, %arg2
|
|
|
|
ret i32 %res
|
|
|
|
}
|
|
|
|
|
|
|
|
; CHECK-LABEL: name: test_ashr
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[ARG1:%[0-9]+]]:_(s32) = COPY $w0
|
|
|
|
; CHECK-NEXT: [[ARG2:%[0-9]+]]:_(s32) = COPY $w1
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK-NEXT: [[RES:%[0-9]+]]:_(s32) = G_ASHR [[ARG1]], [[ARG2]]
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK-NEXT: $w0 = COPY [[RES]]
|
|
|
|
; CHECK-NEXT: RET_ReallyLR implicit $w0
|
2016-08-12 05:01:13 +08:00
|
|
|
define i32 @test_ashr(i32 %arg1, i32 %arg2) {
|
|
|
|
%res = ashr i32 %arg1, %arg2
|
|
|
|
ret i32 %res
|
|
|
|
}
|
2016-08-12 05:40:55 +08:00
|
|
|
|
2016-08-18 23:17:01 +08:00
|
|
|
; CHECK-LABEL: name: test_sdiv
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[ARG1:%[0-9]+]]:_(s32) = COPY $w0
|
|
|
|
; CHECK-NEXT: [[ARG2:%[0-9]+]]:_(s32) = COPY $w1
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK-NEXT: [[RES:%[0-9]+]]:_(s32) = G_SDIV [[ARG1]], [[ARG2]]
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK-NEXT: $w0 = COPY [[RES]]
|
|
|
|
; CHECK-NEXT: RET_ReallyLR implicit $w0
|
2016-08-18 23:17:01 +08:00
|
|
|
define i32 @test_sdiv(i32 %arg1, i32 %arg2) {
|
|
|
|
%res = sdiv i32 %arg1, %arg2
|
|
|
|
ret i32 %res
|
|
|
|
}
|
|
|
|
|
|
|
|
; CHECK-LABEL: name: test_udiv
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[ARG1:%[0-9]+]]:_(s32) = COPY $w0
|
|
|
|
; CHECK-NEXT: [[ARG2:%[0-9]+]]:_(s32) = COPY $w1
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK-NEXT: [[RES:%[0-9]+]]:_(s32) = G_UDIV [[ARG1]], [[ARG2]]
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK-NEXT: $w0 = COPY [[RES]]
|
|
|
|
; CHECK-NEXT: RET_ReallyLR implicit $w0
|
2016-08-18 23:17:01 +08:00
|
|
|
define i32 @test_udiv(i32 %arg1, i32 %arg2) {
|
|
|
|
%res = udiv i32 %arg1, %arg2
|
|
|
|
ret i32 %res
|
|
|
|
}
|
|
|
|
|
|
|
|
; CHECK-LABEL: name: test_srem
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[ARG1:%[0-9]+]]:_(s32) = COPY $w0
|
|
|
|
; CHECK-NEXT: [[ARG2:%[0-9]+]]:_(s32) = COPY $w1
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK-NEXT: [[RES:%[0-9]+]]:_(s32) = G_SREM [[ARG1]], [[ARG2]]
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK-NEXT: $w0 = COPY [[RES]]
|
|
|
|
; CHECK-NEXT: RET_ReallyLR implicit $w0
|
2016-08-18 23:17:01 +08:00
|
|
|
define i32 @test_srem(i32 %arg1, i32 %arg2) {
|
|
|
|
%res = srem i32 %arg1, %arg2
|
|
|
|
ret i32 %res
|
|
|
|
}
|
|
|
|
|
|
|
|
; CHECK-LABEL: name: test_urem
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[ARG1:%[0-9]+]]:_(s32) = COPY $w0
|
|
|
|
; CHECK-NEXT: [[ARG2:%[0-9]+]]:_(s32) = COPY $w1
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK-NEXT: [[RES:%[0-9]+]]:_(s32) = G_UREM [[ARG1]], [[ARG2]]
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK-NEXT: $w0 = COPY [[RES]]
|
|
|
|
; CHECK-NEXT: RET_ReallyLR implicit $w0
|
2016-08-18 23:17:01 +08:00
|
|
|
define i32 @test_urem(i32 %arg1, i32 %arg2) {
|
|
|
|
%res = urem i32 %arg1, %arg2
|
|
|
|
ret i32 %res
|
|
|
|
}
|
|
|
|
|
2016-08-12 05:40:55 +08:00
|
|
|
; CHECK-LABEL: name: test_constant_null
|
2018-03-23 01:31:38 +08:00
|
|
|
; CHECK: [[ZERO:%[0-9]+]]:_(s64) = G_CONSTANT i64 0
|
|
|
|
; CHECK: [[NULL:%[0-9]+]]:_(p0) = G_INTTOPTR [[ZERO]]
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: $x0 = COPY [[NULL]]
|
2016-08-12 05:40:55 +08:00
|
|
|
define i8* @test_constant_null() {
|
|
|
|
ret i8* null
|
|
|
|
}
|
2016-08-16 05:13:17 +08:00
|
|
|
|
|
|
|
; CHECK-LABEL: name: test_struct_memops
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[ADDR:%[0-9]+]]:_(p0) = COPY $x0
|
2018-05-16 18:32:02 +08:00
|
|
|
; CHECK: [[VAL1:%[0-9]+]]:_(s8) = G_LOAD %0(p0) :: (load 1 from %ir.addr, align 4)
|
|
|
|
; CHECK: [[CST1:%[0-9]+]]:_(s64) = G_CONSTANT i64 4
|
|
|
|
; CHECK: [[GEP1:%[0-9]+]]:_(p0) = G_GEP [[ADDR]], [[CST1]](s64)
|
|
|
|
; CHECK: [[VAL2:%[0-9]+]]:_(s32) = G_LOAD [[GEP1]](p0) :: (load 4 from %ir.addr + 4)
|
|
|
|
; CHECK: G_STORE [[VAL1]](s8), [[ADDR]](p0) :: (store 1 into %ir.addr, align 4)
|
|
|
|
; CHECK: [[CST2:%[0-9]+]]:_(s64) = G_CONSTANT i64 4
|
|
|
|
; CHECK: [[GEP2:%[0-9]+]]:_(p0) = G_GEP [[ADDR]], [[CST2]](s64)
|
|
|
|
; CHECK: G_STORE [[VAL2]](s32), [[GEP2]](p0) :: (store 4 into %ir.addr + 4)
|
2016-08-16 05:13:17 +08:00
|
|
|
define void @test_struct_memops({ i8, i32 }* %addr) {
|
|
|
|
%val = load { i8, i32 }, { i8, i32 }* %addr
|
|
|
|
store { i8, i32 } %val, { i8, i32 }* %addr
|
|
|
|
ret void
|
|
|
|
}
|
|
|
|
|
|
|
|
; CHECK-LABEL: name: test_i1_memops
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[ADDR:%[0-9]+]]:_(p0) = COPY $x0
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[VAL:%[0-9]+]]:_(s1) = G_LOAD [[ADDR]](p0) :: (load 1 from %ir.addr)
|
2016-09-12 19:20:10 +08:00
|
|
|
; CHECK: G_STORE [[VAL]](s1), [[ADDR]](p0) :: (store 1 into %ir.addr)
|
2016-08-16 05:13:17 +08:00
|
|
|
define void @test_i1_memops(i1* %addr) {
|
|
|
|
%val = load i1, i1* %addr
|
|
|
|
store i1 %val, i1* %addr
|
|
|
|
ret void
|
|
|
|
}
|
2016-08-18 04:25:25 +08:00
|
|
|
|
|
|
|
; CHECK-LABEL: name: int_comparison
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[LHS:%[0-9]+]]:_(s32) = COPY $w0
|
|
|
|
; CHECK: [[RHS:%[0-9]+]]:_(s32) = COPY $w1
|
|
|
|
; CHECK: [[ADDR:%[0-9]+]]:_(p0) = COPY $x2
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[TST:%[0-9]+]]:_(s1) = G_ICMP intpred(ne), [[LHS]](s32), [[RHS]]
|
2016-09-12 19:20:10 +08:00
|
|
|
; CHECK: G_STORE [[TST]](s1), [[ADDR]](p0)
|
2016-08-18 04:25:25 +08:00
|
|
|
define void @int_comparison(i32 %a, i32 %b, i1* %addr) {
|
|
|
|
%res = icmp ne i32 %a, %b
|
|
|
|
store i1 %res, i1* %addr
|
|
|
|
ret void
|
|
|
|
}
|
2016-08-19 00:05:06 +08:00
|
|
|
|
2016-09-15 18:40:38 +08:00
|
|
|
; CHECK-LABEL: name: ptr_comparison
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[LHS:%[0-9]+]]:_(p0) = COPY $x0
|
|
|
|
; CHECK: [[RHS:%[0-9]+]]:_(p0) = COPY $x1
|
|
|
|
; CHECK: [[ADDR:%[0-9]+]]:_(p0) = COPY $x2
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[TST:%[0-9]+]]:_(s1) = G_ICMP intpred(eq), [[LHS]](p0), [[RHS]]
|
2016-09-15 18:40:38 +08:00
|
|
|
; CHECK: G_STORE [[TST]](s1), [[ADDR]](p0)
|
|
|
|
define void @ptr_comparison(i8* %a, i8* %b, i1* %addr) {
|
|
|
|
%res = icmp eq i8* %a, %b
|
|
|
|
store i1 %res, i1* %addr
|
|
|
|
ret void
|
|
|
|
}
|
|
|
|
|
2016-08-19 00:05:06 +08:00
|
|
|
; CHECK-LABEL: name: test_fadd
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[ARG1:%[0-9]+]]:_(s32) = COPY $s0
|
|
|
|
; CHECK-NEXT: [[ARG2:%[0-9]+]]:_(s32) = COPY $s1
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK-NEXT: [[RES:%[0-9]+]]:_(s32) = G_FADD [[ARG1]], [[ARG2]]
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK-NEXT: $s0 = COPY [[RES]]
|
|
|
|
; CHECK-NEXT: RET_ReallyLR implicit $s0
|
2016-08-19 00:05:06 +08:00
|
|
|
define float @test_fadd(float %arg1, float %arg2) {
|
|
|
|
%res = fadd float %arg1, %arg2
|
|
|
|
ret float %res
|
|
|
|
}
|
|
|
|
|
|
|
|
; CHECK-LABEL: name: test_fsub
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[ARG1:%[0-9]+]]:_(s32) = COPY $s0
|
|
|
|
; CHECK-NEXT: [[ARG2:%[0-9]+]]:_(s32) = COPY $s1
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK-NEXT: [[RES:%[0-9]+]]:_(s32) = G_FSUB [[ARG1]], [[ARG2]]
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK-NEXT: $s0 = COPY [[RES]]
|
|
|
|
; CHECK-NEXT: RET_ReallyLR implicit $s0
|
2016-08-19 00:05:06 +08:00
|
|
|
define float @test_fsub(float %arg1, float %arg2) {
|
|
|
|
%res = fsub float %arg1, %arg2
|
|
|
|
ret float %res
|
|
|
|
}
|
|
|
|
|
|
|
|
; CHECK-LABEL: name: test_fmul
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[ARG1:%[0-9]+]]:_(s32) = COPY $s0
|
|
|
|
; CHECK-NEXT: [[ARG2:%[0-9]+]]:_(s32) = COPY $s1
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK-NEXT: [[RES:%[0-9]+]]:_(s32) = G_FMUL [[ARG1]], [[ARG2]]
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK-NEXT: $s0 = COPY [[RES]]
|
|
|
|
; CHECK-NEXT: RET_ReallyLR implicit $s0
|
2016-08-19 00:05:06 +08:00
|
|
|
define float @test_fmul(float %arg1, float %arg2) {
|
|
|
|
%res = fmul float %arg1, %arg2
|
|
|
|
ret float %res
|
|
|
|
}
|
|
|
|
|
|
|
|
; CHECK-LABEL: name: test_fdiv
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[ARG1:%[0-9]+]]:_(s32) = COPY $s0
|
|
|
|
; CHECK-NEXT: [[ARG2:%[0-9]+]]:_(s32) = COPY $s1
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK-NEXT: [[RES:%[0-9]+]]:_(s32) = G_FDIV [[ARG1]], [[ARG2]]
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK-NEXT: $s0 = COPY [[RES]]
|
|
|
|
; CHECK-NEXT: RET_ReallyLR implicit $s0
|
2016-08-19 00:05:06 +08:00
|
|
|
define float @test_fdiv(float %arg1, float %arg2) {
|
|
|
|
%res = fdiv float %arg1, %arg2
|
|
|
|
ret float %res
|
|
|
|
}
|
|
|
|
|
|
|
|
; CHECK-LABEL: name: test_frem
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[ARG1:%[0-9]+]]:_(s32) = COPY $s0
|
|
|
|
; CHECK-NEXT: [[ARG2:%[0-9]+]]:_(s32) = COPY $s1
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK-NEXT: [[RES:%[0-9]+]]:_(s32) = G_FREM [[ARG1]], [[ARG2]]
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK-NEXT: $s0 = COPY [[RES]]
|
|
|
|
; CHECK-NEXT: RET_ReallyLR implicit $s0
|
2016-08-19 00:05:06 +08:00
|
|
|
define float @test_frem(float %arg1, float %arg2) {
|
|
|
|
%res = frem float %arg1, %arg2
|
|
|
|
ret float %res
|
|
|
|
}
|
2016-08-20 01:17:06 +08:00
|
|
|
|
2019-01-27 07:47:09 +08:00
|
|
|
; CHECK-LABEL: name: test_fneg
|
|
|
|
; CHECK: [[ARG1:%[0-9]+]]:_(s32) = COPY $s0
|
|
|
|
; CHECK-NEXT: [[RES:%[0-9]+]]:_(s32) = G_FNEG [[ARG1]]
|
|
|
|
; CHECK-NEXT: $s0 = COPY [[RES]]
|
|
|
|
; CHECK-NEXT: RET_ReallyLR implicit $s0
|
|
|
|
define float @test_fneg(float %arg1) {
|
|
|
|
%res = fneg float %arg1
|
|
|
|
ret float %res
|
|
|
|
}
|
|
|
|
|
2016-08-20 01:17:06 +08:00
|
|
|
; CHECK-LABEL: name: test_sadd_overflow
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[LHS:%[0-9]+]]:_(s32) = COPY $w0
|
|
|
|
; CHECK: [[RHS:%[0-9]+]]:_(s32) = COPY $w1
|
|
|
|
; CHECK: [[ADDR:%[0-9]+]]:_(p0) = COPY $x2
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[VAL:%[0-9]+]]:_(s32), [[OVERFLOW:%[0-9]+]]:_(s1) = G_SADDO [[LHS]], [[RHS]]
|
2018-05-16 18:32:02 +08:00
|
|
|
; CHECK: G_STORE [[VAL]](s32), [[ADDR]](p0) :: (store 4 into %ir.addr)
|
|
|
|
; CHECK: [[CST:%[0-9]+]]:_(s64) = G_CONSTANT i64 4
|
|
|
|
; CHECK: [[GEP:%[0-9]+]]:_(p0) = G_GEP [[ADDR]], [[CST]](s64)
|
|
|
|
; CHECK: G_STORE [[OVERFLOW]](s1), [[GEP]](p0) :: (store 1 into %ir.addr + 4, align 4)
|
2016-08-20 01:17:06 +08:00
|
|
|
declare { i32, i1 } @llvm.sadd.with.overflow.i32(i32, i32)
|
|
|
|
define void @test_sadd_overflow(i32 %lhs, i32 %rhs, { i32, i1 }* %addr) {
|
|
|
|
%res = call { i32, i1 } @llvm.sadd.with.overflow.i32(i32 %lhs, i32 %rhs)
|
|
|
|
store { i32, i1 } %res, { i32, i1 }* %addr
|
|
|
|
ret void
|
|
|
|
}
|
|
|
|
|
|
|
|
; CHECK-LABEL: name: test_uadd_overflow
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[LHS:%[0-9]+]]:_(s32) = COPY $w0
|
|
|
|
; CHECK: [[RHS:%[0-9]+]]:_(s32) = COPY $w1
|
|
|
|
; CHECK: [[ADDR:%[0-9]+]]:_(p0) = COPY $x2
|
2018-08-29 02:54:10 +08:00
|
|
|
; CHECK: [[VAL:%[0-9]+]]:_(s32), [[OVERFLOW:%[0-9]+]]:_(s1) = G_UADDO [[LHS]], [[RHS]]
|
2018-05-16 18:32:02 +08:00
|
|
|
; CHECK: G_STORE [[VAL]](s32), [[ADDR]](p0) :: (store 4 into %ir.addr)
|
|
|
|
; CHECK: [[CST:%[0-9]+]]:_(s64) = G_CONSTANT i64 4
|
|
|
|
; CHECK: [[GEP:%[0-9]+]]:_(p0) = G_GEP [[ADDR]], [[CST]](s64)
|
|
|
|
; CHECK: G_STORE [[OVERFLOW]](s1), [[GEP]](p0) :: (store 1 into %ir.addr + 4, align 4)
|
2016-08-20 01:17:06 +08:00
|
|
|
declare { i32, i1 } @llvm.uadd.with.overflow.i32(i32, i32)
|
|
|
|
define void @test_uadd_overflow(i32 %lhs, i32 %rhs, { i32, i1 }* %addr) {
|
|
|
|
%res = call { i32, i1 } @llvm.uadd.with.overflow.i32(i32 %lhs, i32 %rhs)
|
|
|
|
store { i32, i1 } %res, { i32, i1 }* %addr
|
|
|
|
ret void
|
|
|
|
}
|
|
|
|
|
|
|
|
; CHECK-LABEL: name: test_ssub_overflow
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[LHS:%[0-9]+]]:_(s32) = COPY $w0
|
|
|
|
; CHECK: [[RHS:%[0-9]+]]:_(s32) = COPY $w1
|
|
|
|
; CHECK: [[ADDR:%[0-9]+]]:_(p0) = COPY $x2
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[VAL:%[0-9]+]]:_(s32), [[OVERFLOW:%[0-9]+]]:_(s1) = G_SSUBO [[LHS]], [[RHS]]
|
2018-05-16 18:32:02 +08:00
|
|
|
; CHECK: G_STORE [[VAL]](s32), [[ADDR]](p0) :: (store 4 into %ir.subr)
|
|
|
|
; CHECK: [[CST:%[0-9]+]]:_(s64) = G_CONSTANT i64 4
|
|
|
|
; CHECK: [[GEP:%[0-9]+]]:_(p0) = G_GEP [[ADDR]], [[CST]](s64)
|
|
|
|
; CHECK: G_STORE [[OVERFLOW]](s1), [[GEP]](p0) :: (store 1 into %ir.subr + 4, align 4)
|
2016-08-20 01:17:06 +08:00
|
|
|
declare { i32, i1 } @llvm.ssub.with.overflow.i32(i32, i32)
|
|
|
|
define void @test_ssub_overflow(i32 %lhs, i32 %rhs, { i32, i1 }* %subr) {
|
|
|
|
%res = call { i32, i1 } @llvm.ssub.with.overflow.i32(i32 %lhs, i32 %rhs)
|
|
|
|
store { i32, i1 } %res, { i32, i1 }* %subr
|
|
|
|
ret void
|
|
|
|
}
|
|
|
|
|
|
|
|
; CHECK-LABEL: name: test_usub_overflow
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[LHS:%[0-9]+]]:_(s32) = COPY $w0
|
|
|
|
; CHECK: [[RHS:%[0-9]+]]:_(s32) = COPY $w1
|
|
|
|
; CHECK: [[ADDR:%[0-9]+]]:_(p0) = COPY $x2
|
2018-08-29 02:54:10 +08:00
|
|
|
; CHECK: [[VAL:%[0-9]+]]:_(s32), [[OVERFLOW:%[0-9]+]]:_(s1) = G_USUBO [[LHS]], [[RHS]]
|
2018-05-16 18:32:02 +08:00
|
|
|
; CHECK: G_STORE [[VAL]](s32), [[ADDR]](p0) :: (store 4 into %ir.subr)
|
|
|
|
; CHECK: [[CST:%[0-9]+]]:_(s64) = G_CONSTANT i64 4
|
|
|
|
; CHECK: [[GEP:%[0-9]+]]:_(p0) = G_GEP [[ADDR]], [[CST]](s64)
|
|
|
|
; CHECK: G_STORE [[OVERFLOW]](s1), [[GEP]](p0) :: (store 1 into %ir.subr + 4, align 4)
|
2016-08-20 01:17:06 +08:00
|
|
|
declare { i32, i1 } @llvm.usub.with.overflow.i32(i32, i32)
|
|
|
|
define void @test_usub_overflow(i32 %lhs, i32 %rhs, { i32, i1 }* %subr) {
|
|
|
|
%res = call { i32, i1 } @llvm.usub.with.overflow.i32(i32 %lhs, i32 %rhs)
|
|
|
|
store { i32, i1 } %res, { i32, i1 }* %subr
|
|
|
|
ret void
|
|
|
|
}
|
|
|
|
|
|
|
|
; CHECK-LABEL: name: test_smul_overflow
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[LHS:%[0-9]+]]:_(s32) = COPY $w0
|
|
|
|
; CHECK: [[RHS:%[0-9]+]]:_(s32) = COPY $w1
|
|
|
|
; CHECK: [[ADDR:%[0-9]+]]:_(p0) = COPY $x2
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[VAL:%[0-9]+]]:_(s32), [[OVERFLOW:%[0-9]+]]:_(s1) = G_SMULO [[LHS]], [[RHS]]
|
2018-05-16 18:32:02 +08:00
|
|
|
; CHECK: G_STORE [[VAL]](s32), [[ADDR]](p0) :: (store 4 into %ir.addr)
|
|
|
|
; CHECK: [[CST:%[0-9]+]]:_(s64) = G_CONSTANT i64 4
|
|
|
|
; CHECK: [[GEP:%[0-9]+]]:_(p0) = G_GEP [[ADDR]], [[CST]](s64)
|
|
|
|
; CHECK: G_STORE [[OVERFLOW]](s1), [[GEP]](p0) :: (store 1 into %ir.addr + 4, align 4)
|
2016-08-20 01:17:06 +08:00
|
|
|
declare { i32, i1 } @llvm.smul.with.overflow.i32(i32, i32)
|
|
|
|
define void @test_smul_overflow(i32 %lhs, i32 %rhs, { i32, i1 }* %addr) {
|
|
|
|
%res = call { i32, i1 } @llvm.smul.with.overflow.i32(i32 %lhs, i32 %rhs)
|
|
|
|
store { i32, i1 } %res, { i32, i1 }* %addr
|
|
|
|
ret void
|
|
|
|
}
|
|
|
|
|
|
|
|
; CHECK-LABEL: name: test_umul_overflow
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[LHS:%[0-9]+]]:_(s32) = COPY $w0
|
|
|
|
; CHECK: [[RHS:%[0-9]+]]:_(s32) = COPY $w1
|
|
|
|
; CHECK: [[ADDR:%[0-9]+]]:_(p0) = COPY $x2
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[VAL:%[0-9]+]]:_(s32), [[OVERFLOW:%[0-9]+]]:_(s1) = G_UMULO [[LHS]], [[RHS]]
|
2018-05-16 18:32:02 +08:00
|
|
|
; CHECK: G_STORE [[VAL]](s32), [[ADDR]](p0) :: (store 4 into %ir.addr)
|
|
|
|
; CHECK: [[CST:%[0-9]+]]:_(s64) = G_CONSTANT i64 4
|
|
|
|
; CHECK: [[GEP:%[0-9]+]]:_(p0) = G_GEP [[ADDR]], [[CST]](s64)
|
|
|
|
; CHECK: G_STORE [[OVERFLOW]](s1), [[GEP]](p0) :: (store 1 into %ir.addr + 4, align 4)
|
2016-08-20 01:17:06 +08:00
|
|
|
declare { i32, i1 } @llvm.umul.with.overflow.i32(i32, i32)
|
|
|
|
define void @test_umul_overflow(i32 %lhs, i32 %rhs, { i32, i1 }* %addr) {
|
|
|
|
%res = call { i32, i1 } @llvm.umul.with.overflow.i32(i32 %lhs, i32 %rhs)
|
|
|
|
store { i32, i1 } %res, { i32, i1 }* %addr
|
|
|
|
ret void
|
|
|
|
}
|
2016-08-20 01:47:05 +08:00
|
|
|
|
|
|
|
; CHECK-LABEL: name: test_extractvalue
|
2018-05-16 18:32:02 +08:00
|
|
|
; CHECK: %0:_(p0) = COPY $x0
|
|
|
|
; CHECK: [[LD1:%[0-9]+]]:_(s8) = G_LOAD %0(p0) :: (load 1 from %ir.addr, align 4)
|
|
|
|
; CHECK: [[CST1:%[0-9]+]]:_(s64) = G_CONSTANT i64 4
|
|
|
|
; CHECK: [[GEP1:%[0-9]+]]:_(p0) = G_GEP %0, [[CST1]](s64)
|
|
|
|
; CHECK: [[LD2:%[0-9]+]]:_(s8) = G_LOAD [[GEP1]](p0) :: (load 1 from %ir.addr + 4, align 4)
|
|
|
|
; CHECK: [[CST2:%[0-9]+]]:_(s64) = G_CONSTANT i64 8
|
|
|
|
; CHECK: [[GEP2:%[0-9]+]]:_(p0) = G_GEP %0, [[CST2]](s64)
|
|
|
|
; CHECK: [[LD3:%[0-9]+]]:_(s32) = G_LOAD [[GEP2]](p0) :: (load 4 from %ir.addr + 8)
|
|
|
|
; CHECK: [[CST3:%[0-9]+]]:_(s64) = G_CONSTANT i64 12
|
|
|
|
; CHECK: [[GEP3:%[0-9]+]]:_(p0) = G_GEP %0, [[CST3]](s64)
|
|
|
|
; CHECK: [[LD4:%[0-9]+]]:_(s32) = G_LOAD [[GEP3]](p0) :: (load 4 from %ir.addr + 12)
|
|
|
|
; CHECK: $w0 = COPY [[LD3]](s32)
|
2016-08-20 01:47:05 +08:00
|
|
|
%struct.nested = type {i8, { i8, i32 }, i32}
|
|
|
|
define i32 @test_extractvalue(%struct.nested* %addr) {
|
|
|
|
%struct = load %struct.nested, %struct.nested* %addr
|
|
|
|
%res = extractvalue %struct.nested %struct, 1, 1
|
|
|
|
ret i32 %res
|
|
|
|
}
|
2016-08-20 02:09:41 +08:00
|
|
|
|
|
|
|
; CHECK-LABEL: name: test_extractvalue_agg
|
2018-05-16 18:32:02 +08:00
|
|
|
; CHECK: %0:_(p0) = COPY $x0
|
|
|
|
; CHECK: %1:_(p0) = COPY $x1
|
|
|
|
; CHECK: [[LD1:%[0-9]+]]:_(s8) = G_LOAD %0(p0) :: (load 1 from %ir.addr, align 4)
|
|
|
|
; CHECK: [[CST1:%[0-9]+]]:_(s64) = G_CONSTANT i64 4
|
|
|
|
; CHECK: [[GEP1:%[0-9]+]]:_(p0) = G_GEP %0, [[CST1]](s64)
|
|
|
|
; CHECK: [[LD2:%[0-9]+]]:_(s8) = G_LOAD [[GEP1]](p0) :: (load 1 from %ir.addr + 4, align 4)
|
|
|
|
; CHECK: [[CST2:%[0-9]+]]:_(s64) = G_CONSTANT i64 8
|
|
|
|
; CHECK: [[GEP2:%[0-9]+]]:_(p0) = G_GEP %0, [[CST2]](s64)
|
|
|
|
; CHECK: [[LD3:%[0-9]+]]:_(s32) = G_LOAD [[GEP2]](p0) :: (load 4 from %ir.addr + 8)
|
|
|
|
; CHECK: [[CST3:%[0-9]+]]:_(s64) = G_CONSTANT i64 12
|
|
|
|
; CHECK: [[GEP3:%[0-9]+]]:_(p0) = G_GEP %0, [[CST3]](s64)
|
|
|
|
; CHECK: [[LD4:%[0-9]+]]:_(s32) = G_LOAD [[GEP3]](p0) :: (load 4 from %ir.addr + 12)
|
|
|
|
; CHECK: G_STORE [[LD2]](s8), %1(p0) :: (store 1 into %ir.addr2, align 4)
|
|
|
|
; CHECK: [[CST4:%[0-9]+]]:_(s64) = G_CONSTANT i64 4
|
|
|
|
; CHECK: [[GEP4:%[0-9]+]]:_(p0) = G_GEP %1, [[CST4]](s64)
|
|
|
|
; CHECK: G_STORE [[LD3]](s32), [[GEP4]](p0) :: (store 4 into %ir.addr2 + 4)
|
2016-08-20 02:09:41 +08:00
|
|
|
define void @test_extractvalue_agg(%struct.nested* %addr, {i8, i32}* %addr2) {
|
|
|
|
%struct = load %struct.nested, %struct.nested* %addr
|
|
|
|
%res = extractvalue %struct.nested %struct, 1
|
|
|
|
store {i8, i32} %res, {i8, i32}* %addr2
|
|
|
|
ret void
|
|
|
|
}
|
2016-08-20 04:08:55 +08:00
|
|
|
|
|
|
|
; CHECK-LABEL: name: test_insertvalue
|
2018-05-16 18:32:02 +08:00
|
|
|
; CHECK: %0:_(p0) = COPY $x0
|
|
|
|
; CHECK: %1:_(s32) = COPY $w1
|
|
|
|
; CHECK: [[LD1:%[0-9]+]]:_(s8) = G_LOAD %0(p0) :: (load 1 from %ir.addr, align 4)
|
|
|
|
; CHECK: [[CST1:%[0-9]+]]:_(s64) = G_CONSTANT i64 4
|
|
|
|
; CHECK: [[GEP1:%[0-9]+]]:_(p0) = G_GEP %0, [[CST1]](s64)
|
|
|
|
; CHECK: [[LD2:%[0-9]+]]:_(s8) = G_LOAD [[GEP1]](p0) :: (load 1 from %ir.addr + 4, align 4)
|
|
|
|
; CHECK: [[CST2:%[0-9]+]]:_(s64) = G_CONSTANT i64 8
|
|
|
|
; CHECK: [[GEP2:%[0-9]+]]:_(p0) = G_GEP %0, [[CST2]](s64)
|
|
|
|
; CHECK: [[LD3:%[0-9]+]]:_(s32) = G_LOAD [[GEP2]](p0) :: (load 4 from %ir.addr + 8)
|
|
|
|
; CHECK: [[CST3:%[0-9]+]]:_(s64) = G_CONSTANT i64 12
|
|
|
|
; CHECK: [[GEP3:%[0-9]+]]:_(p0) = G_GEP %0, [[CST3]](s64)
|
|
|
|
; CHECK: [[LD4:%[0-9]+]]:_(s32) = G_LOAD [[GEP3]](p0) :: (load 4 from %ir.addr + 12)
|
|
|
|
; CHECK: G_STORE [[LD1]](s8), %0(p0) :: (store 1 into %ir.addr, align 4)
|
|
|
|
; CHECK: [[CST4:%[0-9]+]]:_(s64) = G_CONSTANT i64 4
|
|
|
|
; CHECK: [[GEP4:%[0-9]+]]:_(p0) = G_GEP %0, [[CST4]](s64)
|
|
|
|
; CHECK: G_STORE [[LD2]](s8), [[GEP4]](p0) :: (store 1 into %ir.addr + 4, align 4)
|
|
|
|
; CHECK: [[CST5:%[0-9]+]]:_(s64) = G_CONSTANT i64 8
|
|
|
|
; CHECK: [[GEP5:%[0-9]+]]:_(p0) = G_GEP %0, [[CST5]](s64)
|
|
|
|
; CHECK: G_STORE %1(s32), [[GEP5]](p0) :: (store 4 into %ir.addr + 8)
|
|
|
|
; CHECK: [[CST6:%[0-9]+]]:_(s64) = G_CONSTANT i64 12
|
|
|
|
; CHECK: [[GEP6:%[0-9]+]]:_(p0) = G_GEP %0, [[CST6]](s64)
|
|
|
|
; CHECK: G_STORE [[LD4]](s32), [[GEP6]](p0) :: (store 4 into %ir.addr + 12)
|
2016-08-20 04:08:55 +08:00
|
|
|
define void @test_insertvalue(%struct.nested* %addr, i32 %val) {
|
|
|
|
%struct = load %struct.nested, %struct.nested* %addr
|
|
|
|
%newstruct = insertvalue %struct.nested %struct, i32 %val, 1, 1
|
|
|
|
store %struct.nested %newstruct, %struct.nested* %addr
|
|
|
|
ret void
|
|
|
|
}
|
|
|
|
|
2017-03-07 03:04:17 +08:00
|
|
|
define [1 x i64] @test_trivial_insert([1 x i64] %s, i64 %val) {
|
|
|
|
; CHECK-LABEL: name: test_trivial_insert
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[STRUCT:%[0-9]+]]:_(s64) = COPY $x0
|
|
|
|
; CHECK: [[VAL:%[0-9]+]]:_(s64) = COPY $x1
|
2018-05-16 18:32:02 +08:00
|
|
|
; CHECK: $x0 = COPY [[VAL]]
|
2017-03-07 03:04:17 +08:00
|
|
|
%res = insertvalue [1 x i64] %s, i64 %val, 0
|
|
|
|
ret [1 x i64] %res
|
|
|
|
}
|
|
|
|
|
|
|
|
define [1 x i8*] @test_trivial_insert_ptr([1 x i8*] %s, i8* %val) {
|
|
|
|
; CHECK-LABEL: name: test_trivial_insert_ptr
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[STRUCT:%[0-9]+]]:_(s64) = COPY $x0
|
|
|
|
; CHECK: [[VAL:%[0-9]+]]:_(p0) = COPY $x1
|
2018-05-16 18:32:02 +08:00
|
|
|
; CHECK: $x0 = COPY [[VAL]]
|
2017-03-07 03:04:17 +08:00
|
|
|
%res = insertvalue [1 x i8*] %s, i8* %val, 0
|
|
|
|
ret [1 x i8*] %res
|
|
|
|
}
|
|
|
|
|
2016-08-20 04:08:55 +08:00
|
|
|
; CHECK-LABEL: name: test_insertvalue_agg
|
2018-05-16 18:32:02 +08:00
|
|
|
; CHECK: %0:_(p0) = COPY $x0
|
|
|
|
; CHECK: %1:_(p0) = COPY $x1
|
|
|
|
; CHECK: [[LD1:%[0-9]+]]:_(s8) = G_LOAD %1(p0) :: (load 1 from %ir.addr2, align 4)
|
|
|
|
; CHECK: [[CST1:%[0-9]+]]:_(s64) = G_CONSTANT i64 4
|
|
|
|
; CHECK: [[GEP1:%[0-9]+]]:_(p0) = G_GEP %1, [[CST1]](s64)
|
|
|
|
; CHECK: [[LD2:%[0-9]+]]:_(s32) = G_LOAD [[GEP1]](p0) :: (load 4 from %ir.addr2 + 4)
|
|
|
|
; CHECK: [[LD3:%[0-9]+]]:_(s8) = G_LOAD %0(p0) :: (load 1 from %ir.addr, align 4)
|
|
|
|
; CHECK: [[CST2:%[0-9]+]]:_(s64) = G_CONSTANT i64 4
|
|
|
|
; CHECK: [[GEP2:%[0-9]+]]:_(p0) = G_GEP %0, [[CST2]](s64)
|
|
|
|
; CHECK: [[LD4:%[0-9]+]]:_(s8) = G_LOAD [[GEP2]](p0) :: (load 1 from %ir.addr + 4, align 4)
|
|
|
|
; CHECK: [[CST3:%[0-9]+]]:_(s64) = G_CONSTANT i64 8
|
|
|
|
; CHECK: [[GEP3:%[0-9]+]]:_(p0) = G_GEP %0, [[CST3]](s64)
|
|
|
|
; CHECK: [[LD5:%[0-9]+]]:_(s32) = G_LOAD [[GEP3]](p0) :: (load 4 from %ir.addr + 8)
|
|
|
|
; CHECK: [[CST4:%[0-9]+]]:_(s64) = G_CONSTANT i64 12
|
|
|
|
; CHECK: [[GEP4:%[0-9]+]]:_(p0) = G_GEP %0, [[CST4]](s64)
|
|
|
|
; CHECK: [[LD6:%[0-9]+]]:_(s32) = G_LOAD [[GEP4]](p0) :: (load 4 from %ir.addr + 12)
|
|
|
|
; CHECK: G_STORE [[LD3]](s8), %0(p0) :: (store 1 into %ir.addr, align 4)
|
|
|
|
; CHECK: [[CST5:%[0-9]+]]:_(s64) = G_CONSTANT i64 4
|
|
|
|
; CHECK: [[GEP5:%[0-9]+]]:_(p0) = G_GEP %0, [[CST5]](s64)
|
|
|
|
; CHECK: G_STORE [[LD1]](s8), [[GEP5]](p0) :: (store 1 into %ir.addr + 4, align 4)
|
|
|
|
; CHECK: [[CST6:%[0-9]+]]:_(s64) = G_CONSTANT i64 8
|
|
|
|
; CHECK: [[GEP6:%[0-9]+]]:_(p0) = G_GEP %0, [[CST6]](s64)
|
|
|
|
; CHECK: G_STORE [[LD2]](s32), [[GEP6]](p0) :: (store 4 into %ir.addr + 8)
|
|
|
|
; CHECK: [[CST7:%[0-9]+]]:_(s64) = G_CONSTANT i64 12
|
|
|
|
; CHECK: [[GEP7:%[0-9]+]]:_(p0) = G_GEP %0, [[CST7]](s64)
|
|
|
|
; CHECK: G_STORE [[LD6]](s32), [[GEP7]](p0) :: (store 4 into %ir.addr + 12)
|
2016-08-20 04:08:55 +08:00
|
|
|
define void @test_insertvalue_agg(%struct.nested* %addr, {i8, i32}* %addr2) {
|
|
|
|
%smallstruct = load {i8, i32}, {i8, i32}* %addr2
|
|
|
|
%struct = load %struct.nested, %struct.nested* %addr
|
|
|
|
%res = insertvalue %struct.nested %struct, {i8, i32} %smallstruct, 1
|
|
|
|
store %struct.nested %res, %struct.nested* %addr
|
|
|
|
ret void
|
|
|
|
}
|
2016-08-20 04:09:07 +08:00
|
|
|
|
|
|
|
; CHECK-LABEL: name: test_select
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[TST_C:%[0-9]+]]:_(s32) = COPY $w0
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[TST:%[0-9]+]]:_(s1) = G_TRUNC [[TST_C]]
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[LHS:%[0-9]+]]:_(s32) = COPY $w1
|
|
|
|
; CHECK: [[RHS:%[0-9]+]]:_(s32) = COPY $w2
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[RES:%[0-9]+]]:_(s32) = G_SELECT [[TST]](s1), [[LHS]], [[RHS]]
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: $w0 = COPY [[RES]]
|
2016-08-20 04:09:07 +08:00
|
|
|
define i32 @test_select(i1 %tst, i32 %lhs, i32 %rhs) {
|
|
|
|
%res = select i1 %tst, i32 %lhs, i32 %rhs
|
|
|
|
ret i32 %res
|
|
|
|
}
|
2016-08-20 04:09:11 +08:00
|
|
|
|
2016-12-07 02:38:34 +08:00
|
|
|
; CHECK-LABEL: name: test_select_ptr
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[TST_C:%[0-9]+]]:_(s32) = COPY $w0
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[TST:%[0-9]+]]:_(s1) = G_TRUNC [[TST_C]]
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[LHS:%[0-9]+]]:_(p0) = COPY $x1
|
|
|
|
; CHECK: [[RHS:%[0-9]+]]:_(p0) = COPY $x2
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[RES:%[0-9]+]]:_(p0) = G_SELECT [[TST]](s1), [[LHS]], [[RHS]]
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: $x0 = COPY [[RES]]
|
2016-12-07 02:38:34 +08:00
|
|
|
define i8* @test_select_ptr(i1 %tst, i8* %lhs, i8* %rhs) {
|
|
|
|
%res = select i1 %tst, i8* %lhs, i8* %rhs
|
|
|
|
ret i8* %res
|
|
|
|
}
|
|
|
|
|
2017-03-08 04:53:03 +08:00
|
|
|
; CHECK-LABEL: name: test_select_vec
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[TST_C:%[0-9]+]]:_(s32) = COPY $w0
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[TST:%[0-9]+]]:_(s1) = G_TRUNC [[TST_C]]
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[LHS:%[0-9]+]]:_(<4 x s32>) = COPY $q0
|
|
|
|
; CHECK: [[RHS:%[0-9]+]]:_(<4 x s32>) = COPY $q1
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[RES:%[0-9]+]]:_(<4 x s32>) = G_SELECT [[TST]](s1), [[LHS]], [[RHS]]
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: $q0 = COPY [[RES]]
|
2017-03-08 04:53:03 +08:00
|
|
|
define <4 x i32> @test_select_vec(i1 %tst, <4 x i32> %lhs, <4 x i32> %rhs) {
|
|
|
|
%res = select i1 %tst, <4 x i32> %lhs, <4 x i32> %rhs
|
|
|
|
ret <4 x i32> %res
|
|
|
|
}
|
|
|
|
|
2017-03-08 04:53:00 +08:00
|
|
|
; CHECK-LABEL: name: test_vselect_vec
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[TST32:%[0-9]+]]:_(<4 x s32>) = COPY $q0
|
|
|
|
; CHECK: [[LHS:%[0-9]+]]:_(<4 x s32>) = COPY $q1
|
|
|
|
; CHECK: [[RHS:%[0-9]+]]:_(<4 x s32>) = COPY $q2
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[TST:%[0-9]+]]:_(<4 x s1>) = G_TRUNC [[TST32]](<4 x s32>)
|
|
|
|
; CHECK: [[RES:%[0-9]+]]:_(<4 x s32>) = G_SELECT [[TST]](<4 x s1>), [[LHS]], [[RHS]]
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: $q0 = COPY [[RES]]
|
2017-03-08 04:53:00 +08:00
|
|
|
define <4 x i32> @test_vselect_vec(<4 x i32> %tst32, <4 x i32> %lhs, <4 x i32> %rhs) {
|
|
|
|
%tst = trunc <4 x i32> %tst32 to <4 x i1>
|
|
|
|
%res = select <4 x i1> %tst, <4 x i32> %lhs, <4 x i32> %rhs
|
|
|
|
ret <4 x i32> %res
|
|
|
|
}
|
|
|
|
|
2016-08-20 04:09:11 +08:00
|
|
|
; CHECK-LABEL: name: test_fptosi
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[FPADDR:%[0-9]+]]:_(p0) = COPY $x0
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[FP:%[0-9]+]]:_(s32) = G_LOAD [[FPADDR]](p0)
|
|
|
|
; CHECK: [[RES:%[0-9]+]]:_(s64) = G_FPTOSI [[FP]](s32)
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: $x0 = COPY [[RES]]
|
2016-08-20 04:09:11 +08:00
|
|
|
define i64 @test_fptosi(float* %fp.addr) {
|
|
|
|
%fp = load float, float* %fp.addr
|
|
|
|
%res = fptosi float %fp to i64
|
|
|
|
ret i64 %res
|
|
|
|
}
|
|
|
|
|
|
|
|
; CHECK-LABEL: name: test_fptoui
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[FPADDR:%[0-9]+]]:_(p0) = COPY $x0
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[FP:%[0-9]+]]:_(s32) = G_LOAD [[FPADDR]](p0)
|
|
|
|
; CHECK: [[RES:%[0-9]+]]:_(s64) = G_FPTOUI [[FP]](s32)
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: $x0 = COPY [[RES]]
|
2016-08-20 04:09:11 +08:00
|
|
|
define i64 @test_fptoui(float* %fp.addr) {
|
|
|
|
%fp = load float, float* %fp.addr
|
|
|
|
%res = fptoui float %fp to i64
|
|
|
|
ret i64 %res
|
|
|
|
}
|
|
|
|
|
|
|
|
; CHECK-LABEL: name: test_sitofp
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[ADDR:%[0-9]+]]:_(p0) = COPY $x0
|
|
|
|
; CHECK: [[IN:%[0-9]+]]:_(s32) = COPY $w1
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[FP:%[0-9]+]]:_(s64) = G_SITOFP [[IN]](s32)
|
2016-09-12 19:20:10 +08:00
|
|
|
; CHECK: G_STORE [[FP]](s64), [[ADDR]](p0)
|
2016-08-20 04:09:11 +08:00
|
|
|
define void @test_sitofp(double* %addr, i32 %in) {
|
|
|
|
%fp = sitofp i32 %in to double
|
|
|
|
store double %fp, double* %addr
|
|
|
|
ret void
|
|
|
|
}
|
|
|
|
|
|
|
|
; CHECK-LABEL: name: test_uitofp
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[ADDR:%[0-9]+]]:_(p0) = COPY $x0
|
|
|
|
; CHECK: [[IN:%[0-9]+]]:_(s32) = COPY $w1
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[FP:%[0-9]+]]:_(s64) = G_UITOFP [[IN]](s32)
|
2016-09-12 19:20:10 +08:00
|
|
|
; CHECK: G_STORE [[FP]](s64), [[ADDR]](p0)
|
2016-08-20 04:09:11 +08:00
|
|
|
define void @test_uitofp(double* %addr, i32 %in) {
|
|
|
|
%fp = uitofp i32 %in to double
|
|
|
|
store double %fp, double* %addr
|
|
|
|
ret void
|
|
|
|
}
|
2016-08-20 04:09:15 +08:00
|
|
|
|
2016-08-20 04:48:23 +08:00
|
|
|
; CHECK-LABEL: name: test_fpext
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[IN:%[0-9]+]]:_(s32) = COPY $s0
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[RES:%[0-9]+]]:_(s64) = G_FPEXT [[IN]](s32)
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: $d0 = COPY [[RES]]
|
2016-08-20 04:48:23 +08:00
|
|
|
define double @test_fpext(float %in) {
|
|
|
|
%res = fpext float %in to double
|
|
|
|
ret double %res
|
|
|
|
}
|
|
|
|
|
|
|
|
; CHECK-LABEL: name: test_fptrunc
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[IN:%[0-9]+]]:_(s64) = COPY $d0
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[RES:%[0-9]+]]:_(s32) = G_FPTRUNC [[IN]](s64)
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: $s0 = COPY [[RES]]
|
2016-08-20 04:48:23 +08:00
|
|
|
define float @test_fptrunc(double %in) {
|
|
|
|
%res = fptrunc double %in to float
|
|
|
|
ret float %res
|
|
|
|
}
|
|
|
|
|
2016-08-20 04:09:15 +08:00
|
|
|
; CHECK-LABEL: name: test_constant_float
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[ADDR:%[0-9]+]]:_(p0) = COPY $x0
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[TMP:%[0-9]+]]:_(s32) = G_FCONSTANT float 1.500000e+00
|
2016-09-12 19:20:10 +08:00
|
|
|
; CHECK: G_STORE [[TMP]](s32), [[ADDR]](p0)
|
2016-08-20 04:09:15 +08:00
|
|
|
define void @test_constant_float(float* %addr) {
|
|
|
|
store float 1.5, float* %addr
|
|
|
|
ret void
|
|
|
|
}
|
2016-08-20 04:48:16 +08:00
|
|
|
|
|
|
|
; CHECK-LABEL: name: float_comparison
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[LHSADDR:%[0-9]+]]:_(p0) = COPY $x0
|
|
|
|
; CHECK: [[RHSADDR:%[0-9]+]]:_(p0) = COPY $x1
|
|
|
|
; CHECK: [[BOOLADDR:%[0-9]+]]:_(p0) = COPY $x2
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[LHS:%[0-9]+]]:_(s32) = G_LOAD [[LHSADDR]](p0)
|
|
|
|
; CHECK: [[RHS:%[0-9]+]]:_(s32) = G_LOAD [[RHSADDR]](p0)
|
2018-12-19 01:54:52 +08:00
|
|
|
; CHECK: [[TST:%[0-9]+]]:_(s1) = nnan ninf nsz arcp contract afn reassoc G_FCMP floatpred(oge), [[LHS]](s32), [[RHS]]
|
2016-09-12 19:20:10 +08:00
|
|
|
; CHECK: G_STORE [[TST]](s1), [[BOOLADDR]](p0)
|
2016-08-20 04:48:16 +08:00
|
|
|
define void @float_comparison(float* %a.addr, float* %b.addr, i1* %bool.addr) {
|
|
|
|
%a = load float, float* %a.addr
|
|
|
|
%b = load float, float* %b.addr
|
2018-12-19 01:54:52 +08:00
|
|
|
%res = fcmp nnan ninf nsz arcp contract afn reassoc oge float %a, %b
|
2016-08-20 04:48:16 +08:00
|
|
|
store i1 %res, i1* %bool.addr
|
|
|
|
ret void
|
|
|
|
}
|
2016-09-12 20:10:41 +08:00
|
|
|
|
2017-03-09 02:49:54 +08:00
|
|
|
; CHECK-LABEL: name: trivial_float_comparison
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[ENTRY_R1:%[0-9]+]]:_(s1) = G_CONSTANT i1 false
|
|
|
|
; CHECK: [[ENTRY_R2:%[0-9]+]]:_(s1) = G_CONSTANT i1 true
|
|
|
|
; CHECK: [[R1:%[0-9]+]]:_(s1) = COPY [[ENTRY_R1]](s1)
|
|
|
|
; CHECK: [[R2:%[0-9]+]]:_(s1) = COPY [[ENTRY_R2]](s1)
|
2017-03-09 02:49:54 +08:00
|
|
|
; CHECK: G_ADD [[R1]], [[R2]]
|
|
|
|
define i1 @trivial_float_comparison(double %a, double %b) {
|
|
|
|
%r1 = fcmp false double %a, %b
|
|
|
|
%r2 = fcmp true double %a, %b
|
|
|
|
%sum = add i1 %r1, %r2
|
|
|
|
ret i1 %sum
|
|
|
|
}
|
|
|
|
|
2016-09-12 20:10:41 +08:00
|
|
|
@var = global i32 0
|
|
|
|
|
|
|
|
define i32* @test_global() {
|
|
|
|
; CHECK-LABEL: name: test_global
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[TMP:%[0-9]+]]:_(p0) = G_GLOBAL_VALUE @var{{$}}
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: $x0 = COPY [[TMP]](p0)
|
2016-09-12 20:10:41 +08:00
|
|
|
|
|
|
|
ret i32* @var
|
|
|
|
}
|
|
|
|
|
|
|
|
@var1 = addrspace(42) global i32 0
|
|
|
|
define i32 addrspace(42)* @test_global_addrspace() {
|
|
|
|
; CHECK-LABEL: name: test_global
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[TMP:%[0-9]+]]:_(p42) = G_GLOBAL_VALUE @var1{{$}}
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: $x0 = COPY [[TMP]](p42)
|
2016-09-12 20:10:41 +08:00
|
|
|
|
|
|
|
ret i32 addrspace(42)* @var1
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
define void()* @test_global_func() {
|
|
|
|
; CHECK-LABEL: name: test_global_func
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[TMP:%[0-9]+]]:_(p0) = G_GLOBAL_VALUE @allocai64{{$}}
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: $x0 = COPY [[TMP]](p0)
|
2016-09-12 20:10:41 +08:00
|
|
|
|
|
|
|
ret void()* @allocai64
|
|
|
|
}
|
2016-10-19 04:03:45 +08:00
|
|
|
|
Remove alignment argument from memcpy/memmove/memset in favour of alignment attributes (Step 1)
Summary:
This is a resurrection of work first proposed and discussed in Aug 2015:
http://lists.llvm.org/pipermail/llvm-dev/2015-August/089384.html
and initially landed (but then backed out) in Nov 2015:
http://lists.llvm.org/pipermail/llvm-commits/Week-of-Mon-20151109/312083.html
The @llvm.memcpy/memmove/memset intrinsics currently have an explicit argument
which is required to be a constant integer. It represents the alignment of the
dest (and source), and so must be the minimum of the actual alignment of the
two.
This change is the first in a series that allows source and dest to each
have their own alignments by using the alignment attribute on their arguments.
In this change we:
1) Remove the alignment argument.
2) Add alignment attributes to the source & dest arguments. We, temporarily,
require that the alignments for source & dest be equal.
For example, code which used to read:
call void @llvm.memcpy.p0i8.p0i8.i32(i8* %dest, i8* %src, i32 100, i32 4, i1 false)
will now read
call void @llvm.memcpy.p0i8.p0i8.i32(i8* align 4 %dest, i8* align 4 %src, i32 100, i1 false)
Downstream users may have to update their lit tests that check for
@llvm.memcpy/memmove/memset call/declaration patterns. The following extended sed script
may help with updating the majority of your tests, but it does not catch all possible
patterns so some manual checking and updating will be required.
s~declare void @llvm\.mem(set|cpy|move)\.p([^(]*)\((.*), i32, i1\)~declare void @llvm.mem\1.p\2(\3, i1)~g
s~call void @llvm\.memset\.p([^(]*)i8\(i8([^*]*)\* (.*), i8 (.*), i8 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.memset.p\1i8(i8\2* \3, i8 \4, i8 \5, i1 \6)~g
s~call void @llvm\.memset\.p([^(]*)i16\(i8([^*]*)\* (.*), i8 (.*), i16 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.memset.p\1i16(i8\2* \3, i8 \4, i16 \5, i1 \6)~g
s~call void @llvm\.memset\.p([^(]*)i32\(i8([^*]*)\* (.*), i8 (.*), i32 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.memset.p\1i32(i8\2* \3, i8 \4, i32 \5, i1 \6)~g
s~call void @llvm\.memset\.p([^(]*)i64\(i8([^*]*)\* (.*), i8 (.*), i64 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.memset.p\1i64(i8\2* \3, i8 \4, i64 \5, i1 \6)~g
s~call void @llvm\.memset\.p([^(]*)i128\(i8([^*]*)\* (.*), i8 (.*), i128 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.memset.p\1i128(i8\2* \3, i8 \4, i128 \5, i1 \6)~g
s~call void @llvm\.memset\.p([^(]*)i8\(i8([^*]*)\* (.*), i8 (.*), i8 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.memset.p\1i8(i8\2* align \6 \3, i8 \4, i8 \5, i1 \7)~g
s~call void @llvm\.memset\.p([^(]*)i16\(i8([^*]*)\* (.*), i8 (.*), i16 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.memset.p\1i16(i8\2* align \6 \3, i8 \4, i16 \5, i1 \7)~g
s~call void @llvm\.memset\.p([^(]*)i32\(i8([^*]*)\* (.*), i8 (.*), i32 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.memset.p\1i32(i8\2* align \6 \3, i8 \4, i32 \5, i1 \7)~g
s~call void @llvm\.memset\.p([^(]*)i64\(i8([^*]*)\* (.*), i8 (.*), i64 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.memset.p\1i64(i8\2* align \6 \3, i8 \4, i64 \5, i1 \7)~g
s~call void @llvm\.memset\.p([^(]*)i128\(i8([^*]*)\* (.*), i8 (.*), i128 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.memset.p\1i128(i8\2* align \6 \3, i8 \4, i128 \5, i1 \7)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i8\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i8 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.mem\1.p\2i8(i8\3* \4, i8\5* \6, i8 \7, i1 \8)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i16\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i16 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.mem\1.p\2i16(i8\3* \4, i8\5* \6, i16 \7, i1 \8)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i32\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i32 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.mem\1.p\2i32(i8\3* \4, i8\5* \6, i32 \7, i1 \8)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i64\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i64 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.mem\1.p\2i64(i8\3* \4, i8\5* \6, i64 \7, i1 \8)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i128\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i128 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.mem\1.p\2i128(i8\3* \4, i8\5* \6, i128 \7, i1 \8)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i8\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i8 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.mem\1.p\2i8(i8\3* align \8 \4, i8\5* align \8 \6, i8 \7, i1 \9)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i16\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i16 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.mem\1.p\2i16(i8\3* align \8 \4, i8\5* align \8 \6, i16 \7, i1 \9)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i32\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i32 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.mem\1.p\2i32(i8\3* align \8 \4, i8\5* align \8 \6, i32 \7, i1 \9)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i64\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i64 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.mem\1.p\2i64(i8\3* align \8 \4, i8\5* align \8 \6, i64 \7, i1 \9)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i128\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i128 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.mem\1.p\2i128(i8\3* align \8 \4, i8\5* align \8 \6, i128 \7, i1 \9)~g
The remaining changes in the series will:
Step 2) Expand the IRBuilder API to allow creation of memcpy/memmove with differing
source and dest alignments.
Step 3) Update Clang to use the new IRBuilder API.
Step 4) Update Polly to use the new IRBuilder API.
Step 5) Update LLVM passes that create memcpy/memmove calls to use the new IRBuilder API,
and those that use use MemIntrinsicInst::[get|set]Alignment() to use
getDestAlignment() and getSourceAlignment() instead.
Step 6) Remove the single-alignment IRBuilder API for memcpy/memmove, and the
MemIntrinsicInst::[get|set]Alignment() methods.
Reviewers: pete, hfinkel, lhames, reames, bollu
Reviewed By: reames
Subscribers: niosHD, reames, jholewinski, qcolombet, jfb, sanjoy, arsenm, dschuff, dylanmckay, mehdi_amini, sdardis, nemanjai, david2050, nhaehnle, javed.absar, sbc100, jgravelle-google, eraman, aheejin, kbarton, JDevlieghere, asb, rbar, johnrusso, simoncook, jordy.potman.lists, apazos, sabuasal, llvm-commits
Differential Revision: https://reviews.llvm.org/D41675
llvm-svn: 322965
2018-01-20 01:13:12 +08:00
|
|
|
declare void @llvm.memcpy.p0i8.p0i8.i64(i8*, i8*, i64, i1)
|
2016-10-19 04:03:45 +08:00
|
|
|
define void @test_memcpy(i8* %dst, i8* %src, i64 %size) {
|
|
|
|
; CHECK-LABEL: name: test_memcpy
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[DST:%[0-9]+]]:_(p0) = COPY $x0
|
|
|
|
; CHECK: [[SRC:%[0-9]+]]:_(p0) = COPY $x1
|
|
|
|
; CHECK: [[SIZE:%[0-9]+]]:_(s64) = COPY $x2
|
|
|
|
; CHECK: $x0 = COPY [[DST]]
|
|
|
|
; CHECK: $x1 = COPY [[SRC]]
|
|
|
|
; CHECK: $x2 = COPY [[SIZE]]
|
|
|
|
; CHECK: BL &memcpy, csr_aarch64_aapcs, implicit-def $lr, implicit $sp, implicit $x0, implicit $x1, implicit $x2
|
Remove alignment argument from memcpy/memmove/memset in favour of alignment attributes (Step 1)
Summary:
This is a resurrection of work first proposed and discussed in Aug 2015:
http://lists.llvm.org/pipermail/llvm-dev/2015-August/089384.html
and initially landed (but then backed out) in Nov 2015:
http://lists.llvm.org/pipermail/llvm-commits/Week-of-Mon-20151109/312083.html
The @llvm.memcpy/memmove/memset intrinsics currently have an explicit argument
which is required to be a constant integer. It represents the alignment of the
dest (and source), and so must be the minimum of the actual alignment of the
two.
This change is the first in a series that allows source and dest to each
have their own alignments by using the alignment attribute on their arguments.
In this change we:
1) Remove the alignment argument.
2) Add alignment attributes to the source & dest arguments. We, temporarily,
require that the alignments for source & dest be equal.
For example, code which used to read:
call void @llvm.memcpy.p0i8.p0i8.i32(i8* %dest, i8* %src, i32 100, i32 4, i1 false)
will now read
call void @llvm.memcpy.p0i8.p0i8.i32(i8* align 4 %dest, i8* align 4 %src, i32 100, i1 false)
Downstream users may have to update their lit tests that check for
@llvm.memcpy/memmove/memset call/declaration patterns. The following extended sed script
may help with updating the majority of your tests, but it does not catch all possible
patterns so some manual checking and updating will be required.
s~declare void @llvm\.mem(set|cpy|move)\.p([^(]*)\((.*), i32, i1\)~declare void @llvm.mem\1.p\2(\3, i1)~g
s~call void @llvm\.memset\.p([^(]*)i8\(i8([^*]*)\* (.*), i8 (.*), i8 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.memset.p\1i8(i8\2* \3, i8 \4, i8 \5, i1 \6)~g
s~call void @llvm\.memset\.p([^(]*)i16\(i8([^*]*)\* (.*), i8 (.*), i16 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.memset.p\1i16(i8\2* \3, i8 \4, i16 \5, i1 \6)~g
s~call void @llvm\.memset\.p([^(]*)i32\(i8([^*]*)\* (.*), i8 (.*), i32 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.memset.p\1i32(i8\2* \3, i8 \4, i32 \5, i1 \6)~g
s~call void @llvm\.memset\.p([^(]*)i64\(i8([^*]*)\* (.*), i8 (.*), i64 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.memset.p\1i64(i8\2* \3, i8 \4, i64 \5, i1 \6)~g
s~call void @llvm\.memset\.p([^(]*)i128\(i8([^*]*)\* (.*), i8 (.*), i128 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.memset.p\1i128(i8\2* \3, i8 \4, i128 \5, i1 \6)~g
s~call void @llvm\.memset\.p([^(]*)i8\(i8([^*]*)\* (.*), i8 (.*), i8 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.memset.p\1i8(i8\2* align \6 \3, i8 \4, i8 \5, i1 \7)~g
s~call void @llvm\.memset\.p([^(]*)i16\(i8([^*]*)\* (.*), i8 (.*), i16 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.memset.p\1i16(i8\2* align \6 \3, i8 \4, i16 \5, i1 \7)~g
s~call void @llvm\.memset\.p([^(]*)i32\(i8([^*]*)\* (.*), i8 (.*), i32 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.memset.p\1i32(i8\2* align \6 \3, i8 \4, i32 \5, i1 \7)~g
s~call void @llvm\.memset\.p([^(]*)i64\(i8([^*]*)\* (.*), i8 (.*), i64 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.memset.p\1i64(i8\2* align \6 \3, i8 \4, i64 \5, i1 \7)~g
s~call void @llvm\.memset\.p([^(]*)i128\(i8([^*]*)\* (.*), i8 (.*), i128 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.memset.p\1i128(i8\2* align \6 \3, i8 \4, i128 \5, i1 \7)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i8\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i8 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.mem\1.p\2i8(i8\3* \4, i8\5* \6, i8 \7, i1 \8)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i16\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i16 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.mem\1.p\2i16(i8\3* \4, i8\5* \6, i16 \7, i1 \8)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i32\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i32 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.mem\1.p\2i32(i8\3* \4, i8\5* \6, i32 \7, i1 \8)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i64\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i64 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.mem\1.p\2i64(i8\3* \4, i8\5* \6, i64 \7, i1 \8)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i128\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i128 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.mem\1.p\2i128(i8\3* \4, i8\5* \6, i128 \7, i1 \8)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i8\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i8 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.mem\1.p\2i8(i8\3* align \8 \4, i8\5* align \8 \6, i8 \7, i1 \9)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i16\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i16 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.mem\1.p\2i16(i8\3* align \8 \4, i8\5* align \8 \6, i16 \7, i1 \9)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i32\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i32 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.mem\1.p\2i32(i8\3* align \8 \4, i8\5* align \8 \6, i32 \7, i1 \9)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i64\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i64 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.mem\1.p\2i64(i8\3* align \8 \4, i8\5* align \8 \6, i64 \7, i1 \9)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i128\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i128 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.mem\1.p\2i128(i8\3* align \8 \4, i8\5* align \8 \6, i128 \7, i1 \9)~g
The remaining changes in the series will:
Step 2) Expand the IRBuilder API to allow creation of memcpy/memmove with differing
source and dest alignments.
Step 3) Update Clang to use the new IRBuilder API.
Step 4) Update Polly to use the new IRBuilder API.
Step 5) Update LLVM passes that create memcpy/memmove calls to use the new IRBuilder API,
and those that use use MemIntrinsicInst::[get|set]Alignment() to use
getDestAlignment() and getSourceAlignment() instead.
Step 6) Remove the single-alignment IRBuilder API for memcpy/memmove, and the
MemIntrinsicInst::[get|set]Alignment() methods.
Reviewers: pete, hfinkel, lhames, reames, bollu
Reviewed By: reames
Subscribers: niosHD, reames, jholewinski, qcolombet, jfb, sanjoy, arsenm, dschuff, dylanmckay, mehdi_amini, sdardis, nemanjai, david2050, nhaehnle, javed.absar, sbc100, jgravelle-google, eraman, aheejin, kbarton, JDevlieghere, asb, rbar, johnrusso, simoncook, jordy.potman.lists, apazos, sabuasal, llvm-commits
Differential Revision: https://reviews.llvm.org/D41675
llvm-svn: 322965
2018-01-20 01:13:12 +08:00
|
|
|
call void @llvm.memcpy.p0i8.p0i8.i64(i8* %dst, i8* %src, i64 %size, i1 0)
|
2016-10-19 04:03:45 +08:00
|
|
|
ret void
|
|
|
|
}
|
2016-10-19 04:03:51 +08:00
|
|
|
|
Remove alignment argument from memcpy/memmove/memset in favour of alignment attributes (Step 1)
Summary:
This is a resurrection of work first proposed and discussed in Aug 2015:
http://lists.llvm.org/pipermail/llvm-dev/2015-August/089384.html
and initially landed (but then backed out) in Nov 2015:
http://lists.llvm.org/pipermail/llvm-commits/Week-of-Mon-20151109/312083.html
The @llvm.memcpy/memmove/memset intrinsics currently have an explicit argument
which is required to be a constant integer. It represents the alignment of the
dest (and source), and so must be the minimum of the actual alignment of the
two.
This change is the first in a series that allows source and dest to each
have their own alignments by using the alignment attribute on their arguments.
In this change we:
1) Remove the alignment argument.
2) Add alignment attributes to the source & dest arguments. We, temporarily,
require that the alignments for source & dest be equal.
For example, code which used to read:
call void @llvm.memcpy.p0i8.p0i8.i32(i8* %dest, i8* %src, i32 100, i32 4, i1 false)
will now read
call void @llvm.memcpy.p0i8.p0i8.i32(i8* align 4 %dest, i8* align 4 %src, i32 100, i1 false)
Downstream users may have to update their lit tests that check for
@llvm.memcpy/memmove/memset call/declaration patterns. The following extended sed script
may help with updating the majority of your tests, but it does not catch all possible
patterns so some manual checking and updating will be required.
s~declare void @llvm\.mem(set|cpy|move)\.p([^(]*)\((.*), i32, i1\)~declare void @llvm.mem\1.p\2(\3, i1)~g
s~call void @llvm\.memset\.p([^(]*)i8\(i8([^*]*)\* (.*), i8 (.*), i8 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.memset.p\1i8(i8\2* \3, i8 \4, i8 \5, i1 \6)~g
s~call void @llvm\.memset\.p([^(]*)i16\(i8([^*]*)\* (.*), i8 (.*), i16 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.memset.p\1i16(i8\2* \3, i8 \4, i16 \5, i1 \6)~g
s~call void @llvm\.memset\.p([^(]*)i32\(i8([^*]*)\* (.*), i8 (.*), i32 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.memset.p\1i32(i8\2* \3, i8 \4, i32 \5, i1 \6)~g
s~call void @llvm\.memset\.p([^(]*)i64\(i8([^*]*)\* (.*), i8 (.*), i64 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.memset.p\1i64(i8\2* \3, i8 \4, i64 \5, i1 \6)~g
s~call void @llvm\.memset\.p([^(]*)i128\(i8([^*]*)\* (.*), i8 (.*), i128 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.memset.p\1i128(i8\2* \3, i8 \4, i128 \5, i1 \6)~g
s~call void @llvm\.memset\.p([^(]*)i8\(i8([^*]*)\* (.*), i8 (.*), i8 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.memset.p\1i8(i8\2* align \6 \3, i8 \4, i8 \5, i1 \7)~g
s~call void @llvm\.memset\.p([^(]*)i16\(i8([^*]*)\* (.*), i8 (.*), i16 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.memset.p\1i16(i8\2* align \6 \3, i8 \4, i16 \5, i1 \7)~g
s~call void @llvm\.memset\.p([^(]*)i32\(i8([^*]*)\* (.*), i8 (.*), i32 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.memset.p\1i32(i8\2* align \6 \3, i8 \4, i32 \5, i1 \7)~g
s~call void @llvm\.memset\.p([^(]*)i64\(i8([^*]*)\* (.*), i8 (.*), i64 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.memset.p\1i64(i8\2* align \6 \3, i8 \4, i64 \5, i1 \7)~g
s~call void @llvm\.memset\.p([^(]*)i128\(i8([^*]*)\* (.*), i8 (.*), i128 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.memset.p\1i128(i8\2* align \6 \3, i8 \4, i128 \5, i1 \7)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i8\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i8 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.mem\1.p\2i8(i8\3* \4, i8\5* \6, i8 \7, i1 \8)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i16\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i16 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.mem\1.p\2i16(i8\3* \4, i8\5* \6, i16 \7, i1 \8)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i32\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i32 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.mem\1.p\2i32(i8\3* \4, i8\5* \6, i32 \7, i1 \8)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i64\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i64 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.mem\1.p\2i64(i8\3* \4, i8\5* \6, i64 \7, i1 \8)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i128\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i128 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.mem\1.p\2i128(i8\3* \4, i8\5* \6, i128 \7, i1 \8)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i8\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i8 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.mem\1.p\2i8(i8\3* align \8 \4, i8\5* align \8 \6, i8 \7, i1 \9)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i16\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i16 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.mem\1.p\2i16(i8\3* align \8 \4, i8\5* align \8 \6, i16 \7, i1 \9)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i32\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i32 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.mem\1.p\2i32(i8\3* align \8 \4, i8\5* align \8 \6, i32 \7, i1 \9)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i64\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i64 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.mem\1.p\2i64(i8\3* align \8 \4, i8\5* align \8 \6, i64 \7, i1 \9)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i128\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i128 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.mem\1.p\2i128(i8\3* align \8 \4, i8\5* align \8 \6, i128 \7, i1 \9)~g
The remaining changes in the series will:
Step 2) Expand the IRBuilder API to allow creation of memcpy/memmove with differing
source and dest alignments.
Step 3) Update Clang to use the new IRBuilder API.
Step 4) Update Polly to use the new IRBuilder API.
Step 5) Update LLVM passes that create memcpy/memmove calls to use the new IRBuilder API,
and those that use use MemIntrinsicInst::[get|set]Alignment() to use
getDestAlignment() and getSourceAlignment() instead.
Step 6) Remove the single-alignment IRBuilder API for memcpy/memmove, and the
MemIntrinsicInst::[get|set]Alignment() methods.
Reviewers: pete, hfinkel, lhames, reames, bollu
Reviewed By: reames
Subscribers: niosHD, reames, jholewinski, qcolombet, jfb, sanjoy, arsenm, dschuff, dylanmckay, mehdi_amini, sdardis, nemanjai, david2050, nhaehnle, javed.absar, sbc100, jgravelle-google, eraman, aheejin, kbarton, JDevlieghere, asb, rbar, johnrusso, simoncook, jordy.potman.lists, apazos, sabuasal, llvm-commits
Differential Revision: https://reviews.llvm.org/D41675
llvm-svn: 322965
2018-01-20 01:13:12 +08:00
|
|
|
declare void @llvm.memmove.p0i8.p0i8.i64(i8*, i8*, i64, i1)
|
2017-01-31 03:33:07 +08:00
|
|
|
define void @test_memmove(i8* %dst, i8* %src, i64 %size) {
|
|
|
|
; CHECK-LABEL: name: test_memmove
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[DST:%[0-9]+]]:_(p0) = COPY $x0
|
|
|
|
; CHECK: [[SRC:%[0-9]+]]:_(p0) = COPY $x1
|
|
|
|
; CHECK: [[SIZE:%[0-9]+]]:_(s64) = COPY $x2
|
|
|
|
; CHECK: $x0 = COPY [[DST]]
|
|
|
|
; CHECK: $x1 = COPY [[SRC]]
|
|
|
|
; CHECK: $x2 = COPY [[SIZE]]
|
|
|
|
; CHECK: BL &memmove, csr_aarch64_aapcs, implicit-def $lr, implicit $sp, implicit $x0, implicit $x1, implicit $x2
|
Remove alignment argument from memcpy/memmove/memset in favour of alignment attributes (Step 1)
Summary:
This is a resurrection of work first proposed and discussed in Aug 2015:
http://lists.llvm.org/pipermail/llvm-dev/2015-August/089384.html
and initially landed (but then backed out) in Nov 2015:
http://lists.llvm.org/pipermail/llvm-commits/Week-of-Mon-20151109/312083.html
The @llvm.memcpy/memmove/memset intrinsics currently have an explicit argument
which is required to be a constant integer. It represents the alignment of the
dest (and source), and so must be the minimum of the actual alignment of the
two.
This change is the first in a series that allows source and dest to each
have their own alignments by using the alignment attribute on their arguments.
In this change we:
1) Remove the alignment argument.
2) Add alignment attributes to the source & dest arguments. We, temporarily,
require that the alignments for source & dest be equal.
For example, code which used to read:
call void @llvm.memcpy.p0i8.p0i8.i32(i8* %dest, i8* %src, i32 100, i32 4, i1 false)
will now read
call void @llvm.memcpy.p0i8.p0i8.i32(i8* align 4 %dest, i8* align 4 %src, i32 100, i1 false)
Downstream users may have to update their lit tests that check for
@llvm.memcpy/memmove/memset call/declaration patterns. The following extended sed script
may help with updating the majority of your tests, but it does not catch all possible
patterns so some manual checking and updating will be required.
s~declare void @llvm\.mem(set|cpy|move)\.p([^(]*)\((.*), i32, i1\)~declare void @llvm.mem\1.p\2(\3, i1)~g
s~call void @llvm\.memset\.p([^(]*)i8\(i8([^*]*)\* (.*), i8 (.*), i8 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.memset.p\1i8(i8\2* \3, i8 \4, i8 \5, i1 \6)~g
s~call void @llvm\.memset\.p([^(]*)i16\(i8([^*]*)\* (.*), i8 (.*), i16 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.memset.p\1i16(i8\2* \3, i8 \4, i16 \5, i1 \6)~g
s~call void @llvm\.memset\.p([^(]*)i32\(i8([^*]*)\* (.*), i8 (.*), i32 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.memset.p\1i32(i8\2* \3, i8 \4, i32 \5, i1 \6)~g
s~call void @llvm\.memset\.p([^(]*)i64\(i8([^*]*)\* (.*), i8 (.*), i64 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.memset.p\1i64(i8\2* \3, i8 \4, i64 \5, i1 \6)~g
s~call void @llvm\.memset\.p([^(]*)i128\(i8([^*]*)\* (.*), i8 (.*), i128 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.memset.p\1i128(i8\2* \3, i8 \4, i128 \5, i1 \6)~g
s~call void @llvm\.memset\.p([^(]*)i8\(i8([^*]*)\* (.*), i8 (.*), i8 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.memset.p\1i8(i8\2* align \6 \3, i8 \4, i8 \5, i1 \7)~g
s~call void @llvm\.memset\.p([^(]*)i16\(i8([^*]*)\* (.*), i8 (.*), i16 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.memset.p\1i16(i8\2* align \6 \3, i8 \4, i16 \5, i1 \7)~g
s~call void @llvm\.memset\.p([^(]*)i32\(i8([^*]*)\* (.*), i8 (.*), i32 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.memset.p\1i32(i8\2* align \6 \3, i8 \4, i32 \5, i1 \7)~g
s~call void @llvm\.memset\.p([^(]*)i64\(i8([^*]*)\* (.*), i8 (.*), i64 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.memset.p\1i64(i8\2* align \6 \3, i8 \4, i64 \5, i1 \7)~g
s~call void @llvm\.memset\.p([^(]*)i128\(i8([^*]*)\* (.*), i8 (.*), i128 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.memset.p\1i128(i8\2* align \6 \3, i8 \4, i128 \5, i1 \7)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i8\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i8 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.mem\1.p\2i8(i8\3* \4, i8\5* \6, i8 \7, i1 \8)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i16\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i16 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.mem\1.p\2i16(i8\3* \4, i8\5* \6, i16 \7, i1 \8)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i32\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i32 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.mem\1.p\2i32(i8\3* \4, i8\5* \6, i32 \7, i1 \8)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i64\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i64 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.mem\1.p\2i64(i8\3* \4, i8\5* \6, i64 \7, i1 \8)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i128\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i128 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.mem\1.p\2i128(i8\3* \4, i8\5* \6, i128 \7, i1 \8)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i8\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i8 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.mem\1.p\2i8(i8\3* align \8 \4, i8\5* align \8 \6, i8 \7, i1 \9)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i16\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i16 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.mem\1.p\2i16(i8\3* align \8 \4, i8\5* align \8 \6, i16 \7, i1 \9)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i32\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i32 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.mem\1.p\2i32(i8\3* align \8 \4, i8\5* align \8 \6, i32 \7, i1 \9)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i64\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i64 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.mem\1.p\2i64(i8\3* align \8 \4, i8\5* align \8 \6, i64 \7, i1 \9)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i128\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i128 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.mem\1.p\2i128(i8\3* align \8 \4, i8\5* align \8 \6, i128 \7, i1 \9)~g
The remaining changes in the series will:
Step 2) Expand the IRBuilder API to allow creation of memcpy/memmove with differing
source and dest alignments.
Step 3) Update Clang to use the new IRBuilder API.
Step 4) Update Polly to use the new IRBuilder API.
Step 5) Update LLVM passes that create memcpy/memmove calls to use the new IRBuilder API,
and those that use use MemIntrinsicInst::[get|set]Alignment() to use
getDestAlignment() and getSourceAlignment() instead.
Step 6) Remove the single-alignment IRBuilder API for memcpy/memmove, and the
MemIntrinsicInst::[get|set]Alignment() methods.
Reviewers: pete, hfinkel, lhames, reames, bollu
Reviewed By: reames
Subscribers: niosHD, reames, jholewinski, qcolombet, jfb, sanjoy, arsenm, dschuff, dylanmckay, mehdi_amini, sdardis, nemanjai, david2050, nhaehnle, javed.absar, sbc100, jgravelle-google, eraman, aheejin, kbarton, JDevlieghere, asb, rbar, johnrusso, simoncook, jordy.potman.lists, apazos, sabuasal, llvm-commits
Differential Revision: https://reviews.llvm.org/D41675
llvm-svn: 322965
2018-01-20 01:13:12 +08:00
|
|
|
call void @llvm.memmove.p0i8.p0i8.i64(i8* %dst, i8* %src, i64 %size, i1 0)
|
2017-01-31 03:33:07 +08:00
|
|
|
ret void
|
|
|
|
}
|
|
|
|
|
Remove alignment argument from memcpy/memmove/memset in favour of alignment attributes (Step 1)
Summary:
This is a resurrection of work first proposed and discussed in Aug 2015:
http://lists.llvm.org/pipermail/llvm-dev/2015-August/089384.html
and initially landed (but then backed out) in Nov 2015:
http://lists.llvm.org/pipermail/llvm-commits/Week-of-Mon-20151109/312083.html
The @llvm.memcpy/memmove/memset intrinsics currently have an explicit argument
which is required to be a constant integer. It represents the alignment of the
dest (and source), and so must be the minimum of the actual alignment of the
two.
This change is the first in a series that allows source and dest to each
have their own alignments by using the alignment attribute on their arguments.
In this change we:
1) Remove the alignment argument.
2) Add alignment attributes to the source & dest arguments. We, temporarily,
require that the alignments for source & dest be equal.
For example, code which used to read:
call void @llvm.memcpy.p0i8.p0i8.i32(i8* %dest, i8* %src, i32 100, i32 4, i1 false)
will now read
call void @llvm.memcpy.p0i8.p0i8.i32(i8* align 4 %dest, i8* align 4 %src, i32 100, i1 false)
Downstream users may have to update their lit tests that check for
@llvm.memcpy/memmove/memset call/declaration patterns. The following extended sed script
may help with updating the majority of your tests, but it does not catch all possible
patterns so some manual checking and updating will be required.
s~declare void @llvm\.mem(set|cpy|move)\.p([^(]*)\((.*), i32, i1\)~declare void @llvm.mem\1.p\2(\3, i1)~g
s~call void @llvm\.memset\.p([^(]*)i8\(i8([^*]*)\* (.*), i8 (.*), i8 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.memset.p\1i8(i8\2* \3, i8 \4, i8 \5, i1 \6)~g
s~call void @llvm\.memset\.p([^(]*)i16\(i8([^*]*)\* (.*), i8 (.*), i16 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.memset.p\1i16(i8\2* \3, i8 \4, i16 \5, i1 \6)~g
s~call void @llvm\.memset\.p([^(]*)i32\(i8([^*]*)\* (.*), i8 (.*), i32 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.memset.p\1i32(i8\2* \3, i8 \4, i32 \5, i1 \6)~g
s~call void @llvm\.memset\.p([^(]*)i64\(i8([^*]*)\* (.*), i8 (.*), i64 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.memset.p\1i64(i8\2* \3, i8 \4, i64 \5, i1 \6)~g
s~call void @llvm\.memset\.p([^(]*)i128\(i8([^*]*)\* (.*), i8 (.*), i128 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.memset.p\1i128(i8\2* \3, i8 \4, i128 \5, i1 \6)~g
s~call void @llvm\.memset\.p([^(]*)i8\(i8([^*]*)\* (.*), i8 (.*), i8 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.memset.p\1i8(i8\2* align \6 \3, i8 \4, i8 \5, i1 \7)~g
s~call void @llvm\.memset\.p([^(]*)i16\(i8([^*]*)\* (.*), i8 (.*), i16 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.memset.p\1i16(i8\2* align \6 \3, i8 \4, i16 \5, i1 \7)~g
s~call void @llvm\.memset\.p([^(]*)i32\(i8([^*]*)\* (.*), i8 (.*), i32 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.memset.p\1i32(i8\2* align \6 \3, i8 \4, i32 \5, i1 \7)~g
s~call void @llvm\.memset\.p([^(]*)i64\(i8([^*]*)\* (.*), i8 (.*), i64 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.memset.p\1i64(i8\2* align \6 \3, i8 \4, i64 \5, i1 \7)~g
s~call void @llvm\.memset\.p([^(]*)i128\(i8([^*]*)\* (.*), i8 (.*), i128 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.memset.p\1i128(i8\2* align \6 \3, i8 \4, i128 \5, i1 \7)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i8\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i8 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.mem\1.p\2i8(i8\3* \4, i8\5* \6, i8 \7, i1 \8)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i16\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i16 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.mem\1.p\2i16(i8\3* \4, i8\5* \6, i16 \7, i1 \8)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i32\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i32 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.mem\1.p\2i32(i8\3* \4, i8\5* \6, i32 \7, i1 \8)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i64\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i64 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.mem\1.p\2i64(i8\3* \4, i8\5* \6, i64 \7, i1 \8)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i128\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i128 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.mem\1.p\2i128(i8\3* \4, i8\5* \6, i128 \7, i1 \8)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i8\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i8 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.mem\1.p\2i8(i8\3* align \8 \4, i8\5* align \8 \6, i8 \7, i1 \9)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i16\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i16 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.mem\1.p\2i16(i8\3* align \8 \4, i8\5* align \8 \6, i16 \7, i1 \9)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i32\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i32 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.mem\1.p\2i32(i8\3* align \8 \4, i8\5* align \8 \6, i32 \7, i1 \9)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i64\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i64 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.mem\1.p\2i64(i8\3* align \8 \4, i8\5* align \8 \6, i64 \7, i1 \9)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i128\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i128 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.mem\1.p\2i128(i8\3* align \8 \4, i8\5* align \8 \6, i128 \7, i1 \9)~g
The remaining changes in the series will:
Step 2) Expand the IRBuilder API to allow creation of memcpy/memmove with differing
source and dest alignments.
Step 3) Update Clang to use the new IRBuilder API.
Step 4) Update Polly to use the new IRBuilder API.
Step 5) Update LLVM passes that create memcpy/memmove calls to use the new IRBuilder API,
and those that use use MemIntrinsicInst::[get|set]Alignment() to use
getDestAlignment() and getSourceAlignment() instead.
Step 6) Remove the single-alignment IRBuilder API for memcpy/memmove, and the
MemIntrinsicInst::[get|set]Alignment() methods.
Reviewers: pete, hfinkel, lhames, reames, bollu
Reviewed By: reames
Subscribers: niosHD, reames, jholewinski, qcolombet, jfb, sanjoy, arsenm, dschuff, dylanmckay, mehdi_amini, sdardis, nemanjai, david2050, nhaehnle, javed.absar, sbc100, jgravelle-google, eraman, aheejin, kbarton, JDevlieghere, asb, rbar, johnrusso, simoncook, jordy.potman.lists, apazos, sabuasal, llvm-commits
Differential Revision: https://reviews.llvm.org/D41675
llvm-svn: 322965
2018-01-20 01:13:12 +08:00
|
|
|
declare void @llvm.memset.p0i8.i64(i8*, i8, i64, i1)
|
2017-01-31 03:33:07 +08:00
|
|
|
define void @test_memset(i8* %dst, i8 %val, i64 %size) {
|
|
|
|
; CHECK-LABEL: name: test_memset
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[DST:%[0-9]+]]:_(p0) = COPY $x0
|
|
|
|
; CHECK: [[SRC_C:%[0-9]+]]:_(s32) = COPY $w1
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[SRC:%[0-9]+]]:_(s8) = G_TRUNC [[SRC_C]]
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[SIZE:%[0-9]+]]:_(s64) = COPY $x2
|
|
|
|
; CHECK: $x0 = COPY [[DST]]
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[SRC_TMP:%[0-9]+]]:_(s32) = G_ANYEXT [[SRC]]
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: $w1 = COPY [[SRC_TMP]]
|
|
|
|
; CHECK: $x2 = COPY [[SIZE]]
|
|
|
|
; CHECK: BL &memset, csr_aarch64_aapcs, implicit-def $lr, implicit $sp, implicit $x0, implicit $w1, implicit $x2
|
Remove alignment argument from memcpy/memmove/memset in favour of alignment attributes (Step 1)
Summary:
This is a resurrection of work first proposed and discussed in Aug 2015:
http://lists.llvm.org/pipermail/llvm-dev/2015-August/089384.html
and initially landed (but then backed out) in Nov 2015:
http://lists.llvm.org/pipermail/llvm-commits/Week-of-Mon-20151109/312083.html
The @llvm.memcpy/memmove/memset intrinsics currently have an explicit argument
which is required to be a constant integer. It represents the alignment of the
dest (and source), and so must be the minimum of the actual alignment of the
two.
This change is the first in a series that allows source and dest to each
have their own alignments by using the alignment attribute on their arguments.
In this change we:
1) Remove the alignment argument.
2) Add alignment attributes to the source & dest arguments. We, temporarily,
require that the alignments for source & dest be equal.
For example, code which used to read:
call void @llvm.memcpy.p0i8.p0i8.i32(i8* %dest, i8* %src, i32 100, i32 4, i1 false)
will now read
call void @llvm.memcpy.p0i8.p0i8.i32(i8* align 4 %dest, i8* align 4 %src, i32 100, i1 false)
Downstream users may have to update their lit tests that check for
@llvm.memcpy/memmove/memset call/declaration patterns. The following extended sed script
may help with updating the majority of your tests, but it does not catch all possible
patterns so some manual checking and updating will be required.
s~declare void @llvm\.mem(set|cpy|move)\.p([^(]*)\((.*), i32, i1\)~declare void @llvm.mem\1.p\2(\3, i1)~g
s~call void @llvm\.memset\.p([^(]*)i8\(i8([^*]*)\* (.*), i8 (.*), i8 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.memset.p\1i8(i8\2* \3, i8 \4, i8 \5, i1 \6)~g
s~call void @llvm\.memset\.p([^(]*)i16\(i8([^*]*)\* (.*), i8 (.*), i16 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.memset.p\1i16(i8\2* \3, i8 \4, i16 \5, i1 \6)~g
s~call void @llvm\.memset\.p([^(]*)i32\(i8([^*]*)\* (.*), i8 (.*), i32 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.memset.p\1i32(i8\2* \3, i8 \4, i32 \5, i1 \6)~g
s~call void @llvm\.memset\.p([^(]*)i64\(i8([^*]*)\* (.*), i8 (.*), i64 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.memset.p\1i64(i8\2* \3, i8 \4, i64 \5, i1 \6)~g
s~call void @llvm\.memset\.p([^(]*)i128\(i8([^*]*)\* (.*), i8 (.*), i128 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.memset.p\1i128(i8\2* \3, i8 \4, i128 \5, i1 \6)~g
s~call void @llvm\.memset\.p([^(]*)i8\(i8([^*]*)\* (.*), i8 (.*), i8 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.memset.p\1i8(i8\2* align \6 \3, i8 \4, i8 \5, i1 \7)~g
s~call void @llvm\.memset\.p([^(]*)i16\(i8([^*]*)\* (.*), i8 (.*), i16 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.memset.p\1i16(i8\2* align \6 \3, i8 \4, i16 \5, i1 \7)~g
s~call void @llvm\.memset\.p([^(]*)i32\(i8([^*]*)\* (.*), i8 (.*), i32 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.memset.p\1i32(i8\2* align \6 \3, i8 \4, i32 \5, i1 \7)~g
s~call void @llvm\.memset\.p([^(]*)i64\(i8([^*]*)\* (.*), i8 (.*), i64 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.memset.p\1i64(i8\2* align \6 \3, i8 \4, i64 \5, i1 \7)~g
s~call void @llvm\.memset\.p([^(]*)i128\(i8([^*]*)\* (.*), i8 (.*), i128 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.memset.p\1i128(i8\2* align \6 \3, i8 \4, i128 \5, i1 \7)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i8\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i8 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.mem\1.p\2i8(i8\3* \4, i8\5* \6, i8 \7, i1 \8)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i16\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i16 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.mem\1.p\2i16(i8\3* \4, i8\5* \6, i16 \7, i1 \8)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i32\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i32 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.mem\1.p\2i32(i8\3* \4, i8\5* \6, i32 \7, i1 \8)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i64\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i64 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.mem\1.p\2i64(i8\3* \4, i8\5* \6, i64 \7, i1 \8)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i128\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i128 (.*), i32 [01], i1 ([^)]*)\)~call void @llvm.mem\1.p\2i128(i8\3* \4, i8\5* \6, i128 \7, i1 \8)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i8\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i8 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.mem\1.p\2i8(i8\3* align \8 \4, i8\5* align \8 \6, i8 \7, i1 \9)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i16\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i16 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.mem\1.p\2i16(i8\3* align \8 \4, i8\5* align \8 \6, i16 \7, i1 \9)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i32\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i32 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.mem\1.p\2i32(i8\3* align \8 \4, i8\5* align \8 \6, i32 \7, i1 \9)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i64\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i64 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.mem\1.p\2i64(i8\3* align \8 \4, i8\5* align \8 \6, i64 \7, i1 \9)~g
s~call void @llvm\.mem(cpy|move)\.p([^(]*)i128\(i8([^*]*)\* (.*), i8([^*]*)\* (.*), i128 (.*), i32 ([0-9]*), i1 ([^)]*)\)~call void @llvm.mem\1.p\2i128(i8\3* align \8 \4, i8\5* align \8 \6, i128 \7, i1 \9)~g
The remaining changes in the series will:
Step 2) Expand the IRBuilder API to allow creation of memcpy/memmove with differing
source and dest alignments.
Step 3) Update Clang to use the new IRBuilder API.
Step 4) Update Polly to use the new IRBuilder API.
Step 5) Update LLVM passes that create memcpy/memmove calls to use the new IRBuilder API,
and those that use use MemIntrinsicInst::[get|set]Alignment() to use
getDestAlignment() and getSourceAlignment() instead.
Step 6) Remove the single-alignment IRBuilder API for memcpy/memmove, and the
MemIntrinsicInst::[get|set]Alignment() methods.
Reviewers: pete, hfinkel, lhames, reames, bollu
Reviewed By: reames
Subscribers: niosHD, reames, jholewinski, qcolombet, jfb, sanjoy, arsenm, dschuff, dylanmckay, mehdi_amini, sdardis, nemanjai, david2050, nhaehnle, javed.absar, sbc100, jgravelle-google, eraman, aheejin, kbarton, JDevlieghere, asb, rbar, johnrusso, simoncook, jordy.potman.lists, apazos, sabuasal, llvm-commits
Differential Revision: https://reviews.llvm.org/D41675
llvm-svn: 322965
2018-01-20 01:13:12 +08:00
|
|
|
call void @llvm.memset.p0i8.i64(i8* %dst, i8 %val, i64 %size, i1 0)
|
2017-01-31 03:33:07 +08:00
|
|
|
ret void
|
|
|
|
}
|
|
|
|
|
2016-10-19 04:03:51 +08:00
|
|
|
declare i64 @llvm.objectsize.i64(i8*, i1)
|
|
|
|
declare i32 @llvm.objectsize.i32(i8*, i1)
|
|
|
|
define void @test_objectsize(i8* %addr0, i8* %addr1) {
|
|
|
|
; CHECK-LABEL: name: test_objectsize
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[ADDR0:%[0-9]+]]:_(p0) = COPY $x0
|
|
|
|
; CHECK: [[ADDR1:%[0-9]+]]:_(p0) = COPY $x1
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: {{%[0-9]+}}:_(s64) = G_CONSTANT i64 -1
|
|
|
|
; CHECK: {{%[0-9]+}}:_(s64) = G_CONSTANT i64 0
|
|
|
|
; CHECK: {{%[0-9]+}}:_(s32) = G_CONSTANT i32 -1
|
|
|
|
; CHECK: {{%[0-9]+}}:_(s32) = G_CONSTANT i32 0
|
2016-10-19 04:03:51 +08:00
|
|
|
%size64.0 = call i64 @llvm.objectsize.i64(i8* %addr0, i1 0)
|
|
|
|
%size64.intmin = call i64 @llvm.objectsize.i64(i8* %addr0, i1 1)
|
|
|
|
%size32.0 = call i32 @llvm.objectsize.i32(i8* %addr0, i1 0)
|
|
|
|
%size32.intmin = call i32 @llvm.objectsize.i32(i8* %addr0, i1 1)
|
|
|
|
ret void
|
|
|
|
}
|
2016-12-06 05:54:17 +08:00
|
|
|
|
|
|
|
define void @test_large_const(i128* %addr) {
|
|
|
|
; CHECK-LABEL: name: test_large_const
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[ADDR:%[0-9]+]]:_(p0) = COPY $x0
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[VAL:%[0-9]+]]:_(s128) = G_CONSTANT i128 42
|
2016-12-06 05:54:17 +08:00
|
|
|
; CHECK: G_STORE [[VAL]](s128), [[ADDR]](p0)
|
|
|
|
store i128 42, i128* %addr
|
|
|
|
ret void
|
|
|
|
}
|
2016-12-06 06:40:13 +08:00
|
|
|
|
|
|
|
; When there was no formal argument handling (so the first BB was empty) we used
|
|
|
|
; to insert the constants at the end of the block, even if they were encountered
|
2016-12-08 05:05:38 +08:00
|
|
|
; after the block's terminators had been emitted. Also make sure the order is
|
|
|
|
; correct.
|
|
|
|
define i8* @test_const_placement() {
|
2016-12-06 06:40:13 +08:00
|
|
|
; CHECK-LABEL: name: test_const_placement
|
2017-01-05 21:27:52 +08:00
|
|
|
; CHECK: bb.{{[0-9]+}} (%ir-block.{{[0-9]+}}):
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[VAL_INT:%[0-9]+]]:_(s32) = G_CONSTANT i32 42
|
|
|
|
; CHECK: [[VAL:%[0-9]+]]:_(p0) = G_INTTOPTR [[VAL_INT]](s32)
|
2017-12-05 01:18:51 +08:00
|
|
|
; CHECK: bb.{{[0-9]+}}.{{[a-zA-Z0-9.]+}}:
|
2016-12-06 06:40:13 +08:00
|
|
|
br label %next
|
|
|
|
|
|
|
|
next:
|
2016-12-08 05:05:38 +08:00
|
|
|
ret i8* inttoptr(i32 42 to i8*)
|
2016-12-06 06:40:13 +08:00
|
|
|
}
|
2017-02-08 04:08:59 +08:00
|
|
|
|
|
|
|
declare void @llvm.va_end(i8*)
|
|
|
|
define void @test_va_end(i8* %list) {
|
|
|
|
; CHECK-LABEL: name: test_va_end
|
|
|
|
; CHECK-NOT: va_end
|
|
|
|
; CHECK-NOT: INTRINSIC
|
|
|
|
; CHECK: RET_ReallyLR
|
|
|
|
call void @llvm.va_end(i8* %list)
|
|
|
|
ret void
|
|
|
|
}
|
2017-02-09 07:23:32 +08:00
|
|
|
|
2017-02-16 07:22:33 +08:00
|
|
|
define void @test_va_arg(i8* %list) {
|
|
|
|
; CHECK-LABEL: test_va_arg
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[LIST:%[0-9]+]]:_(p0) = COPY $x0
|
2017-02-16 07:22:33 +08:00
|
|
|
; CHECK: G_VAARG [[LIST]](p0), 8
|
|
|
|
; CHECK: G_VAARG [[LIST]](p0), 1
|
|
|
|
; CHECK: G_VAARG [[LIST]](p0), 16
|
|
|
|
|
|
|
|
%v0 = va_arg i8* %list, i64
|
|
|
|
%v1 = va_arg i8* %list, i8
|
|
|
|
%v2 = va_arg i8* %list, i128
|
|
|
|
ret void
|
|
|
|
}
|
|
|
|
|
2017-02-09 07:23:32 +08:00
|
|
|
declare float @llvm.pow.f32(float, float)
|
|
|
|
define float @test_pow_intrin(float %l, float %r) {
|
|
|
|
; CHECK-LABEL: name: test_pow_intrin
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[LHS:%[0-9]+]]:_(s32) = COPY $s0
|
|
|
|
; CHECK: [[RHS:%[0-9]+]]:_(s32) = COPY $s1
|
2018-12-19 01:54:52 +08:00
|
|
|
; CHECK: [[RES:%[0-9]+]]:_(s32) = nnan ninf nsz arcp contract afn reassoc G_FPOW [[LHS]], [[RHS]]
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: $s0 = COPY [[RES]]
|
2018-12-19 01:54:52 +08:00
|
|
|
%res = call nnan ninf nsz arcp contract afn reassoc float @llvm.pow.f32(float %l, float %r)
|
2017-02-09 07:23:32 +08:00
|
|
|
ret float %res
|
|
|
|
}
|
2017-02-11 03:10:38 +08:00
|
|
|
|
2017-06-21 03:25:23 +08:00
|
|
|
declare float @llvm.fma.f32(float, float, float)
|
|
|
|
define float @test_fma_intrin(float %a, float %b, float %c) {
|
|
|
|
; CHECK-LABEL: name: test_fma_intrin
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[A:%[0-9]+]]:_(s32) = COPY $s0
|
|
|
|
; CHECK: [[B:%[0-9]+]]:_(s32) = COPY $s1
|
|
|
|
; CHECK: [[C:%[0-9]+]]:_(s32) = COPY $s2
|
2018-12-19 01:54:52 +08:00
|
|
|
; CHECK: [[RES:%[0-9]+]]:_(s32) = nnan ninf nsz arcp contract afn reassoc G_FMA [[A]], [[B]], [[C]]
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: $s0 = COPY [[RES]]
|
2018-12-19 01:54:52 +08:00
|
|
|
%res = call nnan ninf nsz arcp contract afn reassoc float @llvm.fma.f32(float %a, float %b, float %c)
|
2017-06-21 03:25:23 +08:00
|
|
|
ret float %res
|
|
|
|
}
|
|
|
|
|
2017-06-28 06:19:32 +08:00
|
|
|
declare float @llvm.exp.f32(float)
|
|
|
|
define float @test_exp_intrin(float %a) {
|
|
|
|
; CHECK-LABEL: name: test_exp_intrin
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[A:%[0-9]+]]:_(s32) = COPY $s0
|
2018-12-19 01:54:52 +08:00
|
|
|
; CHECK: [[RES:%[0-9]+]]:_(s32) = nnan ninf nsz arcp contract afn reassoc G_FEXP [[A]]
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: $s0 = COPY [[RES]]
|
2018-12-19 01:54:52 +08:00
|
|
|
%res = call nnan ninf nsz arcp contract afn reassoc float @llvm.exp.f32(float %a)
|
2017-06-28 06:19:32 +08:00
|
|
|
ret float %res
|
|
|
|
}
|
|
|
|
|
|
|
|
declare float @llvm.exp2.f32(float)
|
|
|
|
define float @test_exp2_intrin(float %a) {
|
|
|
|
; CHECK-LABEL: name: test_exp2_intrin
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[A:%[0-9]+]]:_(s32) = COPY $s0
|
2018-12-19 01:54:52 +08:00
|
|
|
; CHECK: [[RES:%[0-9]+]]:_(s32) = nnan ninf nsz arcp contract afn reassoc G_FEXP2 [[A]]
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: $s0 = COPY [[RES]]
|
2018-12-19 01:54:52 +08:00
|
|
|
%res = call nnan ninf nsz arcp contract afn reassoc float @llvm.exp2.f32(float %a)
|
2017-06-28 06:19:32 +08:00
|
|
|
ret float %res
|
|
|
|
}
|
|
|
|
|
2017-06-30 07:43:44 +08:00
|
|
|
declare float @llvm.log.f32(float)
|
|
|
|
define float @test_log_intrin(float %a) {
|
|
|
|
; CHECK-LABEL: name: test_log_intrin
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[A:%[0-9]+]]:_(s32) = COPY $s0
|
2018-12-19 01:54:52 +08:00
|
|
|
; CHECK: [[RES:%[0-9]+]]:_(s32) = nnan ninf nsz arcp contract afn reassoc G_FLOG [[A]]
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: $s0 = COPY [[RES]]
|
2018-12-19 01:54:52 +08:00
|
|
|
%res = call nnan ninf nsz arcp contract afn reassoc float @llvm.log.f32(float %a)
|
2017-06-30 07:43:44 +08:00
|
|
|
ret float %res
|
|
|
|
}
|
|
|
|
|
|
|
|
declare float @llvm.log2.f32(float)
|
|
|
|
define float @test_log2_intrin(float %a) {
|
|
|
|
; CHECK-LABEL: name: test_log2_intrin
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[A:%[0-9]+]]:_(s32) = COPY $s0
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[RES:%[0-9]+]]:_(s32) = G_FLOG2 [[A]]
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: $s0 = COPY [[RES]]
|
2017-06-30 07:43:44 +08:00
|
|
|
%res = call float @llvm.log2.f32(float %a)
|
|
|
|
ret float %res
|
|
|
|
}
|
2018-03-06 06:31:55 +08:00
|
|
|
|
2018-12-08 06:08:02 +08:00
|
|
|
declare float @llvm.log10.f32(float)
|
|
|
|
define float @test_log10_intrin(float %a) {
|
|
|
|
; CHECK-LABEL: name: test_log10_intrin
|
|
|
|
; CHECK: [[A:%[0-9]+]]:_(s32) = COPY $s0
|
2018-12-19 01:54:52 +08:00
|
|
|
; CHECK: [[RES:%[0-9]+]]:_(s32) = nnan ninf nsz arcp contract afn reassoc G_FLOG10 [[A]]
|
2018-12-08 06:08:02 +08:00
|
|
|
; CHECK: $s0 = COPY [[RES]]
|
2018-12-19 01:54:52 +08:00
|
|
|
%res = call nnan ninf nsz arcp contract afn reassoc float @llvm.log10.f32(float %a)
|
2018-12-08 06:08:02 +08:00
|
|
|
ret float %res
|
|
|
|
}
|
|
|
|
|
2018-03-06 06:31:55 +08:00
|
|
|
declare float @llvm.fabs.f32(float)
|
|
|
|
define float @test_fabs_intrin(float %a) {
|
|
|
|
; CHECK-LABEL: name: test_fabs_intrin
|
|
|
|
; CHECK: [[A:%[0-9]+]]:_(s32) = COPY $s0
|
2018-12-19 01:54:52 +08:00
|
|
|
; CHECK: [[RES:%[0-9]+]]:_(s32) = nnan ninf nsz arcp contract afn reassoc G_FABS [[A]]
|
2018-03-06 06:31:55 +08:00
|
|
|
; CHECK: $s0 = COPY [[RES]]
|
2018-12-19 01:54:52 +08:00
|
|
|
%res = call nnan ninf nsz arcp contract afn reassoc float @llvm.fabs.f32(float %a)
|
2018-03-06 06:31:55 +08:00
|
|
|
ret float %res
|
|
|
|
}
|
|
|
|
|
2018-08-21 02:43:19 +08:00
|
|
|
declare float @llvm.trunc.f32(float)
|
|
|
|
define float @test_intrinsic_trunc(float %a) {
|
|
|
|
; CHECK-LABEL: name: test_intrinsic_trunc
|
|
|
|
; CHECK: [[A:%[0-9]+]]:_(s32) = COPY $s0
|
|
|
|
; CHECK: [[RES:%[0-9]+]]:_(s32) = G_INTRINSIC_TRUNC [[A]]
|
|
|
|
; CHECK: $s0 = COPY [[RES]]
|
|
|
|
%res = call float @llvm.trunc.f32(float %a)
|
|
|
|
ret float %res
|
|
|
|
}
|
|
|
|
|
|
|
|
declare float @llvm.round.f32(float)
|
|
|
|
define float @test_intrinsic_round(float %a) {
|
|
|
|
; CHECK-LABEL: name: test_intrinsic_round
|
|
|
|
; CHECK: [[A:%[0-9]+]]:_(s32) = COPY $s0
|
|
|
|
; CHECK: [[RES:%[0-9]+]]:_(s32) = G_INTRINSIC_ROUND [[A]]
|
|
|
|
; CHECK: $s0 = COPY [[RES]]
|
|
|
|
%res = call float @llvm.round.f32(float %a)
|
|
|
|
ret float %res
|
|
|
|
}
|
|
|
|
|
2018-08-04 09:22:12 +08:00
|
|
|
declare i32 @llvm.ctlz.i32(i32, i1)
|
|
|
|
define i32 @test_ctlz_intrinsic_zero_not_undef(i32 %a) {
|
|
|
|
; CHECK-LABEL: name: test_ctlz_intrinsic_zero_not_undef
|
|
|
|
; CHECK: [[A:%[0-9]+]]:_(s32) = COPY $w0
|
|
|
|
; CHECK: [[RES:%[0-9]+]]:_(s32) = G_CTLZ [[A]]
|
|
|
|
; CHECK: $w0 = COPY [[RES]]
|
|
|
|
%res = call i32 @llvm.ctlz.i32(i32 %a, i1 0)
|
|
|
|
ret i32 %res
|
|
|
|
}
|
|
|
|
|
|
|
|
declare i32 @llvm.cttz.i32(i32, i1)
|
|
|
|
define i32 @test_cttz_intrinsic_zero_undef(i32 %a) {
|
|
|
|
; CHECK-LABEL: name: test_cttz_intrinsic_zero_undef
|
|
|
|
; CHECK: [[A:%[0-9]+]]:_(s32) = COPY $w0
|
|
|
|
; CHECK: [[RES:%[0-9]+]]:_(s32) = G_CTTZ_ZERO_UNDEF [[A]]
|
|
|
|
; CHECK: $w0 = COPY [[RES]]
|
|
|
|
%res = call i32 @llvm.cttz.i32(i32 %a, i1 1)
|
|
|
|
ret i32 %res
|
|
|
|
}
|
|
|
|
|
|
|
|
declare i32 @llvm.ctpop.i32(i32)
|
|
|
|
define i32 @test_ctpop_intrinsic(i32 %a) {
|
|
|
|
; CHECK-LABEL: name: test_ctpop
|
|
|
|
; CHECK: [[A:%[0-9]+]]:_(s32) = COPY $w0
|
|
|
|
; CHECK: [[RES:%[0-9]+]]:_(s32) = G_CTPOP [[A]]
|
|
|
|
; CHECK: $w0 = COPY [[RES]]
|
|
|
|
%res = call i32 @llvm.ctpop.i32(i32 %a)
|
|
|
|
ret i32 %res
|
|
|
|
}
|
|
|
|
|
2017-04-11 04:18:21 +08:00
|
|
|
declare void @llvm.lifetime.start.p0i8(i64, i8*)
|
|
|
|
declare void @llvm.lifetime.end.p0i8(i64, i8*)
|
2017-02-11 03:10:38 +08:00
|
|
|
define void @test_lifetime_intrin() {
|
|
|
|
; CHECK-LABEL: name: test_lifetime_intrin
|
|
|
|
; CHECK: RET_ReallyLR
|
2019-01-29 03:22:29 +08:00
|
|
|
; O3-LABEL: name: test_lifetime_intrin
|
|
|
|
; O3: {{%[0-9]+}}:_(p0) = G_FRAME_INDEX %stack.0.slot
|
|
|
|
; O3-NEXT: LIFETIME_START %stack.0.slot
|
|
|
|
; O3-NEXT: LIFETIME_END %stack.0.slot
|
|
|
|
; O3-NEXT: RET_ReallyLR
|
2017-02-11 03:10:38 +08:00
|
|
|
%slot = alloca i8, i32 4
|
2017-04-11 04:18:21 +08:00
|
|
|
call void @llvm.lifetime.start.p0i8(i64 0, i8* %slot)
|
|
|
|
call void @llvm.lifetime.end.p0i8(i64 0, i8* %slot)
|
2017-02-11 03:10:38 +08:00
|
|
|
ret void
|
|
|
|
}
|
2017-02-14 06:14:16 +08:00
|
|
|
|
|
|
|
define void @test_load_store_atomics(i8* %addr) {
|
|
|
|
; CHECK-LABEL: name: test_load_store_atomics
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[ADDR:%[0-9]+]]:_(p0) = COPY $x0
|
2017-12-05 13:52:07 +08:00
|
|
|
; CHECK: [[V0:%[0-9]+]]:_(s8) = G_LOAD [[ADDR]](p0) :: (load unordered 1 from %ir.addr)
|
|
|
|
; CHECK: G_STORE [[V0]](s8), [[ADDR]](p0) :: (store monotonic 1 into %ir.addr)
|
|
|
|
; CHECK: [[V1:%[0-9]+]]:_(s8) = G_LOAD [[ADDR]](p0) :: (load acquire 1 from %ir.addr)
|
|
|
|
; CHECK: G_STORE [[V1]](s8), [[ADDR]](p0) :: (store release 1 into %ir.addr)
|
|
|
|
; CHECK: [[V2:%[0-9]+]]:_(s8) = G_LOAD [[ADDR]](p0) :: (load syncscope("singlethread") seq_cst 1 from %ir.addr)
|
|
|
|
; CHECK: G_STORE [[V2]](s8), [[ADDR]](p0) :: (store syncscope("singlethread") monotonic 1 into %ir.addr)
|
2017-02-14 06:14:16 +08:00
|
|
|
%v0 = load atomic i8, i8* %addr unordered, align 1
|
|
|
|
store atomic i8 %v0, i8* %addr monotonic, align 1
|
|
|
|
|
|
|
|
%v1 = load atomic i8, i8* %addr acquire, align 1
|
|
|
|
store atomic i8 %v1, i8* %addr release, align 1
|
|
|
|
|
2017-07-12 06:23:00 +08:00
|
|
|
%v2 = load atomic i8, i8* %addr syncscope("singlethread") seq_cst, align 1
|
|
|
|
store atomic i8 %v2, i8* %addr syncscope("singlethread") monotonic, align 1
|
2017-02-14 06:14:16 +08:00
|
|
|
|
|
|
|
ret void
|
|
|
|
}
|
2017-03-08 02:03:28 +08:00
|
|
|
|
|
|
|
define float @test_fneg_f32(float %x) {
|
|
|
|
; CHECK-LABEL: name: test_fneg_f32
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[ARG:%[0-9]+]]:_(s32) = COPY $s0
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[RES:%[0-9]+]]:_(s32) = G_FNEG [[ARG]]
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: $s0 = COPY [[RES]](s32)
|
2017-03-08 02:03:28 +08:00
|
|
|
%neg = fsub float -0.000000e+00, %x
|
|
|
|
ret float %neg
|
|
|
|
}
|
|
|
|
|
|
|
|
define double @test_fneg_f64(double %x) {
|
|
|
|
; CHECK-LABEL: name: test_fneg_f64
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[ARG:%[0-9]+]]:_(s64) = COPY $d0
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[RES:%[0-9]+]]:_(s64) = G_FNEG [[ARG]]
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: $d0 = COPY [[RES]](s64)
|
2017-03-08 02:03:28 +08:00
|
|
|
%neg = fsub double -0.000000e+00, %x
|
|
|
|
ret double %neg
|
|
|
|
}
|
2017-03-10 07:36:26 +08:00
|
|
|
|
|
|
|
define void @test_trivial_inlineasm() {
|
|
|
|
; CHECK-LABEL: name: test_trivial_inlineasm
|
2018-01-10 08:56:48 +08:00
|
|
|
; CHECK: INLINEASM &wibble, 1
|
|
|
|
; CHECK: INLINEASM &wibble, 0
|
2017-03-10 07:36:26 +08:00
|
|
|
call void asm sideeffect "wibble", ""()
|
|
|
|
call void asm "wibble", ""()
|
|
|
|
ret void
|
|
|
|
}
|
2017-03-11 03:08:28 +08:00
|
|
|
|
|
|
|
define <2 x i32> @test_insertelement(<2 x i32> %vec, i32 %elt, i32 %idx){
|
|
|
|
; CHECK-LABEL: name: test_insertelement
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[VEC:%[0-9]+]]:_(<2 x s32>) = COPY $d0
|
|
|
|
; CHECK: [[ELT:%[0-9]+]]:_(s32) = COPY $w0
|
|
|
|
; CHECK: [[IDX:%[0-9]+]]:_(s32) = COPY $w1
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[RES:%[0-9]+]]:_(<2 x s32>) = G_INSERT_VECTOR_ELT [[VEC]], [[ELT]](s32), [[IDX]](s32)
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: $d0 = COPY [[RES]](<2 x s32>)
|
2017-03-11 03:08:28 +08:00
|
|
|
%res = insertelement <2 x i32> %vec, i32 %elt, i32 %idx
|
|
|
|
ret <2 x i32> %res
|
|
|
|
}
|
|
|
|
|
|
|
|
define i32 @test_extractelement(<2 x i32> %vec, i32 %idx) {
|
|
|
|
; CHECK-LABEL: name: test_extractelement
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[VEC:%[0-9]+]]:_(<2 x s32>) = COPY $d0
|
|
|
|
; CHECK: [[IDX:%[0-9]+]]:_(s32) = COPY $w0
|
2018-10-25 22:04:54 +08:00
|
|
|
; CHECK: [[IDXEXT:%[0-9]+]]:_(s64) = G_SEXT [[IDX]]
|
|
|
|
; CHECK: [[RES:%[0-9]+]]:_(s32) = G_EXTRACT_VECTOR_ELT [[VEC]](<2 x s32>), [[IDXEXT]](s64)
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: $w0 = COPY [[RES]](s32)
|
2017-03-11 03:08:28 +08:00
|
|
|
%res = extractelement <2 x i32> %vec, i32 %idx
|
|
|
|
ret i32 %res
|
|
|
|
}
|
|
|
|
|
2018-10-25 22:04:54 +08:00
|
|
|
define i32 @test_extractelement_const_idx(<2 x i32> %vec) {
|
|
|
|
; CHECK-LABEL: name: test_extractelement
|
|
|
|
; CHECK: [[VEC:%[0-9]+]]:_(<2 x s32>) = COPY $d0
|
|
|
|
; CHECK: [[IDX:%[0-9]+]]:_(s64) = G_CONSTANT i64 1
|
|
|
|
; CHECK: [[RES:%[0-9]+]]:_(s32) = G_EXTRACT_VECTOR_ELT [[VEC]](<2 x s32>), [[IDX]](s64)
|
|
|
|
; CHECK: $w0 = COPY [[RES]](s32)
|
|
|
|
%res = extractelement <2 x i32> %vec, i32 1
|
|
|
|
ret i32 %res
|
|
|
|
}
|
|
|
|
|
2017-03-11 03:08:28 +08:00
|
|
|
define i32 @test_singleelementvector(i32 %elt){
|
|
|
|
; CHECK-LABEL: name: test_singleelementvector
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[ELT:%[0-9]+]]:_(s32) = COPY $w0
|
2017-03-11 03:08:28 +08:00
|
|
|
; CHECK-NOT: G_INSERT_VECTOR_ELT
|
|
|
|
; CHECK-NOT: G_EXTRACT_VECTOR_ELT
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: $w0 = COPY [[ELT]](s32)
|
2017-03-11 03:08:28 +08:00
|
|
|
%vec = insertelement <1 x i32> undef, i32 %elt, i32 0
|
|
|
|
%res = extractelement <1 x i32> %vec, i32 0
|
|
|
|
ret i32 %res
|
|
|
|
}
|
2017-03-11 05:23:13 +08:00
|
|
|
|
|
|
|
define <2 x i32> @test_constantaggzerovector_v2i32() {
|
|
|
|
; CHECK-LABEL: name: test_constantaggzerovector_v2i32
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[ZERO:%[0-9]+]]:_(s32) = G_CONSTANT i32 0
|
2018-12-11 02:44:58 +08:00
|
|
|
; CHECK: [[VEC:%[0-9]+]]:_(<2 x s32>) = G_BUILD_VECTOR [[ZERO]](s32), [[ZERO]](s32)
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: $d0 = COPY [[VEC]](<2 x s32>)
|
2017-03-11 05:23:13 +08:00
|
|
|
ret <2 x i32> zeroinitializer
|
|
|
|
}
|
|
|
|
|
|
|
|
define <2 x float> @test_constantaggzerovector_v2f32() {
|
|
|
|
; CHECK-LABEL: name: test_constantaggzerovector_v2f32
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[ZERO:%[0-9]+]]:_(s32) = G_FCONSTANT float 0.000000e+00
|
2018-12-11 02:44:58 +08:00
|
|
|
; CHECK: [[VEC:%[0-9]+]]:_(<2 x s32>) = G_BUILD_VECTOR [[ZERO]](s32), [[ZERO]](s32)
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: $d0 = COPY [[VEC]](<2 x s32>)
|
2017-03-11 05:23:13 +08:00
|
|
|
ret <2 x float> zeroinitializer
|
|
|
|
}
|
|
|
|
|
|
|
|
define i32 @test_constantaggzerovector_v3i32() {
|
|
|
|
; CHECK-LABEL: name: test_constantaggzerovector_v3i32
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[ZERO:%[0-9]+]]:_(s32) = G_CONSTANT i32 0
|
2018-12-11 02:44:58 +08:00
|
|
|
; CHECK: [[VEC:%[0-9]+]]:_(<3 x s32>) = G_BUILD_VECTOR [[ZERO]](s32), [[ZERO]](s32), [[ZERO]](s32)
|
2017-03-11 05:23:13 +08:00
|
|
|
; CHECK: G_EXTRACT_VECTOR_ELT [[VEC]](<3 x s32>)
|
|
|
|
%elt = extractelement <3 x i32> zeroinitializer, i32 1
|
|
|
|
ret i32 %elt
|
|
|
|
}
|
GlobalISel: Translate ConstantDataVector
Reviewers: qcolombet, aditya_nandakumar, dsanders, t.p.northover, javed.absar, ab
Reviewed By: qcolombet, dsanders, ab
Subscribers: dberris, rovka, llvm-commits, kristof.beyls
Differential Revision: https://reviews.llvm.org/D30216
llvm-svn: 297670
2017-03-14 05:36:19 +08:00
|
|
|
|
|
|
|
define <2 x i32> @test_constantdatavector_v2i32() {
|
|
|
|
; CHECK-LABEL: name: test_constantdatavector_v2i32
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[C1:%[0-9]+]]:_(s32) = G_CONSTANT i32 1
|
|
|
|
; CHECK: [[C2:%[0-9]+]]:_(s32) = G_CONSTANT i32 2
|
2018-12-11 02:44:58 +08:00
|
|
|
; CHECK: [[VEC:%[0-9]+]]:_(<2 x s32>) = G_BUILD_VECTOR [[C1]](s32), [[C2]](s32)
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: $d0 = COPY [[VEC]](<2 x s32>)
|
GlobalISel: Translate ConstantDataVector
Reviewers: qcolombet, aditya_nandakumar, dsanders, t.p.northover, javed.absar, ab
Reviewed By: qcolombet, dsanders, ab
Subscribers: dberris, rovka, llvm-commits, kristof.beyls
Differential Revision: https://reviews.llvm.org/D30216
llvm-svn: 297670
2017-03-14 05:36:19 +08:00
|
|
|
ret <2 x i32> <i32 1, i32 2>
|
|
|
|
}
|
|
|
|
|
|
|
|
define i32 @test_constantdatavector_v3i32() {
|
|
|
|
; CHECK-LABEL: name: test_constantdatavector_v3i32
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[C1:%[0-9]+]]:_(s32) = G_CONSTANT i32 1
|
|
|
|
; CHECK: [[C2:%[0-9]+]]:_(s32) = G_CONSTANT i32 2
|
|
|
|
; CHECK: [[C3:%[0-9]+]]:_(s32) = G_CONSTANT i32 3
|
2018-12-11 02:44:58 +08:00
|
|
|
; CHECK: [[VEC:%[0-9]+]]:_(<3 x s32>) = G_BUILD_VECTOR [[C1]](s32), [[C2]](s32), [[C3]](s32)
|
GlobalISel: Translate ConstantDataVector
Reviewers: qcolombet, aditya_nandakumar, dsanders, t.p.northover, javed.absar, ab
Reviewed By: qcolombet, dsanders, ab
Subscribers: dberris, rovka, llvm-commits, kristof.beyls
Differential Revision: https://reviews.llvm.org/D30216
llvm-svn: 297670
2017-03-14 05:36:19 +08:00
|
|
|
; CHECK: G_EXTRACT_VECTOR_ELT [[VEC]](<3 x s32>)
|
|
|
|
%elt = extractelement <3 x i32> <i32 1, i32 2, i32 3>, i32 1
|
|
|
|
ret i32 %elt
|
|
|
|
}
|
|
|
|
|
|
|
|
define <4 x i32> @test_constantdatavector_v4i32() {
|
|
|
|
; CHECK-LABEL: name: test_constantdatavector_v4i32
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[C1:%[0-9]+]]:_(s32) = G_CONSTANT i32 1
|
|
|
|
; CHECK: [[C2:%[0-9]+]]:_(s32) = G_CONSTANT i32 2
|
|
|
|
; CHECK: [[C3:%[0-9]+]]:_(s32) = G_CONSTANT i32 3
|
|
|
|
; CHECK: [[C4:%[0-9]+]]:_(s32) = G_CONSTANT i32 4
|
2018-12-11 02:44:58 +08:00
|
|
|
; CHECK: [[VEC:%[0-9]+]]:_(<4 x s32>) = G_BUILD_VECTOR [[C1]](s32), [[C2]](s32), [[C3]](s32), [[C4]](s32)
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: $q0 = COPY [[VEC]](<4 x s32>)
|
GlobalISel: Translate ConstantDataVector
Reviewers: qcolombet, aditya_nandakumar, dsanders, t.p.northover, javed.absar, ab
Reviewed By: qcolombet, dsanders, ab
Subscribers: dberris, rovka, llvm-commits, kristof.beyls
Differential Revision: https://reviews.llvm.org/D30216
llvm-svn: 297670
2017-03-14 05:36:19 +08:00
|
|
|
ret <4 x i32> <i32 1, i32 2, i32 3, i32 4>
|
|
|
|
}
|
|
|
|
|
|
|
|
define <2 x double> @test_constantdatavector_v2f64() {
|
|
|
|
; CHECK-LABEL: name: test_constantdatavector_v2f64
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[FC1:%[0-9]+]]:_(s64) = G_FCONSTANT double 1.000000e+00
|
|
|
|
; CHECK: [[FC2:%[0-9]+]]:_(s64) = G_FCONSTANT double 2.000000e+00
|
2018-12-11 02:44:58 +08:00
|
|
|
; CHECK: [[VEC:%[0-9]+]]:_(<2 x s64>) = G_BUILD_VECTOR [[FC1]](s64), [[FC2]](s64)
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: $q0 = COPY [[VEC]](<2 x s64>)
|
GlobalISel: Translate ConstantDataVector
Reviewers: qcolombet, aditya_nandakumar, dsanders, t.p.northover, javed.absar, ab
Reviewed By: qcolombet, dsanders, ab
Subscribers: dberris, rovka, llvm-commits, kristof.beyls
Differential Revision: https://reviews.llvm.org/D30216
llvm-svn: 297670
2017-03-14 05:36:19 +08:00
|
|
|
ret <2 x double> <double 1.0, double 2.0>
|
|
|
|
}
|
2017-03-15 07:45:06 +08:00
|
|
|
|
|
|
|
define i32 @test_constantaggzerovector_v1s32(i32 %arg){
|
|
|
|
; CHECK-LABEL: name: test_constantaggzerovector_v1s32
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[ARG:%[0-9]+]]:_(s32) = COPY $w0
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[C0:%[0-9]+]]:_(s32) = G_CONSTANT i32 0
|
2017-03-15 07:45:06 +08:00
|
|
|
; CHECK-NOT: G_MERGE_VALUES
|
|
|
|
; CHECK: G_ADD [[ARG]], [[C0]]
|
|
|
|
%vec = insertelement <1 x i32> undef, i32 %arg, i32 0
|
|
|
|
%add = add <1 x i32> %vec, zeroinitializer
|
|
|
|
%res = extractelement <1 x i32> %add, i32 0
|
|
|
|
ret i32 %res
|
|
|
|
}
|
|
|
|
|
|
|
|
define i32 @test_constantdatavector_v1s32(i32 %arg){
|
|
|
|
; CHECK-LABEL: name: test_constantdatavector_v1s32
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[ARG:%[0-9]+]]:_(s32) = COPY $w0
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[C1:%[0-9]+]]:_(s32) = G_CONSTANT i32 1
|
2017-03-15 07:45:06 +08:00
|
|
|
; CHECK-NOT: G_MERGE_VALUES
|
|
|
|
; CHECK: G_ADD [[ARG]], [[C1]]
|
|
|
|
%vec = insertelement <1 x i32> undef, i32 %arg, i32 0
|
|
|
|
%add = add <1 x i32> %vec, <i32 1>
|
|
|
|
%res = extractelement <1 x i32> %add, i32 0
|
|
|
|
ret i32 %res
|
|
|
|
}
|
2017-03-20 22:40:18 +08:00
|
|
|
|
|
|
|
declare ghccc float @different_call_conv_target(float %x)
|
|
|
|
define float @test_different_call_conv_target(float %x) {
|
|
|
|
; CHECK-LABEL: name: test_different_call_conv
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[X:%[0-9]+]]:_(s32) = COPY $s0
|
|
|
|
; CHECK: $s8 = COPY [[X]]
|
|
|
|
; CHECK: BL @different_call_conv_target, csr_aarch64_aapcs, implicit-def $lr, implicit $sp, implicit $s8, implicit-def $s0
|
2017-03-20 22:40:18 +08:00
|
|
|
%res = call ghccc float @different_call_conv_target(float %x)
|
|
|
|
ret float %res
|
|
|
|
}
|
2017-03-21 16:44:13 +08:00
|
|
|
|
|
|
|
define <2 x i32> @test_shufflevector_s32_v2s32(i32 %arg) {
|
|
|
|
; CHECK-LABEL: name: test_shufflevector_s32_v2s32
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[ARG:%[0-9]+]]:_(s32) = COPY $w0
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK-DAG: [[UNDEF:%[0-9]+]]:_(s32) = G_IMPLICIT_DEF
|
|
|
|
; CHECK-DAG: [[C0:%[0-9]+]]:_(s32) = G_CONSTANT i32 0
|
2018-12-11 02:44:58 +08:00
|
|
|
; CHECK-DAG: [[MASK:%[0-9]+]]:_(<2 x s32>) = G_BUILD_VECTOR [[C0]](s32), [[C0]](s32)
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[VEC:%[0-9]+]]:_(<2 x s32>) = G_SHUFFLE_VECTOR [[ARG]](s32), [[UNDEF]], [[MASK]](<2 x s32>)
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: $d0 = COPY [[VEC]](<2 x s32>)
|
2017-03-21 16:44:13 +08:00
|
|
|
%vec = insertelement <1 x i32> undef, i32 %arg, i32 0
|
|
|
|
%res = shufflevector <1 x i32> %vec, <1 x i32> undef, <2 x i32> zeroinitializer
|
|
|
|
ret <2 x i32> %res
|
|
|
|
}
|
|
|
|
|
|
|
|
define i32 @test_shufflevector_v2s32_s32(<2 x i32> %arg) {
|
|
|
|
; CHECK-LABEL: name: test_shufflevector_v2s32_s32
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[ARG:%[0-9]+]]:_(<2 x s32>) = COPY $d0
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK-DAG: [[UNDEF:%[0-9]+]]:_(<2 x s32>) = G_IMPLICIT_DEF
|
|
|
|
; CHECK-DAG: [[C1:%[0-9]+]]:_(s32) = G_CONSTANT i32 1
|
|
|
|
; CHECK: [[RES:%[0-9]+]]:_(s32) = G_SHUFFLE_VECTOR [[ARG]](<2 x s32>), [[UNDEF]], [[C1]](s32)
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: $w0 = COPY [[RES]](s32)
|
2017-03-21 16:44:13 +08:00
|
|
|
%vec = shufflevector <2 x i32> %arg, <2 x i32> undef, <1 x i32> <i32 1>
|
|
|
|
%res = extractelement <1 x i32> %vec, i32 0
|
|
|
|
ret i32 %res
|
|
|
|
}
|
|
|
|
|
|
|
|
define <2 x i32> @test_shufflevector_v2s32_v2s32(<2 x i32> %arg) {
|
|
|
|
; CHECK-LABEL: name: test_shufflevector_v2s32_v2s32
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[ARG:%[0-9]+]]:_(<2 x s32>) = COPY $d0
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK-DAG: [[UNDEF:%[0-9]+]]:_(<2 x s32>) = G_IMPLICIT_DEF
|
|
|
|
; CHECK-DAG: [[C1:%[0-9]+]]:_(s32) = G_CONSTANT i32 1
|
|
|
|
; CHECK-DAG: [[C0:%[0-9]+]]:_(s32) = G_CONSTANT i32 0
|
2018-12-11 02:44:58 +08:00
|
|
|
; CHECK-DAG: [[MASK:%[0-9]+]]:_(<2 x s32>) = G_BUILD_VECTOR [[C1]](s32), [[C0]](s32)
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[VEC:%[0-9]+]]:_(<2 x s32>) = G_SHUFFLE_VECTOR [[ARG]](<2 x s32>), [[UNDEF]], [[MASK]](<2 x s32>)
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: $d0 = COPY [[VEC]](<2 x s32>)
|
2017-03-21 16:44:13 +08:00
|
|
|
%res = shufflevector <2 x i32> %arg, <2 x i32> undef, <2 x i32> <i32 1, i32 0>
|
|
|
|
ret <2 x i32> %res
|
|
|
|
}
|
|
|
|
|
|
|
|
define i32 @test_shufflevector_v2s32_v3s32(<2 x i32> %arg) {
|
|
|
|
; CHECK-LABEL: name: test_shufflevector_v2s32_v3s32
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[ARG:%[0-9]+]]:_(<2 x s32>) = COPY $d0
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK-DAG: [[UNDEF:%[0-9]+]]:_(<2 x s32>) = G_IMPLICIT_DEF
|
|
|
|
; CHECK-DAG: [[C1:%[0-9]+]]:_(s32) = G_CONSTANT i32 1
|
|
|
|
; CHECK-DAG: [[C0:%[0-9]+]]:_(s32) = G_CONSTANT i32 0
|
2018-12-11 02:44:58 +08:00
|
|
|
; CHECK-DAG: [[MASK:%[0-9]+]]:_(<3 x s32>) = G_BUILD_VECTOR [[C1]](s32), [[C0]](s32), [[C1]](s32)
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[VEC:%[0-9]+]]:_(<3 x s32>) = G_SHUFFLE_VECTOR [[ARG]](<2 x s32>), [[UNDEF]], [[MASK]](<3 x s32>)
|
2017-03-21 16:44:13 +08:00
|
|
|
; CHECK: G_EXTRACT_VECTOR_ELT [[VEC]](<3 x s32>)
|
|
|
|
%vec = shufflevector <2 x i32> %arg, <2 x i32> undef, <3 x i32> <i32 1, i32 0, i32 1>
|
|
|
|
%res = extractelement <3 x i32> %vec, i32 0
|
|
|
|
ret i32 %res
|
|
|
|
}
|
|
|
|
|
|
|
|
define <4 x i32> @test_shufflevector_v2s32_v4s32(<2 x i32> %arg1, <2 x i32> %arg2) {
|
|
|
|
; CHECK-LABEL: name: test_shufflevector_v2s32_v4s32
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[ARG1:%[0-9]+]]:_(<2 x s32>) = COPY $d0
|
|
|
|
; CHECK: [[ARG2:%[0-9]+]]:_(<2 x s32>) = COPY $d1
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[C0:%[0-9]+]]:_(s32) = G_CONSTANT i32 0
|
|
|
|
; CHECK: [[C1:%[0-9]+]]:_(s32) = G_CONSTANT i32 1
|
|
|
|
; CHECK: [[C2:%[0-9]+]]:_(s32) = G_CONSTANT i32 2
|
|
|
|
; CHECK: [[C3:%[0-9]+]]:_(s32) = G_CONSTANT i32 3
|
2018-12-11 02:44:58 +08:00
|
|
|
; CHECK: [[MASK:%[0-9]+]]:_(<4 x s32>) = G_BUILD_VECTOR [[C0]](s32), [[C1]](s32), [[C2]](s32), [[C3]](s32)
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[VEC:%[0-9]+]]:_(<4 x s32>) = G_SHUFFLE_VECTOR [[ARG1]](<2 x s32>), [[ARG2]], [[MASK]](<4 x s32>)
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: $q0 = COPY [[VEC]](<4 x s32>)
|
2017-03-21 16:44:13 +08:00
|
|
|
%res = shufflevector <2 x i32> %arg1, <2 x i32> %arg2, <4 x i32> <i32 0, i32 1, i32 2, i32 3>
|
|
|
|
ret <4 x i32> %res
|
|
|
|
}
|
|
|
|
|
|
|
|
define <2 x i32> @test_shufflevector_v4s32_v2s32(<4 x i32> %arg) {
|
|
|
|
; CHECK-LABEL: name: test_shufflevector_v4s32_v2s32
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[ARG:%[0-9]+]]:_(<4 x s32>) = COPY $q0
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK-DAG: [[UNDEF:%[0-9]+]]:_(<4 x s32>) = G_IMPLICIT_DEF
|
|
|
|
; CHECK-DAG: [[C1:%[0-9]+]]:_(s32) = G_CONSTANT i32 1
|
|
|
|
; CHECK-DAG: [[C3:%[0-9]+]]:_(s32) = G_CONSTANT i32 3
|
2018-12-11 02:44:58 +08:00
|
|
|
; CHECK-DAG: [[MASK:%[0-9]+]]:_(<2 x s32>) = G_BUILD_VECTOR [[C1]](s32), [[C3]](s32)
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[VEC:%[0-9]+]]:_(<2 x s32>) = G_SHUFFLE_VECTOR [[ARG]](<4 x s32>), [[UNDEF]], [[MASK]](<2 x s32>)
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: $d0 = COPY [[VEC]](<2 x s32>)
|
2017-03-21 16:44:13 +08:00
|
|
|
%res = shufflevector <4 x i32> %arg, <4 x i32> undef, <2 x i32> <i32 1, i32 3>
|
|
|
|
ret <2 x i32> %res
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
define <16 x i8> @test_shufflevector_v8s8_v16s8(<8 x i8> %arg1, <8 x i8> %arg2) {
|
|
|
|
; CHECK-LABEL: name: test_shufflevector_v8s8_v16s8
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[ARG1:%[0-9]+]]:_(<8 x s8>) = COPY $d0
|
|
|
|
; CHECK: [[ARG2:%[0-9]+]]:_(<8 x s8>) = COPY $d1
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[C0:%[0-9]+]]:_(s32) = G_CONSTANT i32 0
|
|
|
|
; CHECK: [[C8:%[0-9]+]]:_(s32) = G_CONSTANT i32 8
|
|
|
|
; CHECK: [[C1:%[0-9]+]]:_(s32) = G_CONSTANT i32 1
|
|
|
|
; CHECK: [[C9:%[0-9]+]]:_(s32) = G_CONSTANT i32 9
|
|
|
|
; CHECK: [[C2:%[0-9]+]]:_(s32) = G_CONSTANT i32 2
|
|
|
|
; CHECK: [[C10:%[0-9]+]]:_(s32) = G_CONSTANT i32 10
|
|
|
|
; CHECK: [[C3:%[0-9]+]]:_(s32) = G_CONSTANT i32 3
|
|
|
|
; CHECK: [[C11:%[0-9]+]]:_(s32) = G_CONSTANT i32 11
|
|
|
|
; CHECK: [[C4:%[0-9]+]]:_(s32) = G_CONSTANT i32 4
|
|
|
|
; CHECK: [[C12:%[0-9]+]]:_(s32) = G_CONSTANT i32 12
|
|
|
|
; CHECK: [[C5:%[0-9]+]]:_(s32) = G_CONSTANT i32 5
|
|
|
|
; CHECK: [[C13:%[0-9]+]]:_(s32) = G_CONSTANT i32 13
|
|
|
|
; CHECK: [[C6:%[0-9]+]]:_(s32) = G_CONSTANT i32 6
|
|
|
|
; CHECK: [[C14:%[0-9]+]]:_(s32) = G_CONSTANT i32 14
|
|
|
|
; CHECK: [[C7:%[0-9]+]]:_(s32) = G_CONSTANT i32 7
|
|
|
|
; CHECK: [[C15:%[0-9]+]]:_(s32) = G_CONSTANT i32 15
|
2018-12-11 02:44:58 +08:00
|
|
|
; CHECK: [[MASK:%[0-9]+]]:_(<16 x s32>) = G_BUILD_VECTOR [[C0]](s32), [[C8]](s32), [[C1]](s32), [[C9]](s32), [[C2]](s32), [[C10]](s32), [[C3]](s32), [[C11]](s32), [[C4]](s32), [[C12]](s32), [[C5]](s32), [[C13]](s32), [[C6]](s32), [[C14]](s32), [[C7]](s32), [[C15]](s32)
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[VEC:%[0-9]+]]:_(<16 x s8>) = G_SHUFFLE_VECTOR [[ARG1]](<8 x s8>), [[ARG2]], [[MASK]](<16 x s32>)
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: $q0 = COPY [[VEC]](<16 x s8>)
|
2017-03-21 16:44:13 +08:00
|
|
|
%res = shufflevector <8 x i8> %arg1, <8 x i8> %arg2, <16 x i32> <i32 0, i32 8, i32 1, i32 9, i32 2, i32 10, i32 3, i32 11, i32 4, i32 12, i32 5, i32 13, i32 6, i32 14, i32 7, i32 15>
|
|
|
|
ret <16 x i8> %res
|
|
|
|
}
|
2017-05-05 05:43:12 +08:00
|
|
|
|
|
|
|
; CHECK-LABEL: test_constant_vector
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[UNDEF:%[0-9]+]]:_(s16) = G_IMPLICIT_DEF
|
|
|
|
; CHECK: [[F:%[0-9]+]]:_(s16) = G_FCONSTANT half 0xH3C00
|
2018-12-11 02:44:58 +08:00
|
|
|
; CHECK: [[M:%[0-9]+]]:_(<4 x s16>) = G_BUILD_VECTOR [[UNDEF]](s16), [[UNDEF]](s16), [[UNDEF]](s16), [[F]](s16)
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: $d0 = COPY [[M]](<4 x s16>)
|
2017-05-05 05:43:12 +08:00
|
|
|
define <4 x half> @test_constant_vector() {
|
|
|
|
ret <4 x half> <half undef, half undef, half undef, half 0xH3C00>
|
|
|
|
}
|
2017-06-06 06:17:17 +08:00
|
|
|
|
|
|
|
define i32 @test_target_mem_intrinsic(i32* %addr) {
|
|
|
|
; CHECK-LABEL: name: test_target_mem_intrinsic
|
2018-02-01 06:04:26 +08:00
|
|
|
; CHECK: [[ADDR:%[0-9]+]]:_(p0) = COPY $x0
|
2017-10-25 02:04:54 +08:00
|
|
|
; CHECK: [[VAL:%[0-9]+]]:_(s64) = G_INTRINSIC_W_SIDE_EFFECTS intrinsic(@llvm.aarch64.ldxr), [[ADDR]](p0) :: (volatile load 4 from %ir.addr)
|
2017-06-06 06:17:17 +08:00
|
|
|
; CHECK: G_TRUNC [[VAL]](s64)
|
|
|
|
%val = call i64 @llvm.aarch64.ldxr.p0i32(i32* %addr)
|
|
|
|
%trunc = trunc i64 %val to i32
|
|
|
|
ret i32 %trunc
|
|
|
|
}
|
|
|
|
|
|
|
|
declare i64 @llvm.aarch64.ldxr.p0i32(i32*) nounwind
|
2017-12-01 04:06:02 +08:00
|
|
|
|
|
|
|
%zerosize_type = type {}
|
|
|
|
|
|
|
|
define %zerosize_type @test_empty_load_store(%zerosize_type *%ptr, %zerosize_type %in) noinline optnone {
|
|
|
|
; CHECK-LABEL: name: test_empty_load_store
|
|
|
|
; CHECK-NOT: G_STORE
|
|
|
|
; CHECK-NOT: G_LOAD
|
|
|
|
; CHECK: RET_ReallyLR
|
|
|
|
entry:
|
|
|
|
store %zerosize_type undef, %zerosize_type* undef, align 4
|
|
|
|
%val = load %zerosize_type, %zerosize_type* %ptr, align 4
|
|
|
|
ret %zerosize_type %in
|
|
|
|
}
|
2018-05-16 18:32:02 +08:00
|
|
|
|
|
|
|
|
|
|
|
define i64 @test_phi_loop(i32 %n) {
|
|
|
|
; CHECK-LABEL: name: test_phi_loop
|
|
|
|
; CHECK: [[ARG1:%[0-9]+]]:_(s32) = COPY $w0
|
|
|
|
; CHECK: [[CST1:%[0-9]+]]:_(s32) = G_CONSTANT i32 1
|
|
|
|
; CHECK: [[CST2:%[0-9]+]]:_(s32) = G_CONSTANT i32 0
|
|
|
|
; CHECK: [[CST3:%[0-9]+]]:_(s64) = G_CONSTANT i64 0
|
|
|
|
; CHECK: [[CST4:%[0-9]+]]:_(s64) = G_CONSTANT i64 1
|
|
|
|
|
|
|
|
; CHECK: [[PN1:%[0-9]+]]:_(s32) = G_PHI [[ARG1]](s32), %bb.1, [[SUB:%[0-9]+]](s32), %bb.2
|
|
|
|
; CHECK: [[PN2:%[0-9]+]]:_(s64) = G_PHI [[CST3]](s64), %bb.1, [[PN3:%[0-9]+]](s64), %bb.2
|
|
|
|
; CHECK: [[PN3]]:_(s64) = G_PHI [[CST4]](s64), %bb.1, [[ADD:%[0-9]+]](s64), %bb.2
|
|
|
|
; CHECK: [[ADD]]:_(s64) = G_ADD [[PN2]], [[PN3]]
|
|
|
|
; CHECK: [[SUB]]:_(s32) = G_SUB [[PN1]], [[CST1]]
|
|
|
|
; CHECK: [[CMP:%[0-9]+]]:_(s1) = G_ICMP intpred(sle), [[PN1]](s32), [[CST2]]
|
|
|
|
; CHECK: G_BRCOND [[CMP]](s1), %bb.3
|
|
|
|
; CHECK: G_BR %bb.2
|
|
|
|
|
|
|
|
; CHECK: $x0 = COPY [[PN2]](s64)
|
|
|
|
; CHECK: RET_ReallyLR implicit $x0
|
|
|
|
entry:
|
|
|
|
br label %loop
|
|
|
|
|
|
|
|
loop:
|
|
|
|
%counter = phi i32 [ %n, %entry ], [ %counter.dec, %loop ]
|
|
|
|
%elem = phi { i64, i64 } [ { i64 0, i64 1 }, %entry ], [ %updated, %loop ]
|
|
|
|
%prev = extractvalue { i64, i64 } %elem, 0
|
|
|
|
%curr = extractvalue { i64, i64 } %elem, 1
|
|
|
|
%next = add i64 %prev, %curr
|
|
|
|
%shifted = insertvalue { i64, i64 } %elem, i64 %curr, 0
|
|
|
|
%updated = insertvalue { i64, i64 } %shifted, i64 %next, 1
|
|
|
|
%counter.dec = sub i32 %counter, 1
|
|
|
|
%cond = icmp sle i32 %counter, 0
|
|
|
|
br i1 %cond, label %exit, label %loop
|
|
|
|
|
|
|
|
exit:
|
|
|
|
%res = extractvalue { i64, i64 } %elem, 0
|
|
|
|
ret i64 %res
|
|
|
|
}
|
|
|
|
|
|
|
|
define void @test_phi_diamond({ i8, i16, i32 }* %a.ptr, { i8, i16, i32 }* %b.ptr, i1 %selector, { i8, i16, i32 }* %dst) {
|
|
|
|
; CHECK-LABEL: name: test_phi_diamond
|
|
|
|
; CHECK: [[ARG1:%[0-9]+]]:_(p0) = COPY $x0
|
|
|
|
; CHECK: [[ARG2:%[0-9]+]]:_(p0) = COPY $x1
|
|
|
|
; CHECK: [[ARG3:%[0-9]+]]:_(s32) = COPY $w2
|
|
|
|
; CHECK: [[TRUNC:%[0-9]+]]:_(s1) = G_TRUNC [[ARG3]](s32)
|
|
|
|
; CHECK: [[ARG4:%[0-9]+]]:_(p0) = COPY $x3
|
|
|
|
; CHECK: G_BRCOND [[TRUNC]](s1), %bb.2
|
|
|
|
; CHECK: G_BR %bb.3
|
|
|
|
|
|
|
|
; CHECK: [[LD1:%[0-9]+]]:_(s8) = G_LOAD [[ARG1]](p0) :: (load 1 from %ir.a.ptr, align 4)
|
|
|
|
; CHECK: [[CST1:%[0-9]+]]:_(s64) = G_CONSTANT i64 2
|
|
|
|
; CHECK: [[GEP1:%[0-9]+]]:_(p0) = G_GEP [[ARG1]], [[CST1]](s64)
|
|
|
|
; CHECK: [[LD2:%[0-9]+]]:_(s16) = G_LOAD [[GEP1]](p0) :: (load 2 from %ir.a.ptr + 2)
|
|
|
|
; CHECK: [[CST2:%[0-9]+]]:_(s64) = G_CONSTANT i64 4
|
|
|
|
; CHECK: [[GEP2:%[0-9]+]]:_(p0) = G_GEP [[ARG1]], [[CST2]](s64)
|
|
|
|
; CHECK: [[LD3:%[0-9]+]]:_(s32) = G_LOAD [[GEP2]](p0) :: (load 4 from %ir.a.ptr + 4)
|
|
|
|
; CHECK: G_BR %bb.4
|
|
|
|
|
|
|
|
; CHECK: [[LD4:%[0-9]+]]:_(s8) = G_LOAD [[ARG2]](p0) :: (load 1 from %ir.b.ptr, align 4)
|
|
|
|
; CHECK: [[CST3:%[0-9]+]]:_(s64) = G_CONSTANT i64 2
|
|
|
|
; CHECK: [[GEP3:%[0-9]+]]:_(p0) = G_GEP [[ARG2]], [[CST3]](s64)
|
|
|
|
; CHECK: [[LD5:%[0-9]+]]:_(s16) = G_LOAD [[GEP3]](p0) :: (load 2 from %ir.b.ptr + 2)
|
|
|
|
; CHECK: [[CST4:%[0-9]+]]:_(s64) = G_CONSTANT i64 4
|
|
|
|
; CHECK: [[GEP4:%[0-9]+]]:_(p0) = G_GEP [[ARG2]], [[CST4]](s64)
|
|
|
|
; CHECK: [[LD6:%[0-9]+]]:_(s32) = G_LOAD [[GEP4]](p0) :: (load 4 from %ir.b.ptr + 4)
|
|
|
|
|
|
|
|
; CHECK: [[PN1:%[0-9]+]]:_(s8) = G_PHI [[LD1]](s8), %bb.2, [[LD4]](s8), %bb.3
|
|
|
|
; CHECK: [[PN2:%[0-9]+]]:_(s16) = G_PHI [[LD2]](s16), %bb.2, [[LD5]](s16), %bb.3
|
|
|
|
; CHECK: [[PN3:%[0-9]+]]:_(s32) = G_PHI [[LD3]](s32), %bb.2, [[LD6]](s32), %bb.3
|
|
|
|
; CHECK: G_STORE [[PN1]](s8), [[ARG4]](p0) :: (store 1 into %ir.dst, align 4)
|
|
|
|
; CHECK: [[CST5:%[0-9]+]]:_(s64) = G_CONSTANT i64 2
|
|
|
|
; CHECK: [[GEP5:%[0-9]+]]:_(p0) = G_GEP [[ARG4]], [[CST5]](s64)
|
|
|
|
; CHECK: G_STORE [[PN2]](s16), [[GEP5]](p0) :: (store 2 into %ir.dst + 2)
|
|
|
|
; CHECK: [[CST6:%[0-9]+]]:_(s64) = G_CONSTANT i64 4
|
|
|
|
; CHECK: [[GEP6:%[0-9]+]]:_(p0) = G_GEP [[ARG4]], [[CST6]](s64)
|
|
|
|
; CHECK: G_STORE [[PN3]](s32), [[GEP6]](p0) :: (store 4 into %ir.dst + 4)
|
|
|
|
; CHECK: RET_ReallyLR
|
|
|
|
|
|
|
|
entry:
|
|
|
|
br i1 %selector, label %store.a, label %store.b
|
|
|
|
|
|
|
|
store.a:
|
|
|
|
%a = load { i8, i16, i32 }, { i8, i16, i32 }* %a.ptr
|
|
|
|
br label %join
|
|
|
|
|
|
|
|
store.b:
|
|
|
|
%b = load { i8, i16, i32 }, { i8, i16, i32 }* %b.ptr
|
|
|
|
br label %join
|
|
|
|
|
|
|
|
join:
|
|
|
|
%v = phi { i8, i16, i32 } [ %a, %store.a ], [ %b, %store.b ]
|
|
|
|
store { i8, i16, i32 } %v, { i8, i16, i32 }* %dst
|
|
|
|
ret void
|
|
|
|
}
|
|
|
|
|
|
|
|
%agg.inner.inner = type {i64, i64}
|
|
|
|
%agg.inner = type {i16, i8, %agg.inner.inner }
|
|
|
|
%agg.nested = type {i32, i32, %agg.inner, i32}
|
|
|
|
|
|
|
|
define void @test_nested_aggregate_const(%agg.nested *%ptr) {
|
|
|
|
; CHECK-LABEL: name: test_nested_aggregate_const
|
|
|
|
; CHECK: [[BASE:%[0-9]+]]:_(p0) = COPY $x0
|
|
|
|
; CHECK: [[CST1:%[0-9]+]]:_(s32) = G_CONSTANT i32 1
|
|
|
|
; CHECK: [[CST2:%[0-9]+]]:_(s16) = G_CONSTANT i16 2
|
|
|
|
; CHECK: [[CST3:%[0-9]+]]:_(s8) = G_CONSTANT i8 3
|
|
|
|
; CHECK: [[CST4:%[0-9]+]]:_(s64) = G_CONSTANT i64 5
|
|
|
|
; CHECK: [[CST5:%[0-9]+]]:_(s64) = G_CONSTANT i64 8
|
|
|
|
; CHECK: [[CST6:%[0-9]+]]:_(s32) = G_CONSTANT i32 13
|
|
|
|
; CHECK: G_STORE [[CST1]](s32), [[BASE]](p0) :: (store 4 into %ir.ptr, align 8)
|
|
|
|
; CHECK: [[CST7:%[0-9]+]]:_(s64) = G_CONSTANT i64 4
|
|
|
|
; CHECK: [[GEP1:%[0-9]+]]:_(p0) = G_GEP [[BASE]], [[CST7]](s64)
|
|
|
|
; CHECK: G_STORE [[CST1]](s32), [[GEP1]](p0) :: (store 4 into %ir.ptr + 4)
|
|
|
|
; CHECK: [[CST8:%[0-9]+]]:_(s64) = G_CONSTANT i64 8
|
|
|
|
; CHECK: [[GEP2:%[0-9]+]]:_(p0) = G_GEP [[BASE]], [[CST8]](s64)
|
|
|
|
; CHECK: G_STORE [[CST2]](s16), [[GEP2]](p0) :: (store 2 into %ir.ptr + 8, align 8)
|
|
|
|
; CHECK: [[CST9:%[0-9]+]]:_(s64) = G_CONSTANT i64 10
|
|
|
|
; CHECK: [[GEP3:%[0-9]+]]:_(p0) = G_GEP [[BASE]], [[CST9]](s64)
|
|
|
|
; CHECK: G_STORE [[CST3]](s8), [[GEP3]](p0) :: (store 1 into %ir.ptr + 10, align 2)
|
|
|
|
; CHECK: [[CST10:%[0-9]+]]:_(s64) = G_CONSTANT i64 16
|
|
|
|
; CHECK: [[GEP4:%[0-9]+]]:_(p0) = G_GEP [[BASE]], [[CST10]](s64)
|
|
|
|
; CHECK: G_STORE [[CST4]](s64), [[GEP4]](p0) :: (store 8 into %ir.ptr + 16)
|
|
|
|
; CHECK: [[CST11:%[0-9]+]]:_(s64) = G_CONSTANT i64 24
|
|
|
|
; CHECK: [[GEP5:%[0-9]+]]:_(p0) = G_GEP [[BASE]], [[CST11]](s64)
|
|
|
|
; CHECK: G_STORE [[CST5]](s64), [[GEP5]](p0) :: (store 8 into %ir.ptr + 24)
|
|
|
|
; CHECK: [[CST12:%[0-9]+]]:_(s64) = G_CONSTANT i64 32
|
|
|
|
; CHECK: [[GEP6:%[0-9]+]]:_(p0) = G_GEP [[BASE]], [[CST12]](s64)
|
|
|
|
; CHECK: G_STORE [[CST6]](s32), [[GEP6]](p0) :: (store 4 into %ir.ptr + 32, align 8)
|
|
|
|
store %agg.nested { i32 1, i32 1, %agg.inner { i16 2, i8 3, %agg.inner.inner {i64 5, i64 8} }, i32 13}, %agg.nested *%ptr
|
|
|
|
ret void
|
|
|
|
}
|
2018-06-01 21:20:32 +08:00
|
|
|
|
|
|
|
define i1 @return_i1_zext() {
|
|
|
|
; AAPCS ABI says that booleans can only be 1 or 0, so we need to zero-extend.
|
|
|
|
; CHECK-LABEL: name: return_i1_zext
|
|
|
|
; CHECK: [[CST:%[0-9]+]]:_(s1) = G_CONSTANT i1 true
|
|
|
|
; CHECK: [[ZEXT:%[0-9]+]]:_(s8) = G_ZEXT [[CST]](s1)
|
|
|
|
; CHECK: [[ANYEXT:%[0-9]+]]:_(s32) = G_ANYEXT [[ZEXT]](s8)
|
|
|
|
; CHECK: $w0 = COPY [[ANYEXT]](s32)
|
|
|
|
; CHECK: RET_ReallyLR implicit $w0
|
|
|
|
ret i1 true
|
|
|
|
}
|
[globalisel][irtranslator] Add support for atomicrmw and (strong) cmpxchg
Summary:
This patch adds support for the atomicrmw instructions and the strong
cmpxchg instruction to the IRTranslator.
I've left out weak cmpxchg because LangRef.rst isn't entirely clear on what
difference it makes to the backend. As far as I can tell from the code, it
only matters to AtomicExpandPass which is run at the LLVM-IR level.
Reviewers: ab, t.p.northover, qcolombet, rovka, aditya_nandakumar, volkan, javed.absar
Reviewed By: qcolombet
Subscribers: kristof.beyls, javed.absar, igorb, llvm-commits
Differential Revision: https://reviews.llvm.org/D40092
llvm-svn: 336589
2018-07-10 03:33:40 +08:00
|
|
|
|
|
|
|
; Try one cmpxchg
|
|
|
|
define i32 @test_atomic_cmpxchg_1(i32* %addr) {
|
|
|
|
; CHECK-LABEL: name: test_atomic_cmpxchg_1
|
|
|
|
; CHECK: bb.1.entry:
|
|
|
|
; CHECK-NEXT: successors: %bb.{{[^)]+}}
|
|
|
|
; CHECK-NEXT: liveins: $x0
|
|
|
|
; CHECK: [[ADDR:%[0-9]+]]:_(p0) = COPY $x0
|
|
|
|
; CHECK-NEXT: [[OLDVAL:%[0-9]+]]:_(s32) = G_CONSTANT i32 0
|
|
|
|
; CHECK-NEXT: [[NEWVAL:%[0-9]+]]:_(s32) = G_CONSTANT i32 1
|
|
|
|
; CHECK: bb.2.repeat:
|
|
|
|
; CHECK-NEXT: successors: %bb.3({{[^)]+}}), %bb.2({{[^)]+}})
|
|
|
|
; CHECK: [[OLDVALRES:%[0-9]+]]:_(s32), [[SUCCESS:%[0-9]+]]:_(s1) = G_ATOMIC_CMPXCHG_WITH_SUCCESS [[ADDR]](p0), [[OLDVAL]], [[NEWVAL]] :: (load store monotonic monotonic 4 on %ir.addr)
|
|
|
|
; CHECK-NEXT: G_BRCOND [[SUCCESS]](s1), %bb.3
|
|
|
|
; CHECK-NEXT: G_BR %bb.2
|
|
|
|
; CHECK: bb.3.done:
|
|
|
|
entry:
|
|
|
|
br label %repeat
|
|
|
|
repeat:
|
|
|
|
%val_success = cmpxchg i32* %addr, i32 0, i32 1 monotonic monotonic
|
|
|
|
%value_loaded = extractvalue { i32, i1 } %val_success, 0
|
|
|
|
%success = extractvalue { i32, i1 } %val_success, 1
|
|
|
|
br i1 %success, label %done, label %repeat
|
|
|
|
done:
|
|
|
|
ret i32 %value_loaded
|
|
|
|
}
|
|
|
|
|
|
|
|
; Try one cmpxchg with a small type and high atomic ordering.
|
|
|
|
define i16 @test_atomic_cmpxchg_2(i16* %addr) {
|
|
|
|
; CHECK-LABEL: name: test_atomic_cmpxchg_2
|
|
|
|
; CHECK: bb.1.entry:
|
|
|
|
; CHECK-NEXT: successors: %bb.2({{[^)]+}})
|
|
|
|
; CHECK-NEXT: liveins: $x0
|
|
|
|
; CHECK: [[ADDR:%[0-9]+]]:_(p0) = COPY $x0
|
|
|
|
; CHECK-NEXT: [[OLDVAL:%[0-9]+]]:_(s16) = G_CONSTANT i16 0
|
|
|
|
; CHECK-NEXT: [[NEWVAL:%[0-9]+]]:_(s16) = G_CONSTANT i16 1
|
|
|
|
; CHECK: bb.2.repeat:
|
|
|
|
; CHECK-NEXT: successors: %bb.3({{[^)]+}}), %bb.2({{[^)]+}})
|
|
|
|
; CHECK: [[OLDVALRES:%[0-9]+]]:_(s16), [[SUCCESS:%[0-9]+]]:_(s1) = G_ATOMIC_CMPXCHG_WITH_SUCCESS [[ADDR]](p0), [[OLDVAL]], [[NEWVAL]] :: (load store seq_cst seq_cst 2 on %ir.addr)
|
|
|
|
; CHECK-NEXT: G_BRCOND [[SUCCESS]](s1), %bb.3
|
|
|
|
; CHECK-NEXT: G_BR %bb.2
|
|
|
|
; CHECK: bb.3.done:
|
|
|
|
entry:
|
|
|
|
br label %repeat
|
|
|
|
repeat:
|
|
|
|
%val_success = cmpxchg i16* %addr, i16 0, i16 1 seq_cst seq_cst
|
|
|
|
%value_loaded = extractvalue { i16, i1 } %val_success, 0
|
|
|
|
%success = extractvalue { i16, i1 } %val_success, 1
|
|
|
|
br i1 %success, label %done, label %repeat
|
|
|
|
done:
|
|
|
|
ret i16 %value_loaded
|
|
|
|
}
|
|
|
|
|
|
|
|
; Try one cmpxchg where the success order and failure order differ.
|
|
|
|
define i64 @test_atomic_cmpxchg_3(i64* %addr) {
|
|
|
|
; CHECK-LABEL: name: test_atomic_cmpxchg_3
|
|
|
|
; CHECK: bb.1.entry:
|
|
|
|
; CHECK-NEXT: successors: %bb.2({{[^)]+}})
|
|
|
|
; CHECK-NEXT: liveins: $x0
|
|
|
|
; CHECK: [[ADDR:%[0-9]+]]:_(p0) = COPY $x0
|
|
|
|
; CHECK-NEXT: [[OLDVAL:%[0-9]+]]:_(s64) = G_CONSTANT i64 0
|
|
|
|
; CHECK-NEXT: [[NEWVAL:%[0-9]+]]:_(s64) = G_CONSTANT i64 1
|
|
|
|
; CHECK: bb.2.repeat:
|
|
|
|
; CHECK-NEXT: successors: %bb.3({{[^)]+}}), %bb.2({{[^)]+}})
|
|
|
|
; CHECK: [[OLDVALRES:%[0-9]+]]:_(s64), [[SUCCESS:%[0-9]+]]:_(s1) = G_ATOMIC_CMPXCHG_WITH_SUCCESS [[ADDR]](p0), [[OLDVAL]], [[NEWVAL]] :: (load store seq_cst acquire 8 on %ir.addr)
|
|
|
|
; CHECK-NEXT: G_BRCOND [[SUCCESS]](s1), %bb.3
|
|
|
|
; CHECK-NEXT: G_BR %bb.2
|
|
|
|
; CHECK: bb.3.done:
|
|
|
|
entry:
|
|
|
|
br label %repeat
|
|
|
|
repeat:
|
|
|
|
%val_success = cmpxchg i64* %addr, i64 0, i64 1 seq_cst acquire
|
|
|
|
%value_loaded = extractvalue { i64, i1 } %val_success, 0
|
|
|
|
%success = extractvalue { i64, i1 } %val_success, 1
|
|
|
|
br i1 %success, label %done, label %repeat
|
|
|
|
done:
|
|
|
|
ret i64 %value_loaded
|
|
|
|
}
|
|
|
|
|
|
|
|
; Try a monotonic atomicrmw xchg
|
|
|
|
; AArch64 will expand some atomicrmw's at the LLVM-IR level so we use a wide type to avoid this.
|
|
|
|
define i32 @test_atomicrmw_xchg(i256* %addr) {
|
|
|
|
; CHECK-LABEL: name: test_atomicrmw_xchg
|
|
|
|
; CHECK: bb.1 (%ir-block.{{[0-9]+}}):
|
|
|
|
; CHECK-NEXT: liveins: $x0
|
|
|
|
; CHECK: [[ADDR:%[0-9]+]]:_(p0) = COPY $x0
|
|
|
|
; CHECK-NEXT: [[VAL:%[0-9]+]]:_(s256) = G_CONSTANT i256 1
|
|
|
|
; CHECK-NEXT: [[OLDVALRES:%[0-9]+]]:_(s256) = G_ATOMICRMW_XCHG [[ADDR]](p0), [[VAL]] :: (load store monotonic 32 on %ir.addr)
|
|
|
|
; CHECK-NEXT: [[RES:%[0-9]+]]:_(s32) = G_TRUNC [[OLDVALRES]]
|
|
|
|
%oldval = atomicrmw xchg i256* %addr, i256 1 monotonic
|
|
|
|
; FIXME: We currently can't lower 'ret i256' and it's not the purpose of this
|
|
|
|
; test so work around it by truncating to i32 for now.
|
|
|
|
%oldval.trunc = trunc i256 %oldval to i32
|
|
|
|
ret i32 %oldval.trunc
|
|
|
|
}
|
|
|
|
|
|
|
|
; Try an acquire atomicrmw add
|
|
|
|
; AArch64 will expand some atomicrmw's at the LLVM-IR level so we use a wide type to avoid this.
|
|
|
|
define i32 @test_atomicrmw_add(i256* %addr) {
|
|
|
|
; CHECK-LABEL: name: test_atomicrmw_add
|
|
|
|
; CHECK: bb.1 (%ir-block.{{[0-9]+}}):
|
|
|
|
; CHECK-NEXT: liveins: $x0
|
|
|
|
; CHECK: [[ADDR:%[0-9]+]]:_(p0) = COPY $x0
|
|
|
|
; CHECK-NEXT: [[VAL:%[0-9]+]]:_(s256) = G_CONSTANT i256 1
|
|
|
|
; CHECK-NEXT: [[OLDVALRES:%[0-9]+]]:_(s256) = G_ATOMICRMW_ADD [[ADDR]](p0), [[VAL]] :: (load store acquire 32 on %ir.addr)
|
|
|
|
; CHECK-NEXT: [[RES:%[0-9]+]]:_(s32) = G_TRUNC [[OLDVALRES]]
|
|
|
|
%oldval = atomicrmw add i256* %addr, i256 1 acquire
|
|
|
|
; FIXME: We currently can't lower 'ret i256' and it's not the purpose of this
|
|
|
|
; test so work around it by truncating to i32 for now.
|
|
|
|
%oldval.trunc = trunc i256 %oldval to i32
|
|
|
|
ret i32 %oldval.trunc
|
|
|
|
}
|
|
|
|
|
|
|
|
; Try a release atomicrmw sub
|
|
|
|
; AArch64 will expand some atomicrmw's at the LLVM-IR level so we use a wide type to avoid this.
|
|
|
|
define i32 @test_atomicrmw_sub(i256* %addr) {
|
|
|
|
; CHECK-LABEL: name: test_atomicrmw_sub
|
|
|
|
; CHECK: bb.1 (%ir-block.{{[0-9]+}}):
|
|
|
|
; CHECK-NEXT: liveins: $x0
|
|
|
|
; CHECK: [[ADDR:%[0-9]+]]:_(p0) = COPY $x0
|
|
|
|
; CHECK-NEXT: [[VAL:%[0-9]+]]:_(s256) = G_CONSTANT i256 1
|
|
|
|
; CHECK-NEXT: [[OLDVALRES:%[0-9]+]]:_(s256) = G_ATOMICRMW_SUB [[ADDR]](p0), [[VAL]] :: (load store release 32 on %ir.addr)
|
|
|
|
; CHECK-NEXT: [[RES:%[0-9]+]]:_(s32) = G_TRUNC [[OLDVALRES]]
|
|
|
|
%oldval = atomicrmw sub i256* %addr, i256 1 release
|
|
|
|
; FIXME: We currently can't lower 'ret i256' and it's not the purpose of this
|
|
|
|
; test so work around it by truncating to i32 for now.
|
|
|
|
%oldval.trunc = trunc i256 %oldval to i32
|
|
|
|
ret i32 %oldval.trunc
|
|
|
|
}
|
|
|
|
|
|
|
|
; Try an acq_rel atomicrmw and
|
|
|
|
; AArch64 will expand some atomicrmw's at the LLVM-IR level so we use a wide type to avoid this.
|
|
|
|
define i32 @test_atomicrmw_and(i256* %addr) {
|
|
|
|
; CHECK-LABEL: name: test_atomicrmw_and
|
|
|
|
; CHECK: bb.1 (%ir-block.{{[0-9]+}}):
|
|
|
|
; CHECK-NEXT: liveins: $x0
|
|
|
|
; CHECK: [[ADDR:%[0-9]+]]:_(p0) = COPY $x0
|
|
|
|
; CHECK-NEXT: [[VAL:%[0-9]+]]:_(s256) = G_CONSTANT i256 1
|
|
|
|
; CHECK-NEXT: [[OLDVALRES:%[0-9]+]]:_(s256) = G_ATOMICRMW_AND [[ADDR]](p0), [[VAL]] :: (load store acq_rel 32 on %ir.addr)
|
|
|
|
; CHECK-NEXT: [[RES:%[0-9]+]]:_(s32) = G_TRUNC [[OLDVALRES]]
|
|
|
|
%oldval = atomicrmw and i256* %addr, i256 1 acq_rel
|
|
|
|
; FIXME: We currently can't lower 'ret i256' and it's not the purpose of this
|
|
|
|
; test so work around it by truncating to i32 for now.
|
|
|
|
%oldval.trunc = trunc i256 %oldval to i32
|
|
|
|
ret i32 %oldval.trunc
|
|
|
|
}
|
|
|
|
|
|
|
|
; Try an seq_cst atomicrmw nand
|
|
|
|
; AArch64 will expand some atomicrmw's at the LLVM-IR level so we use a wide type to avoid this.
|
|
|
|
define i32 @test_atomicrmw_nand(i256* %addr) {
|
|
|
|
; CHECK-LABEL: name: test_atomicrmw_nand
|
|
|
|
; CHECK: bb.1 (%ir-block.{{[0-9]+}}):
|
|
|
|
; CHECK-NEXT: liveins: $x0
|
|
|
|
; CHECK: [[ADDR:%[0-9]+]]:_(p0) = COPY $x0
|
|
|
|
; CHECK-NEXT: [[VAL:%[0-9]+]]:_(s256) = G_CONSTANT i256 1
|
|
|
|
; CHECK-NEXT: [[OLDVALRES:%[0-9]+]]:_(s256) = G_ATOMICRMW_NAND [[ADDR]](p0), [[VAL]] :: (load store seq_cst 32 on %ir.addr)
|
|
|
|
; CHECK-NEXT: [[RES:%[0-9]+]]:_(s32) = G_TRUNC [[OLDVALRES]]
|
|
|
|
%oldval = atomicrmw nand i256* %addr, i256 1 seq_cst
|
|
|
|
; FIXME: We currently can't lower 'ret i256' and it's not the purpose of this
|
|
|
|
; test so work around it by truncating to i32 for now.
|
|
|
|
%oldval.trunc = trunc i256 %oldval to i32
|
|
|
|
ret i32 %oldval.trunc
|
|
|
|
}
|
|
|
|
|
|
|
|
; Try an seq_cst atomicrmw or
|
|
|
|
; AArch64 will expand some atomicrmw's at the LLVM-IR level so we use a wide type to avoid this.
|
|
|
|
define i32 @test_atomicrmw_or(i256* %addr) {
|
|
|
|
; CHECK-LABEL: name: test_atomicrmw_or
|
|
|
|
; CHECK: bb.1 (%ir-block.{{[0-9]+}}):
|
|
|
|
; CHECK-NEXT: liveins: $x0
|
|
|
|
; CHECK: [[ADDR:%[0-9]+]]:_(p0) = COPY $x0
|
|
|
|
; CHECK-NEXT: [[VAL:%[0-9]+]]:_(s256) = G_CONSTANT i256 1
|
|
|
|
; CHECK-NEXT: [[OLDVALRES:%[0-9]+]]:_(s256) = G_ATOMICRMW_OR [[ADDR]](p0), [[VAL]] :: (load store seq_cst 32 on %ir.addr)
|
|
|
|
; CHECK-NEXT: [[RES:%[0-9]+]]:_(s32) = G_TRUNC [[OLDVALRES]]
|
|
|
|
%oldval = atomicrmw or i256* %addr, i256 1 seq_cst
|
|
|
|
; FIXME: We currently can't lower 'ret i256' and it's not the purpose of this
|
|
|
|
; test so work around it by truncating to i32 for now.
|
|
|
|
%oldval.trunc = trunc i256 %oldval to i32
|
|
|
|
ret i32 %oldval.trunc
|
|
|
|
}
|
|
|
|
|
|
|
|
; Try an seq_cst atomicrmw xor
|
|
|
|
; AArch64 will expand some atomicrmw's at the LLVM-IR level so we use a wide type to avoid this.
|
|
|
|
define i32 @test_atomicrmw_xor(i256* %addr) {
|
|
|
|
; CHECK-LABEL: name: test_atomicrmw_xor
|
|
|
|
; CHECK: bb.1 (%ir-block.{{[0-9]+}}):
|
|
|
|
; CHECK-NEXT: liveins: $x0
|
|
|
|
; CHECK: [[ADDR:%[0-9]+]]:_(p0) = COPY $x0
|
|
|
|
; CHECK-NEXT: [[VAL:%[0-9]+]]:_(s256) = G_CONSTANT i256 1
|
|
|
|
; CHECK-NEXT: [[OLDVALRES:%[0-9]+]]:_(s256) = G_ATOMICRMW_XOR [[ADDR]](p0), [[VAL]] :: (load store seq_cst 32 on %ir.addr)
|
|
|
|
; CHECK-NEXT: [[RES:%[0-9]+]]:_(s32) = G_TRUNC [[OLDVALRES]]
|
|
|
|
%oldval = atomicrmw xor i256* %addr, i256 1 seq_cst
|
|
|
|
; FIXME: We currently can't lower 'ret i256' and it's not the purpose of this
|
|
|
|
; test so work around it by truncating to i32 for now.
|
|
|
|
%oldval.trunc = trunc i256 %oldval to i32
|
|
|
|
ret i32 %oldval.trunc
|
|
|
|
}
|
|
|
|
|
|
|
|
; Try an seq_cst atomicrmw min
|
|
|
|
; AArch64 will expand some atomicrmw's at the LLVM-IR level so we use a wide type to avoid this.
|
|
|
|
define i32 @test_atomicrmw_min(i256* %addr) {
|
|
|
|
; CHECK-LABEL: name: test_atomicrmw_min
|
|
|
|
; CHECK: bb.1 (%ir-block.{{[0-9]+}}):
|
|
|
|
; CHECK-NEXT: liveins: $x0
|
|
|
|
; CHECK: [[ADDR:%[0-9]+]]:_(p0) = COPY $x0
|
|
|
|
; CHECK-NEXT: [[VAL:%[0-9]+]]:_(s256) = G_CONSTANT i256 1
|
|
|
|
; CHECK-NEXT: [[OLDVALRES:%[0-9]+]]:_(s256) = G_ATOMICRMW_MIN [[ADDR]](p0), [[VAL]] :: (load store seq_cst 32 on %ir.addr)
|
|
|
|
; CHECK-NEXT: [[RES:%[0-9]+]]:_(s32) = G_TRUNC [[OLDVALRES]]
|
|
|
|
%oldval = atomicrmw min i256* %addr, i256 1 seq_cst
|
|
|
|
; FIXME: We currently can't lower 'ret i256' and it's not the purpose of this
|
|
|
|
; test so work around it by truncating to i32 for now.
|
|
|
|
%oldval.trunc = trunc i256 %oldval to i32
|
|
|
|
ret i32 %oldval.trunc
|
|
|
|
}
|
|
|
|
|
|
|
|
; Try an seq_cst atomicrmw max
|
|
|
|
; AArch64 will expand some atomicrmw's at the LLVM-IR level so we use a wide type to avoid this.
|
|
|
|
define i32 @test_atomicrmw_max(i256* %addr) {
|
|
|
|
; CHECK-LABEL: name: test_atomicrmw_max
|
|
|
|
; CHECK: bb.1 (%ir-block.{{[0-9]+}}):
|
|
|
|
; CHECK-NEXT: liveins: $x0
|
|
|
|
; CHECK: [[ADDR:%[0-9]+]]:_(p0) = COPY $x0
|
|
|
|
; CHECK-NEXT: [[VAL:%[0-9]+]]:_(s256) = G_CONSTANT i256 1
|
|
|
|
; CHECK-NEXT: [[OLDVALRES:%[0-9]+]]:_(s256) = G_ATOMICRMW_MAX [[ADDR]](p0), [[VAL]] :: (load store seq_cst 32 on %ir.addr)
|
|
|
|
; CHECK-NEXT: [[RES:%[0-9]+]]:_(s32) = G_TRUNC [[OLDVALRES]]
|
|
|
|
%oldval = atomicrmw max i256* %addr, i256 1 seq_cst
|
|
|
|
; FIXME: We currently can't lower 'ret i256' and it's not the purpose of this
|
|
|
|
; test so work around it by truncating to i32 for now.
|
|
|
|
%oldval.trunc = trunc i256 %oldval to i32
|
|
|
|
ret i32 %oldval.trunc
|
|
|
|
}
|
|
|
|
|
|
|
|
; Try an seq_cst atomicrmw unsigned min
|
|
|
|
; AArch64 will expand some atomicrmw's at the LLVM-IR level so we use a wide type to avoid this.
|
|
|
|
define i32 @test_atomicrmw_umin(i256* %addr) {
|
|
|
|
; CHECK-LABEL: name: test_atomicrmw_umin
|
|
|
|
; CHECK: bb.1 (%ir-block.{{[0-9]+}}):
|
|
|
|
; CHECK-NEXT: liveins: $x0
|
|
|
|
; CHECK: [[ADDR:%[0-9]+]]:_(p0) = COPY $x0
|
|
|
|
; CHECK-NEXT: [[VAL:%[0-9]+]]:_(s256) = G_CONSTANT i256 1
|
|
|
|
; CHECK-NEXT: [[OLDVALRES:%[0-9]+]]:_(s256) = G_ATOMICRMW_UMIN [[ADDR]](p0), [[VAL]] :: (load store seq_cst 32 on %ir.addr)
|
|
|
|
; CHECK-NEXT: [[RES:%[0-9]+]]:_(s32) = G_TRUNC [[OLDVALRES]]
|
|
|
|
%oldval = atomicrmw umin i256* %addr, i256 1 seq_cst
|
|
|
|
; FIXME: We currently can't lower 'ret i256' and it's not the purpose of this
|
|
|
|
; test so work around it by truncating to i32 for now.
|
|
|
|
%oldval.trunc = trunc i256 %oldval to i32
|
|
|
|
ret i32 %oldval.trunc
|
|
|
|
}
|
|
|
|
|
|
|
|
; Try an seq_cst atomicrmw unsigned max
|
|
|
|
; AArch64 will expand some atomicrmw's at the LLVM-IR level so we use a wide type to avoid this.
|
|
|
|
define i32 @test_atomicrmw_umax(i256* %addr) {
|
|
|
|
; CHECK-LABEL: name: test_atomicrmw_umax
|
|
|
|
; CHECK: bb.1 (%ir-block.{{[0-9]+}}):
|
|
|
|
; CHECK-NEXT: liveins: $x0
|
|
|
|
; CHECK: [[ADDR:%[0-9]+]]:_(p0) = COPY $x0
|
|
|
|
; CHECK-NEXT: [[VAL:%[0-9]+]]:_(s256) = G_CONSTANT i256 1
|
|
|
|
; CHECK-NEXT: [[OLDVALRES:%[0-9]+]]:_(s256) = G_ATOMICRMW_UMAX [[ADDR]](p0), [[VAL]] :: (load store seq_cst 32 on %ir.addr)
|
|
|
|
; CHECK-NEXT: [[RES:%[0-9]+]]:_(s32) = G_TRUNC [[OLDVALRES]]
|
|
|
|
%oldval = atomicrmw umax i256* %addr, i256 1 seq_cst
|
|
|
|
; FIXME: We currently can't lower 'ret i256' and it's not the purpose of this
|
|
|
|
; test so work around it by truncating to i32 for now.
|
|
|
|
%oldval.trunc = trunc i256 %oldval to i32
|
|
|
|
ret i32 %oldval.trunc
|
|
|
|
}
|
2018-07-31 08:08:50 +08:00
|
|
|
|
|
|
|
@addr = global i8* null
|
|
|
|
|
|
|
|
define void @test_blockaddress() {
|
|
|
|
; CHECK-LABEL: name: test_blockaddress
|
|
|
|
; CHECK: [[BADDR:%[0-9]+]]:_(p0) = G_BLOCK_ADDR blockaddress(@test_blockaddress, %ir-block.block)
|
|
|
|
; CHECK: G_STORE [[BADDR]](p0)
|
|
|
|
store i8* blockaddress(@test_blockaddress, %block), i8** @addr
|
|
|
|
indirectbr i8* blockaddress(@test_blockaddress, %block), [label %block]
|
|
|
|
block:
|
|
|
|
ret void
|
|
|
|
}
|
2018-10-06 05:02:46 +08:00
|
|
|
|
|
|
|
%t = type { i32 }
|
|
|
|
declare {}* @llvm.invariant.start.p0i8(i64, i8* nocapture) readonly nounwind
|
|
|
|
declare void @llvm.invariant.end.p0i8({}*, i64, i8* nocapture) nounwind
|
|
|
|
define void @test_invariant_intrin() {
|
|
|
|
; CHECK-LABEL: name: test_invariant_intrin
|
|
|
|
; CHECK: %{{[0-9]+}}:_(s64) = G_IMPLICIT_DEF
|
|
|
|
; CHECK-NEXT: RET_ReallyLR
|
|
|
|
%x = alloca %t
|
|
|
|
%y = bitcast %t* %x to i8*
|
|
|
|
%inv = call {}* @llvm.invariant.start.p0i8(i64 8, i8* %y)
|
|
|
|
call void @llvm.invariant.end.p0i8({}* %inv, i64 8, i8* %y)
|
|
|
|
ret void
|
|
|
|
}
|
2018-12-20 03:01:36 +08:00
|
|
|
|
|
|
|
declare float @llvm.ceil.f32(float)
|
|
|
|
define float @test_ceil_f32(float %x) {
|
|
|
|
; CHECK-LABEL: name: test_ceil_f32
|
|
|
|
; CHECK: %{{[0-9]+}}:_(s32) = G_FCEIL %{{[0-9]+}}
|
|
|
|
%y = call float @llvm.ceil.f32(float %x)
|
|
|
|
ret float %y
|
|
|
|
}
|
|
|
|
|
|
|
|
declare double @llvm.ceil.f64(double)
|
|
|
|
define double @test_ceil_f64(double %x) {
|
|
|
|
; CHECK-LABEL: name: test_ceil_f64
|
|
|
|
; CHECK: %{{[0-9]+}}:_(s64) = G_FCEIL %{{[0-9]+}}
|
|
|
|
%y = call double @llvm.ceil.f64(double %x)
|
|
|
|
ret double %y
|
|
|
|
}
|
|
|
|
|
|
|
|
declare <2 x float> @llvm.ceil.v2f32(<2 x float>)
|
|
|
|
define <2 x float> @test_ceil_v2f32(<2 x float> %x) {
|
|
|
|
; CHECK-LABEL: name: test_ceil_v2f32
|
|
|
|
; CHECK: %{{[0-9]+}}:_(<2 x s32>) = G_FCEIL %{{[0-9]+}}
|
|
|
|
%y = call <2 x float> @llvm.ceil.v2f32(<2 x float> %x)
|
|
|
|
ret <2 x float> %y
|
|
|
|
}
|
|
|
|
|
|
|
|
declare <4 x float> @llvm.ceil.v4f32(<4 x float>)
|
|
|
|
define <4 x float> @test_ceil_v4f32(<4 x float> %x) {
|
|
|
|
; CHECK-LABEL: name: test_ceil_v4f32
|
|
|
|
; CHECK: %{{[0-9]+}}:_(<4 x s32>) = G_FCEIL %{{[0-9]+}}
|
|
|
|
; SELECT: %{{[0-9]+}}:fpr128 = FRINTPv4f32 %{{[0-9]+}}
|
|
|
|
%y = call <4 x float> @llvm.ceil.v4f32(<4 x float> %x)
|
|
|
|
ret <4 x float> %y
|
|
|
|
}
|
|
|
|
|
|
|
|
declare <2 x double> @llvm.ceil.v2f64(<2 x double>)
|
|
|
|
define <2 x double> @test_ceil_v2f64(<2 x double> %x) {
|
|
|
|
; CHECK-LABEL: name: test_ceil_v2f64
|
|
|
|
; CHECK: %{{[0-9]+}}:_(<2 x s64>) = G_FCEIL %{{[0-9]+}}
|
|
|
|
%y = call <2 x double> @llvm.ceil.v2f64(<2 x double> %x)
|
|
|
|
ret <2 x double> %y
|
|
|
|
}
|
2019-01-29 02:34:17 +08:00
|
|
|
|
|
|
|
declare float @llvm.cos.f32(float)
|
|
|
|
define float @test_cos_f32(float %x) {
|
|
|
|
; CHECK-LABEL: name: test_cos_f32
|
|
|
|
; CHECK: %{{[0-9]+}}:_(s32) = G_FCOS %{{[0-9]+}}
|
|
|
|
%y = call float @llvm.cos.f32(float %x)
|
|
|
|
ret float %y
|
|
|
|
}
|
|
|
|
|
|
|
|
declare float @llvm.sin.f32(float)
|
|
|
|
define float @test_sin_f32(float %x) {
|
|
|
|
; CHECK-LABEL: name: test_sin_f32
|
|
|
|
; CHECK: %{{[0-9]+}}:_(s32) = G_FSIN %{{[0-9]+}}
|
|
|
|
%y = call float @llvm.sin.f32(float %x)
|
|
|
|
ret float %y
|
|
|
|
}
|
2019-01-31 04:58:14 +08:00
|
|
|
|
|
|
|
declare float @llvm.sqrt.f32(float)
|
|
|
|
define float @test_sqrt_f32(float %x) {
|
|
|
|
; CHECK-LABEL: name: test_sqrt_f32
|
|
|
|
; CHECK: %{{[0-9]+}}:_(s32) = G_FSQRT %{{[0-9]+}}
|
|
|
|
%y = call float @llvm.sqrt.f32(float %x)
|
|
|
|
ret float %y
|
|
|
|
}
|
2019-02-01 07:41:23 +08:00
|
|
|
|
|
|
|
; CHECK-LABEL: name: test_llvm.aarch64.neon.ld3.v4i32.p0i32
|
|
|
|
; CHECK: %1:_(s384) = G_INTRINSIC_W_SIDE_EFFECTS intrinsic(@llvm.aarch64.neon.ld3), %0(p0) :: (load 48 from %ir.ptr, align 64)
|
|
|
|
define void @test_llvm.aarch64.neon.ld3.v4i32.p0i32(i32* %ptr) {
|
|
|
|
%arst = call { <4 x i32>, <4 x i32>, <4 x i32> } @llvm.aarch64.neon.ld3.v4i32.p0i32(i32* %ptr)
|
|
|
|
ret void
|
|
|
|
}
|
|
|
|
|
|
|
|
declare { <4 x i32>, <4 x i32>, <4 x i32> } @llvm.aarch64.neon.ld3.v4i32.p0i32(i32*) #3
|