Skip to content

Commit 2c9185e

Browse files
author
Fei Gao
committedJun 12, 2024
8321308: AArch64: Fix matching predication for cbz/cbnz
Reviewed-by: fyang, adinn, aph
1 parent 5a8a9fd commit 2c9185e

File tree

3 files changed

+131
-15
lines changed

3 files changed

+131
-15
lines changed
 

‎src/hotspot/cpu/aarch64/aarch64.ad

+19-15
Original file line numberDiff line numberDiff line change
@@ -5628,24 +5628,24 @@ operand cmpOpLtGe()
56285628
// used for certain unsigned integral comparisons which can be
56295629
// converted to cbxx or tbxx instructions
56305630

5631-
operand cmpOpUEqNeLtGe()
5631+
operand cmpOpUEqNeLeGt()
56325632
%{
56335633
match(Bool);
56345634
op_cost(0);
56355635

5636-
predicate(n->as_Bool()->_test._test == BoolTest::eq
5637-
|| n->as_Bool()->_test._test == BoolTest::ne
5638-
|| n->as_Bool()->_test._test == BoolTest::lt
5639-
|| n->as_Bool()->_test._test == BoolTest::ge);
5636+
predicate(n->as_Bool()->_test._test == BoolTest::eq ||
5637+
n->as_Bool()->_test._test == BoolTest::ne ||
5638+
n->as_Bool()->_test._test == BoolTest::le ||
5639+
n->as_Bool()->_test._test == BoolTest::gt);
56405640

56415641
format %{ "" %}
56425642
interface(COND_INTER) %{
56435643
equal(0x0, "eq");
56445644
not_equal(0x1, "ne");
5645-
less(0xb, "lt");
5646-
greater_equal(0xa, "ge");
5647-
less_equal(0xd, "le");
5648-
greater(0xc, "gt");
5645+
less(0x3, "lo");
5646+
greater_equal(0x2, "hs");
5647+
less_equal(0x9, "ls");
5648+
greater(0x8, "hi");
56495649
overflow(0x6, "vs");
56505650
no_overflow(0x7, "vc");
56515651
%}
@@ -15687,7 +15687,7 @@ instruct cmpP_narrowOop_imm0_branch(cmpOpEqNe cmp, iRegN oop, immP0 zero, label
1568715687
ins_pipe(pipe_cmp_branch);
1568815688
%}
1568915689

15690-
instruct cmpUI_imm0_branch(cmpOpUEqNeLtGe cmp, iRegIorL2I op1, immI0 op2, label labl, rFlagsRegU cr) %{
15690+
instruct cmpUI_imm0_branch(cmpOpUEqNeLeGt cmp, iRegIorL2I op1, immI0 op2, label labl) %{
1569115691
match(If cmp (CmpU op1 op2));
1569215692
effect(USE labl);
1569315693

@@ -15696,15 +15696,17 @@ instruct cmpUI_imm0_branch(cmpOpUEqNeLtGe cmp, iRegIorL2I op1, immI0 op2, label
1569615696
ins_encode %{
1569715697
Label* L = $labl$$label;
1569815698
Assembler::Condition cond = (Assembler::Condition)$cmp$$cmpcode;
15699-
if (cond == Assembler::EQ || cond == Assembler::LS)
15699+
if (cond == Assembler::EQ || cond == Assembler::LS) {
1570015700
__ cbzw($op1$$Register, *L);
15701-
else
15701+
} else {
15702+
assert(cond == Assembler::NE || cond == Assembler::HI, "unexpected condition");
1570215703
__ cbnzw($op1$$Register, *L);
15704+
}
1570315705
%}
1570415706
ins_pipe(pipe_cmp_branch);
1570515707
%}
1570615708

15707-
instruct cmpUL_imm0_branch(cmpOpUEqNeLtGe cmp, iRegL op1, immL0 op2, label labl, rFlagsRegU cr) %{
15709+
instruct cmpUL_imm0_branch(cmpOpUEqNeLeGt cmp, iRegL op1, immL0 op2, label labl) %{
1570815710
match(If cmp (CmpUL op1 op2));
1570915711
effect(USE labl);
1571015712

@@ -15713,10 +15715,12 @@ instruct cmpUL_imm0_branch(cmpOpUEqNeLtGe cmp, iRegL op1, immL0 op2, label labl,
1571315715
ins_encode %{
1571415716
Label* L = $labl$$label;
1571515717
Assembler::Condition cond = (Assembler::Condition)$cmp$$cmpcode;
15716-
if (cond == Assembler::EQ || cond == Assembler::LS)
15718+
if (cond == Assembler::EQ || cond == Assembler::LS) {
1571715719
__ cbz($op1$$Register, *L);
15718-
else
15720+
} else {
15721+
assert(cond == Assembler::NE || cond == Assembler::HI, "unexpected condition");
1571915722
__ cbnz($op1$$Register, *L);
15723+
}
1572015724
%}
1572115725
ins_pipe(pipe_cmp_branch);
1572215726
%}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,92 @@
1+
/*
2+
* Copyright (c) 2024, Arm Limited. All rights reserved.
3+
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4+
*
5+
* This code is free software; you can redistribute it and/or modify it
6+
* under the terms of the GNU General Public License version 2 only, as
7+
* published by the Free Software Foundation.
8+
*
9+
* This code is distributed in the hope that it will be useful, but WITHOUT
10+
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
11+
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
12+
* version 2 for more details (a copy is included in the LICENSE file that
13+
* accompanied this code).
14+
*
15+
* You should have received a copy of the GNU General Public License version
16+
* 2 along with this work; if not, write to the Free Software Foundation,
17+
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
18+
*
19+
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
20+
* or visit www.oracle.com if you need additional information or have any
21+
* questions.
22+
*/
23+
24+
package compiler.c2.irTests;
25+
26+
import compiler.lib.ir_framework.*;
27+
import jdk.test.lib.Asserts;
28+
29+
/*
30+
* @test
31+
* @bug 8321308
32+
* @summary AArch64: Fix matching predication for cbz/cbnz
33+
* @requires os.arch=="aarch64"
34+
* @library /test/lib /
35+
* @run driver compiler.c2.irTests.TestArrLenCheckOptimization
36+
*/
37+
38+
public class TestArrLenCheckOptimization {
39+
40+
int result = 0;
41+
42+
@Test
43+
@IR(counts = {IRNode.CBZW_LS, "1"})
44+
void test_le_int(int ia[]) {
45+
result += ia[0];
46+
}
47+
48+
@Test
49+
@IR(counts = {IRNode.CBNZW_HI, "1"})
50+
void test_gt_int(int ia[]) {
51+
if (ia.length > 0) {
52+
result += 0x88;
53+
} else {
54+
result -= 1;
55+
}
56+
}
57+
58+
@Test
59+
@IR(counts = {IRNode.CBZ_LS, "1"})
60+
void test_le_long(int ia[]) {
61+
if (Long.compareUnsigned(ia.length, 0) > 0) {
62+
result += 0x80;
63+
} else {
64+
result -= 1;
65+
}
66+
}
67+
68+
@Test
69+
@IR(counts = {IRNode.CBZ_HI, "1"})
70+
void test_gt_long(int ia[]) {
71+
if (Long.compareUnsigned(ia.length, 0) > 0) {
72+
result += 0x82;
73+
} else {
74+
result -= 1;
75+
}
76+
}
77+
78+
@Run(test = {"test_le_int", "test_gt_int", "test_le_long", "test_gt_long"},
79+
mode = RunMode.STANDALONE)
80+
public void test_runner() {
81+
for (int i = 0; i < 10_000; i++) {
82+
test_le_int(new int[1]);
83+
test_gt_int(new int[0]);
84+
test_le_long(new int[1]);
85+
test_gt_long(new int[0]);
86+
}
87+
}
88+
89+
public static void main(String [] args) {
90+
TestFramework.run();
91+
}
92+
}

‎test/hotspot/jtreg/compiler/lib/ir_framework/IRNode.java

+20
Original file line numberDiff line numberDiff line change
@@ -378,6 +378,26 @@ public class IRNode {
378378
beforeMatchingNameRegex(CAST_LL, "CastLL");
379379
}
380380

381+
public static final String CBNZW_HI = PREFIX + "CBNZW_HI" + POSTFIX;
382+
static {
383+
optoOnly(CBNZW_HI, "cbwhi");
384+
}
385+
386+
public static final String CBZW_LS = PREFIX + "CBZW_LS" + POSTFIX;
387+
static {
388+
optoOnly(CBZW_LS, "cbwls");
389+
}
390+
391+
public static final String CBZ_LS = PREFIX + "CBZ_LS" + POSTFIX;
392+
static {
393+
optoOnly(CBZ_LS, "cbls");
394+
}
395+
396+
public static final String CBZ_HI = PREFIX + "CBZ_HI" + POSTFIX;
397+
static {
398+
optoOnly(CBZ_HI, "cbhi");
399+
}
400+
381401
public static final String CHECKCAST_ARRAY = PREFIX + "CHECKCAST_ARRAY" + POSTFIX;
382402
static {
383403
String regex = "(((?i:cmp|CLFI|CLR).*precise \\[.*:|.*(?i:mov|mv|or).*precise \\[.*:.*\\R.*(cmp|CMP|CLR))" + END;

0 commit comments

Comments
 (0)
Please sign in to comment.