445 lines
12 KiB
Plaintext
445 lines
12 KiB
Plaintext
|
# RUN: llc -mtriple=aarch64--linux-gnu -mattr=+sve -run-pass=peephole-opt -verify-machineinstrs %s -o - | FileCheck %s
|
||
|
|
||
|
# Test instruction sequences where PTEST is redundant and thus gets removed.
|
||
|
---
|
||
|
name: whilels_b8_s32
|
||
|
alignment: 2
|
||
|
tracksRegLiveness: true
|
||
|
registers:
|
||
|
- { id: 0, class: gpr32 }
|
||
|
- { id: 1, class: gpr32 }
|
||
|
- { id: 2, class: ppr }
|
||
|
- { id: 3, class: ppr }
|
||
|
- { id: 4, class: gpr32 }
|
||
|
- { id: 5, class: gpr32 }
|
||
|
liveins:
|
||
|
- { reg: '$w0', virtual-reg: '%0' }
|
||
|
- { reg: '$w1', virtual-reg: '%1' }
|
||
|
frameInfo:
|
||
|
maxCallFrameSize: 0
|
||
|
body: |
|
||
|
bb.0.entry:
|
||
|
liveins: $w0, $w1
|
||
|
|
||
|
; Here we check the expected sequence with subsequent tests
|
||
|
; just asserting there is no PTEST instruction.
|
||
|
;
|
||
|
; CHECK-LABEL: name: whilels_b8_s32
|
||
|
; CHECK: %3:ppr = WHILELS_PWW_B %0, %1, implicit-def $nzcv
|
||
|
; CHECK-NEXT: %4:gpr32 = COPY $wzr
|
||
|
; CHECK-NEXT: %5:gpr32 = CSINCWr %4, $wzr, 0, implicit $nzcv
|
||
|
%1:gpr32 = COPY $w1
|
||
|
%0:gpr32 = COPY $w0
|
||
|
%2:ppr = PTRUE_B 31
|
||
|
%3:ppr = WHILELS_PWW_B %0, %1, implicit-def dead $nzcv
|
||
|
PTEST_PP killed %2, killed %3, implicit-def $nzcv
|
||
|
%4:gpr32 = COPY $wzr
|
||
|
%5:gpr32 = CSINCWr %4, $wzr, 0, implicit $nzcv
|
||
|
$w0 = COPY %5
|
||
|
RET_ReallyLR implicit $w0
|
||
|
|
||
|
...
|
||
|
---
|
||
|
name: whilels_b8_s64
|
||
|
alignment: 2
|
||
|
tracksRegLiveness: true
|
||
|
registers:
|
||
|
- { id: 0, class: gpr64 }
|
||
|
- { id: 1, class: gpr64 }
|
||
|
- { id: 2, class: ppr }
|
||
|
- { id: 3, class: ppr }
|
||
|
- { id: 4, class: gpr32 }
|
||
|
- { id: 5, class: gpr32 }
|
||
|
liveins:
|
||
|
- { reg: '$x0', virtual-reg: '%0' }
|
||
|
- { reg: '$x1', virtual-reg: '%1' }
|
||
|
frameInfo:
|
||
|
maxCallFrameSize: 0
|
||
|
body: |
|
||
|
bb.0.entry:
|
||
|
liveins: $x0, $x1
|
||
|
|
||
|
; CHECK-LABEL: name: whilels_b8_s64
|
||
|
; CHECK-NOT: PTEST
|
||
|
%1:gpr64 = COPY $x1
|
||
|
%0:gpr64 = COPY $x0
|
||
|
%2:ppr = PTRUE_B 31
|
||
|
%3:ppr = WHILELS_PXX_B %0, %1, implicit-def dead $nzcv
|
||
|
PTEST_PP killed %2, killed %3, implicit-def $nzcv
|
||
|
%4:gpr32 = COPY $wzr
|
||
|
%5:gpr32 = CSINCWr %4, $wzr, 0, implicit $nzcv
|
||
|
$w0 = COPY %5
|
||
|
RET_ReallyLR implicit $w0
|
||
|
|
||
|
...
|
||
|
---
|
||
|
name: whilels_b16_s32
|
||
|
alignment: 2
|
||
|
tracksRegLiveness: true
|
||
|
registers:
|
||
|
- { id: 0, class: gpr32 }
|
||
|
- { id: 1, class: gpr32 }
|
||
|
- { id: 2, class: ppr }
|
||
|
- { id: 3, class: ppr }
|
||
|
- { id: 4, class: ppr }
|
||
|
- { id: 5, class: ppr }
|
||
|
- { id: 6, class: gpr32 }
|
||
|
- { id: 7, class: gpr32 }
|
||
|
liveins:
|
||
|
- { reg: '$w0', virtual-reg: '%0' }
|
||
|
- { reg: '$w1', virtual-reg: '%1' }
|
||
|
frameInfo:
|
||
|
maxCallFrameSize: 0
|
||
|
body: |
|
||
|
bb.0.entry:
|
||
|
liveins: $w0, $w1
|
||
|
|
||
|
; CHECK-LABEL: name: whilels_b16_s32
|
||
|
; CHECK-NOT: PTEST
|
||
|
%1:gpr32 = COPY $w1
|
||
|
%0:gpr32 = COPY $w0
|
||
|
%2:ppr = PTRUE_H 31
|
||
|
%4:ppr = WHILELS_PWW_H %0, %1, implicit-def dead $nzcv
|
||
|
PTEST_PP %2, %4, implicit-def $nzcv
|
||
|
%6:gpr32 = COPY $wzr
|
||
|
%7:gpr32 = CSINCWr %6, $wzr, 0, implicit $nzcv
|
||
|
$w0 = COPY %7
|
||
|
RET_ReallyLR implicit $w0
|
||
|
|
||
|
...
|
||
|
---
|
||
|
name: whilels_b16_s64
|
||
|
alignment: 2
|
||
|
tracksRegLiveness: true
|
||
|
registers:
|
||
|
- { id: 0, class: gpr64 }
|
||
|
- { id: 1, class: gpr64 }
|
||
|
- { id: 2, class: ppr }
|
||
|
- { id: 3, class: ppr }
|
||
|
- { id: 4, class: ppr }
|
||
|
- { id: 5, class: ppr }
|
||
|
- { id: 6, class: gpr32 }
|
||
|
- { id: 7, class: gpr32 }
|
||
|
liveins:
|
||
|
- { reg: '$x0', virtual-reg: '%0' }
|
||
|
- { reg: '$x1', virtual-reg: '%1' }
|
||
|
frameInfo:
|
||
|
maxCallFrameSize: 0
|
||
|
body: |
|
||
|
bb.0.entry:
|
||
|
liveins: $x0, $x1
|
||
|
|
||
|
; CHECK-LABEL: name: whilels_b16_s64
|
||
|
; CHECK-NOT: PTEST
|
||
|
%1:gpr64 = COPY $x1
|
||
|
%0:gpr64 = COPY $x0
|
||
|
%2:ppr = PTRUE_H 31
|
||
|
%4:ppr = WHILELS_PXX_H %0, %1, implicit-def dead $nzcv
|
||
|
PTEST_PP %2, %4, implicit-def $nzcv
|
||
|
%6:gpr32 = COPY $wzr
|
||
|
%7:gpr32 = CSINCWr %6, $wzr, 0, implicit $nzcv
|
||
|
$w0 = COPY %7
|
||
|
RET_ReallyLR implicit $w0
|
||
|
|
||
|
...
|
||
|
---
|
||
|
name: whilels_b32_s32
|
||
|
alignment: 2
|
||
|
tracksRegLiveness: true
|
||
|
registers:
|
||
|
- { id: 0, class: gpr32 }
|
||
|
- { id: 1, class: gpr32 }
|
||
|
- { id: 2, class: ppr }
|
||
|
- { id: 3, class: ppr }
|
||
|
- { id: 4, class: ppr }
|
||
|
- { id: 5, class: ppr }
|
||
|
- { id: 6, class: gpr32 }
|
||
|
- { id: 7, class: gpr32 }
|
||
|
liveins:
|
||
|
- { reg: '$w0', virtual-reg: '%0' }
|
||
|
- { reg: '$w1', virtual-reg: '%1' }
|
||
|
frameInfo:
|
||
|
maxCallFrameSize: 0
|
||
|
body: |
|
||
|
bb.0.entry:
|
||
|
liveins: $w0, $w1
|
||
|
|
||
|
; CHECK-LABEL: name: whilels_b32_s32
|
||
|
; CHECK-NOT: PTEST
|
||
|
%1:gpr32 = COPY $w1
|
||
|
%0:gpr32 = COPY $w0
|
||
|
%2:ppr = PTRUE_S 31
|
||
|
%4:ppr = WHILELS_PWW_S %0, %1, implicit-def dead $nzcv
|
||
|
PTEST_PP %2, %4, implicit-def $nzcv
|
||
|
%6:gpr32 = COPY $wzr
|
||
|
%7:gpr32 = CSINCWr %6, $wzr, 0, implicit $nzcv
|
||
|
$w0 = COPY %7
|
||
|
RET_ReallyLR implicit $w0
|
||
|
|
||
|
...
|
||
|
---
|
||
|
name: whilels_b32_s64
|
||
|
alignment: 2
|
||
|
tracksRegLiveness: true
|
||
|
registers:
|
||
|
- { id: 0, class: gpr64 }
|
||
|
- { id: 1, class: gpr64 }
|
||
|
- { id: 2, class: ppr }
|
||
|
- { id: 3, class: ppr }
|
||
|
- { id: 4, class: ppr }
|
||
|
- { id: 5, class: ppr }
|
||
|
- { id: 6, class: gpr32 }
|
||
|
- { id: 7, class: gpr32 }
|
||
|
liveins:
|
||
|
- { reg: '$x0', virtual-reg: '%0' }
|
||
|
- { reg: '$x1', virtual-reg: '%1' }
|
||
|
frameInfo:
|
||
|
maxCallFrameSize: 0
|
||
|
body: |
|
||
|
bb.0.entry:
|
||
|
liveins: $x0, $x1
|
||
|
|
||
|
; CHECK-LABEL: name: whilels_b32_s64
|
||
|
; CHECK-NOT: PTEST
|
||
|
%1:gpr64 = COPY $x1
|
||
|
%0:gpr64 = COPY $x0
|
||
|
%2:ppr = PTRUE_S 31
|
||
|
%4:ppr = WHILELS_PXX_S %0, %1, implicit-def dead $nzcv
|
||
|
PTEST_PP %2, %4, implicit-def $nzcv
|
||
|
%6:gpr32 = COPY $wzr
|
||
|
%7:gpr32 = CSINCWr %6, $wzr, 0, implicit $nzcv
|
||
|
$w0 = COPY %7
|
||
|
RET_ReallyLR implicit $w0
|
||
|
|
||
|
...
|
||
|
---
|
||
|
name: whilels_b64_s32
|
||
|
alignment: 2
|
||
|
tracksRegLiveness: true
|
||
|
registers:
|
||
|
- { id: 0, class: gpr32 }
|
||
|
- { id: 1, class: gpr32 }
|
||
|
- { id: 2, class: ppr }
|
||
|
- { id: 3, class: ppr }
|
||
|
- { id: 4, class: ppr }
|
||
|
- { id: 5, class: ppr }
|
||
|
- { id: 6, class: gpr32 }
|
||
|
- { id: 7, class: gpr32 }
|
||
|
liveins:
|
||
|
- { reg: '$w0', virtual-reg: '%0' }
|
||
|
- { reg: '$w1', virtual-reg: '%1' }
|
||
|
frameInfo:
|
||
|
maxCallFrameSize: 0
|
||
|
body: |
|
||
|
bb.0.entry:
|
||
|
liveins: $w0, $w1
|
||
|
|
||
|
; CHECK-LABEL: name: whilels_b64_s32
|
||
|
; CHECK-NOT: PTEST
|
||
|
%1:gpr32 = COPY $w1
|
||
|
%0:gpr32 = COPY $w0
|
||
|
%2:ppr = PTRUE_D 31
|
||
|
%4:ppr = WHILELS_PWW_D %0, %1, implicit-def dead $nzcv
|
||
|
PTEST_PP %2, %4, implicit-def $nzcv
|
||
|
%6:gpr32 = COPY $wzr
|
||
|
%7:gpr32 = CSINCWr %6, $wzr, 0, implicit $nzcv
|
||
|
$w0 = COPY %7
|
||
|
RET_ReallyLR implicit $w0
|
||
|
|
||
|
...
|
||
|
---
|
||
|
name: whilels_b64_s64
|
||
|
alignment: 2
|
||
|
tracksRegLiveness: true
|
||
|
registers:
|
||
|
- { id: 0, class: gpr64 }
|
||
|
- { id: 1, class: gpr64 }
|
||
|
- { id: 2, class: ppr }
|
||
|
- { id: 3, class: ppr }
|
||
|
- { id: 4, class: ppr }
|
||
|
- { id: 5, class: ppr }
|
||
|
- { id: 6, class: gpr32 }
|
||
|
- { id: 7, class: gpr32 }
|
||
|
liveins:
|
||
|
- { reg: '$x0', virtual-reg: '%0' }
|
||
|
- { reg: '$x1', virtual-reg: '%1' }
|
||
|
frameInfo:
|
||
|
maxCallFrameSize: 0
|
||
|
body: |
|
||
|
bb.0.entry:
|
||
|
liveins: $x0, $x1
|
||
|
|
||
|
; CHECK-LABEL: name: whilels_b64_s64
|
||
|
; CHECK-NOT: PTEST
|
||
|
%1:gpr64 = COPY $x1
|
||
|
%0:gpr64 = COPY $x0
|
||
|
%2:ppr = PTRUE_D 31
|
||
|
%4:ppr = WHILELS_PXX_D %0, %1, implicit-def dead $nzcv
|
||
|
PTEST_PP %2, %4, implicit-def $nzcv
|
||
|
%6:gpr32 = COPY $wzr
|
||
|
%7:gpr32 = CSINCWr %6, $wzr, 0, implicit $nzcv
|
||
|
$w0 = COPY %7
|
||
|
RET_ReallyLR implicit $w0
|
||
|
|
||
|
...
|
||
|
---
|
||
|
name: whilels_b32_s32_keep_ptest_not_all_active
|
||
|
alignment: 2
|
||
|
tracksRegLiveness: true
|
||
|
registers:
|
||
|
- { id: 0, class: gpr32 }
|
||
|
- { id: 1, class: gpr32 }
|
||
|
- { id: 2, class: ppr }
|
||
|
- { id: 3, class: ppr }
|
||
|
- { id: 4, class: gpr32 }
|
||
|
- { id: 5, class: gpr32 }
|
||
|
liveins:
|
||
|
- { reg: '$w0', virtual-reg: '%0' }
|
||
|
- { reg: '$w1', virtual-reg: '%1' }
|
||
|
frameInfo:
|
||
|
maxCallFrameSize: 0
|
||
|
body: |
|
||
|
bb.0.entry:
|
||
|
liveins: $w0, $w1
|
||
|
|
||
|
; PTEST is not redundant when it's Pg operand is not an all active predicate
|
||
|
; of element size matching the WHILELS, which is the implicitly predicate
|
||
|
; used by WHILE when calculating the condition codes.
|
||
|
;
|
||
|
; CHECK-LABEL: name: whilels_b32_s32_keep_ptest_not_all_active
|
||
|
; CHECK: %3:ppr = WHILELS_PWW_S %0, %1, implicit-def dead $nzcv
|
||
|
; CHECK-NEXT: PTEST_PP killed %2, killed %3, implicit-def $nzcv
|
||
|
; CHECK-NEXT: %4:gpr32 = COPY $wzr
|
||
|
; CHECK-NEXT: %5:gpr32 = CSINCWr %4, $wzr, 0, implicit $nzcv
|
||
|
%1:gpr32 = COPY $w1
|
||
|
%0:gpr32 = COPY $w0
|
||
|
%2:ppr = PTRUE_S 5
|
||
|
%3:ppr = WHILELS_PWW_S %0, %1, implicit-def dead $nzcv
|
||
|
PTEST_PP killed %2, killed %3, implicit-def $nzcv
|
||
|
%4:gpr32 = COPY $wzr
|
||
|
%5:gpr32 = CSINCWr %4, $wzr, 0, implicit $nzcv
|
||
|
$w0 = COPY %5
|
||
|
RET_ReallyLR implicit $w0
|
||
|
|
||
|
...
|
||
|
---
|
||
|
name: whilels_b32_s32_keep_ptest_of_bytes
|
||
|
alignment: 2
|
||
|
tracksRegLiveness: true
|
||
|
registers:
|
||
|
- { id: 0, class: gpr32 }
|
||
|
- { id: 1, class: gpr32 }
|
||
|
- { id: 2, class: ppr }
|
||
|
- { id: 3, class: ppr }
|
||
|
- { id: 4, class: gpr32 }
|
||
|
- { id: 5, class: gpr32 }
|
||
|
liveins:
|
||
|
- { reg: '$w0', virtual-reg: '%0' }
|
||
|
- { reg: '$w1', virtual-reg: '%1' }
|
||
|
frameInfo:
|
||
|
maxCallFrameSize: 0
|
||
|
body: |
|
||
|
bb.0.entry:
|
||
|
liveins: $w0, $w1
|
||
|
|
||
|
; PTEST is not redundant when it's Pg operand is not an all active predicate
|
||
|
; of element size matching the WHILELS, which is the implicitly predicate
|
||
|
; used by WHILE when calculating the condition codes.
|
||
|
;
|
||
|
; CHECK-LABEL: name: whilels_b32_s32_keep_ptest_of_bytes
|
||
|
; CHECK: %3:ppr = WHILELS_PWW_S %0, %1, implicit-def dead $nzcv
|
||
|
; CHECK-NEXT: PTEST_PP killed %2, killed %3, implicit-def $nzcv
|
||
|
; CHECK-NEXT: %4:gpr32 = COPY $wzr
|
||
|
; CHECK-NEXT: %5:gpr32 = CSINCWr %4, $wzr, 0, implicit $nzcv
|
||
|
%1:gpr32 = COPY $w1
|
||
|
%0:gpr32 = COPY $w0
|
||
|
%2:ppr = PTRUE_B 31
|
||
|
%3:ppr = WHILELS_PWW_S %0, %1, implicit-def dead $nzcv
|
||
|
PTEST_PP killed %2, killed %3, implicit-def $nzcv
|
||
|
%4:gpr32 = COPY $wzr
|
||
|
%5:gpr32 = CSINCWr %4, $wzr, 0, implicit $nzcv
|
||
|
$w0 = COPY %5
|
||
|
RET_ReallyLR implicit $w0
|
||
|
|
||
|
...
|
||
|
---
|
||
|
name: whilels_b32_s32_keep_ptest_of_halfs
|
||
|
alignment: 2
|
||
|
tracksRegLiveness: true
|
||
|
registers:
|
||
|
- { id: 0, class: gpr32 }
|
||
|
- { id: 1, class: gpr32 }
|
||
|
- { id: 2, class: ppr }
|
||
|
- { id: 3, class: ppr }
|
||
|
- { id: 4, class: gpr32 }
|
||
|
- { id: 5, class: gpr32 }
|
||
|
liveins:
|
||
|
- { reg: '$w0', virtual-reg: '%0' }
|
||
|
- { reg: '$w1', virtual-reg: '%1' }
|
||
|
frameInfo:
|
||
|
maxCallFrameSize: 0
|
||
|
body: |
|
||
|
bb.0.entry:
|
||
|
liveins: $w0, $w1
|
||
|
|
||
|
; PTEST is not redundant when it's Pg operand is not an all active predicate
|
||
|
; of element size matching the WHILELS, which is the implicitly predicate
|
||
|
; used by WHILE when calculating the condition codes.
|
||
|
;
|
||
|
; CHECK-LABEL: name: whilels_b32_s32_keep_ptest_of_halfs
|
||
|
; CHECK: %3:ppr = WHILELS_PWW_S %0, %1, implicit-def dead $nzcv
|
||
|
; CHECK-NEXT: PTEST_PP killed %2, killed %3, implicit-def $nzcv
|
||
|
; CHECK-NEXT: %4:gpr32 = COPY $wzr
|
||
|
; CHECK-NEXT: %5:gpr32 = CSINCWr %4, $wzr, 0, implicit $nzcv
|
||
|
%1:gpr32 = COPY $w1
|
||
|
%0:gpr32 = COPY $w0
|
||
|
%2:ppr = PTRUE_H 31
|
||
|
%3:ppr = WHILELS_PWW_S %0, %1, implicit-def dead $nzcv
|
||
|
PTEST_PP killed %2, killed %3, implicit-def $nzcv
|
||
|
%4:gpr32 = COPY $wzr
|
||
|
%5:gpr32 = CSINCWr %4, $wzr, 0, implicit $nzcv
|
||
|
$w0 = COPY %5
|
||
|
RET_ReallyLR implicit $w0
|
||
|
|
||
|
...
|
||
|
---
|
||
|
name: whilels_b32_s32_keep_ptest_of_doublewords
|
||
|
alignment: 2
|
||
|
tracksRegLiveness: true
|
||
|
registers:
|
||
|
- { id: 0, class: gpr32 }
|
||
|
- { id: 1, class: gpr32 }
|
||
|
- { id: 2, class: ppr }
|
||
|
- { id: 3, class: ppr }
|
||
|
- { id: 4, class: gpr32 }
|
||
|
- { id: 5, class: gpr32 }
|
||
|
liveins:
|
||
|
- { reg: '$w0', virtual-reg: '%0' }
|
||
|
- { reg: '$w1', virtual-reg: '%1' }
|
||
|
frameInfo:
|
||
|
maxCallFrameSize: 0
|
||
|
body: |
|
||
|
bb.0.entry:
|
||
|
liveins: $w0, $w1
|
||
|
|
||
|
; PTEST is not redundant when it's Pg operand is not an all active predicate
|
||
|
; of element size matching the WHILELS, which is the implicitly predicate
|
||
|
; used by WHILE when calculating the condition codes.
|
||
|
;
|
||
|
; CHECK-LABEL: name: whilels_b32_s32_keep_ptest_of_doublewords
|
||
|
; CHECK: %3:ppr = WHILELS_PWW_S %0, %1, implicit-def dead $nzcv
|
||
|
; CHECK-NEXT: PTEST_PP killed %2, killed %3, implicit-def $nzcv
|
||
|
; CHECK-NEXT: %4:gpr32 = COPY $wzr
|
||
|
; CHECK-NEXT: %5:gpr32 = CSINCWr %4, $wzr, 0, implicit $nzcv
|
||
|
%1:gpr32 = COPY $w1
|
||
|
%0:gpr32 = COPY $w0
|
||
|
%2:ppr = PTRUE_D 31
|
||
|
%3:ppr = WHILELS_PWW_S %0, %1, implicit-def dead $nzcv
|
||
|
PTEST_PP killed %2, killed %3, implicit-def $nzcv
|
||
|
%4:gpr32 = COPY $wzr
|
||
|
%5:gpr32 = CSINCWr %4, $wzr, 0, implicit $nzcv
|
||
|
$w0 = COPY %5
|
||
|
RET_ReallyLR implicit $w0
|
||
|
|
||
|
...
|