blob: 3a13bc7a7f6772343f5763a1ae19cb52ccc64875 [file] [edit]
; NOTE: Assertions have been autogenerated by utils/update_llc_test_checks.py UTC_ARGS: --version 6
; RUN: llc -verify-machineinstrs -mtriple=aarch64 < %s | FileCheck %s --check-prefixes=CHECK,CHECK-SD
; RUN: llc -verify-machineinstrs -mtriple=aarch64 -cgp-verify-bfi-updates=true < %s | FileCheck %s --check-prefixes=CHECK,CHECK-SD
; RUN: llc -verify-machineinstrs -mtriple=aarch64 -global-isel < %s | FileCheck %s --check-prefixes=CHECK,CHECK-GI
; RUN: llc -verify-machineinstrs -mtriple=aarch64 -global-isel -O0 < %s | FileCheck %s --check-prefixes=CHECK-GIO0
define void @test1(i64 %A, i64 %B) {
; CHECK-LABEL: test1:
; CHECK: // %bb.0: // %entry
; CHECK-NEXT: tbz w1, #3, .LBB0_3
; CHECK-NEXT: // %bb.1: // %entry
; CHECK-NEXT: tbz w0, #2, .LBB0_3
; CHECK-NEXT: // %bb.2: // %if.then2
; CHECK-NEXT: b foo
; CHECK-NEXT: .LBB0_3: // %if.end3
; CHECK-NEXT: ret
;
; CHECK-GIO0-LABEL: test1:
; CHECK-GIO0: // %bb.0: // %entry
; CHECK-GIO0-NEXT: sub sp, sp, #32
; CHECK-GIO0-NEXT: str x30, [sp, #16] // 8-byte Spill
; CHECK-GIO0-NEXT: .cfi_def_cfa_offset 32
; CHECK-GIO0-NEXT: .cfi_offset w30, -16
; CHECK-GIO0-NEXT: str x0, [sp] // 8-byte Spill
; CHECK-GIO0-NEXT: str x1, [sp, #8] // 8-byte Spill
; CHECK-GIO0-NEXT: mov w8, w1
; CHECK-GIO0-NEXT: tbz w8, #3, .LBB0_3
; CHECK-GIO0-NEXT: b .LBB0_1
; CHECK-GIO0-NEXT: .LBB0_1: // %entry
; CHECK-GIO0-NEXT: ldr x0, [sp] // 8-byte Reload
; CHECK-GIO0-NEXT: mov w8, w0
; CHECK-GIO0-NEXT: tbz w8, #2, .LBB0_3
; CHECK-GIO0-NEXT: b .LBB0_2
; CHECK-GIO0-NEXT: .LBB0_2: // %if.then2
; CHECK-GIO0-NEXT: ldr x1, [sp, #8] // 8-byte Reload
; CHECK-GIO0-NEXT: ldr x0, [sp] // 8-byte Reload
; CHECK-GIO0-NEXT: bl foo
; CHECK-GIO0-NEXT: b .LBB0_3
; CHECK-GIO0-NEXT: .LBB0_3: // %if.end3
; CHECK-GIO0-NEXT: ldr x30, [sp, #16] // 8-byte Reload
; CHECK-GIO0-NEXT: add sp, sp, #32
; CHECK-GIO0-NEXT: ret
entry:
%and = and i64 %A, 4
%notlhs = icmp eq i64 %and, 0
%and.1 = and i64 %B, 8
%0 = icmp eq i64 %and.1, 0
%1 = or i1 %0, %notlhs
br i1 %1, label %if.end3, label %if.then2
if.then2: ; preds = %entry
tail call void @foo(i64 %A, i64 %B)
br label %if.end3
if.end3: ; preds = %if.then2, %entry
ret void
}
define void @test2(i64 %A, ptr readonly %B) #0 {
; CHECK-LABEL: test2:
; CHECK: // %bb.0: // %entry
; CHECK-NEXT: cbz x1, .LBB1_3
; CHECK-NEXT: // %bb.1: // %entry
; CHECK-NEXT: tbz w0, #3, .LBB1_3
; CHECK-NEXT: // %bb.2: // %if.then2
; CHECK-NEXT: ldr x1, [x1]
; CHECK-NEXT: b foo
; CHECK-NEXT: .LBB1_3: // %if.end3
; CHECK-NEXT: ret
;
; CHECK-GIO0-LABEL: test2:
; CHECK-GIO0: // %bb.0: // %entry
; CHECK-GIO0-NEXT: sub sp, sp, #32
; CHECK-GIO0-NEXT: str x30, [sp, #16] // 8-byte Spill
; CHECK-GIO0-NEXT: .cfi_def_cfa_offset 32
; CHECK-GIO0-NEXT: .cfi_offset w30, -16
; CHECK-GIO0-NEXT: str x0, [sp] // 8-byte Spill
; CHECK-GIO0-NEXT: str x1, [sp, #8] // 8-byte Spill
; CHECK-GIO0-NEXT: cbz x1, .LBB1_3
; CHECK-GIO0-NEXT: b .LBB1_1
; CHECK-GIO0-NEXT: .LBB1_1: // %entry
; CHECK-GIO0-NEXT: ldr x0, [sp] // 8-byte Reload
; CHECK-GIO0-NEXT: mov w8, w0
; CHECK-GIO0-NEXT: tbz w8, #3, .LBB1_3
; CHECK-GIO0-NEXT: b .LBB1_2
; CHECK-GIO0-NEXT: .LBB1_2: // %if.then2
; CHECK-GIO0-NEXT: ldr x0, [sp] // 8-byte Reload
; CHECK-GIO0-NEXT: ldr x8, [sp, #8] // 8-byte Reload
; CHECK-GIO0-NEXT: ldr x1, [x8]
; CHECK-GIO0-NEXT: bl foo
; CHECK-GIO0-NEXT: b .LBB1_3
; CHECK-GIO0-NEXT: .LBB1_3: // %if.end3
; CHECK-GIO0-NEXT: ldr x30, [sp, #16] // 8-byte Reload
; CHECK-GIO0-NEXT: add sp, sp, #32
; CHECK-GIO0-NEXT: ret
entry:
%tobool = icmp eq ptr %B, null
%and = and i64 %A, 8
%tobool1 = icmp eq i64 %and, 0
%or.cond = or i1 %tobool, %tobool1
br i1 %or.cond, label %if.end3, label %if.then2
if.then2: ; preds = %entry
%0 = load i64, ptr %B, align 4
tail call void @foo(i64 %A, i64 %0)
br label %if.end3
if.end3: ; preds = %entry, %if.then2
ret void
}
; Make sure we use the W variant when log2(mask) is < 32.
define void @test3(i64 %A, i64 %B) {
; CHECK-LABEL: test3:
; CHECK: // %bb.0: // %entry
; CHECK-NEXT: tbz w1, #3, .LBB2_3
; CHECK-NEXT: // %bb.1: // %entry
; CHECK-NEXT: tbz w0, #28, .LBB2_3
; CHECK-NEXT: // %bb.2: // %if.end3
; CHECK-NEXT: ret
; CHECK-NEXT: .LBB2_3: // %if.then2
; CHECK-NEXT: b foo
;
; CHECK-GIO0-LABEL: test3:
; CHECK-GIO0: // %bb.0: // %entry
; CHECK-GIO0-NEXT: sub sp, sp, #32
; CHECK-GIO0-NEXT: str x30, [sp, #16] // 8-byte Spill
; CHECK-GIO0-NEXT: .cfi_def_cfa_offset 32
; CHECK-GIO0-NEXT: .cfi_offset w30, -16
; CHECK-GIO0-NEXT: str x0, [sp] // 8-byte Spill
; CHECK-GIO0-NEXT: str x1, [sp, #8] // 8-byte Spill
; CHECK-GIO0-NEXT: mov w8, w1
; CHECK-GIO0-NEXT: tbz w8, #3, .LBB2_2
; CHECK-GIO0-NEXT: b .LBB2_1
; CHECK-GIO0-NEXT: .LBB2_1: // %entry
; CHECK-GIO0-NEXT: ldr x0, [sp] // 8-byte Reload
; CHECK-GIO0-NEXT: mov w8, w0
; CHECK-GIO0-NEXT: tbnz w8, #28, .LBB2_3
; CHECK-GIO0-NEXT: b .LBB2_2
; CHECK-GIO0-NEXT: .LBB2_2: // %if.then2
; CHECK-GIO0-NEXT: ldr x1, [sp, #8] // 8-byte Reload
; CHECK-GIO0-NEXT: ldr x0, [sp] // 8-byte Reload
; CHECK-GIO0-NEXT: bl foo
; CHECK-GIO0-NEXT: b .LBB2_3
; CHECK-GIO0-NEXT: .LBB2_3: // %if.end3
; CHECK-GIO0-NEXT: ldr x30, [sp, #16] // 8-byte Reload
; CHECK-GIO0-NEXT: add sp, sp, #32
; CHECK-GIO0-NEXT: ret
entry:
%shift = shl i64 1, 28
%and = and i64 %A, %shift
%notlhs = icmp eq i64 %and, 0
%and.1 = and i64 %B, 8
%0 = icmp eq i64 %and.1, 0
%1 = or i1 %0, %notlhs
br i1 %1, label %if.then2, label %if.end3
if.then2: ; preds = %entry
tail call void @foo(i64 %A, i64 %B)
br label %if.end3
if.end3: ; preds = %if.then2, %entry
ret void
}
define void @test4(i64 %A, i64 %B) {
; CHECK-LABEL: test4:
; CHECK: // %bb.0: // %entry
; CHECK-NEXT: tbz w1, #3, .LBB3_3
; CHECK-NEXT: // %bb.1: // %entry
; CHECK-NEXT: tbz x0, #35, .LBB3_3
; CHECK-NEXT: // %bb.2: // %if.end3
; CHECK-NEXT: ret
; CHECK-NEXT: .LBB3_3: // %if.then2
; CHECK-NEXT: b foo
;
; CHECK-GIO0-LABEL: test4:
; CHECK-GIO0: // %bb.0: // %entry
; CHECK-GIO0-NEXT: sub sp, sp, #32
; CHECK-GIO0-NEXT: str x30, [sp, #16] // 8-byte Spill
; CHECK-GIO0-NEXT: .cfi_def_cfa_offset 32
; CHECK-GIO0-NEXT: .cfi_offset w30, -16
; CHECK-GIO0-NEXT: str x0, [sp] // 8-byte Spill
; CHECK-GIO0-NEXT: str x1, [sp, #8] // 8-byte Spill
; CHECK-GIO0-NEXT: mov w8, w1
; CHECK-GIO0-NEXT: tbz w8, #3, .LBB3_2
; CHECK-GIO0-NEXT: b .LBB3_1
; CHECK-GIO0-NEXT: .LBB3_1: // %entry
; CHECK-GIO0-NEXT: ldr x8, [sp] // 8-byte Reload
; CHECK-GIO0-NEXT: tbnz x8, #35, .LBB3_3
; CHECK-GIO0-NEXT: b .LBB3_2
; CHECK-GIO0-NEXT: .LBB3_2: // %if.then2
; CHECK-GIO0-NEXT: ldr x1, [sp, #8] // 8-byte Reload
; CHECK-GIO0-NEXT: ldr x0, [sp] // 8-byte Reload
; CHECK-GIO0-NEXT: bl foo
; CHECK-GIO0-NEXT: b .LBB3_3
; CHECK-GIO0-NEXT: .LBB3_3: // %if.end3
; CHECK-GIO0-NEXT: ldr x30, [sp, #16] // 8-byte Reload
; CHECK-GIO0-NEXT: add sp, sp, #32
; CHECK-GIO0-NEXT: ret
entry:
%shift = shl i64 1, 35
%and = and i64 %A, %shift
%notlhs = icmp eq i64 %and, 0
%and.1 = and i64 %B, 8
%0 = icmp eq i64 %and.1, 0
%1 = or i1 %0, %notlhs
br i1 %1, label %if.then2, label %if.end3
if.then2: ; preds = %entry
tail call void @foo(i64 %A, i64 %B)
br label %if.end3
if.end3: ; preds = %if.then2, %entry
ret void
}
define i32 @tbzfromextract(<8 x i8> %b) {
; CHECK-LABEL: tbzfromextract:
; CHECK: // %bb.0: // %common.ret
; CHECK-NEXT: mov w0, #1 // =0x1
; CHECK-NEXT: ret
;
; CHECK-GIO0-LABEL: tbzfromextract:
; CHECK-GIO0: // %bb.0:
; CHECK-GIO0-NEXT: fmov d1, d0
; CHECK-GIO0-NEXT: // implicit-def: $q0
; CHECK-GIO0-NEXT: fmov d0, d1
; CHECK-GIO0-NEXT: umov w8, v0.b[0]
; CHECK-GIO0-NEXT: tbnz w8, #31, .LBB4_2
; CHECK-GIO0-NEXT: b .LBB4_1
; CHECK-GIO0-NEXT: .LBB4_1: // %land.rhs
; CHECK-GIO0-NEXT: mov w0, #1 // =0x1
; CHECK-GIO0-NEXT: ret
; CHECK-GIO0-NEXT: .LBB4_2: // %land.end
; CHECK-GIO0-NEXT: mov w0, wzr
; CHECK-GIO0-NEXT: ret
%e = extractelement <8 x i8> %b, i32 0
%z = zext i8 %e to i32
%cmp = icmp sge i32 %z, 0
br i1 %cmp, label %land.rhs, label %land.end
land.rhs:
ret i32 1
land.end:
ret i32 0
}
declare void @foo(i64, i64)
;; NOTE: These prefixes are unused and the list is autogenerated. Do not add tests below this line:
; CHECK-GI: {{.*}}
; CHECK-SD: {{.*}}