Skip to content

Commit c856b34

Browse files
committedMar 25, 2025
8352587: C2 SuperWord: we must avoid Multiversioning for PeelMainPost loops
Reviewed-by: chagedorn, kvn
1 parent 993eae4 commit c856b34

File tree

5 files changed

+139
-5
lines changed

5 files changed

+139
-5
lines changed
 

‎src/hotspot/share/opto/loopTransform.cpp

+9-4
Original file line numberDiff line numberDiff line change
@@ -3429,10 +3429,15 @@ bool IdealLoopTree::iteration_split_impl(PhaseIdealLoop *phase, Node_List &old_n
34293429
return false;
34303430
}
34313431

3432-
// We are going to add pre-loop and post-loop.
3433-
// But should we also multi-version for auto-vectorization speculative
3434-
// checks, i.e. fast and slow-paths?
3435-
phase->maybe_multiversion_for_auto_vectorization_runtime_checks(this, old_new);
3432+
if (!peel_only) {
3433+
// We are going to add pre-loop and post-loop (PreMainPost).
3434+
// But should we also multiversion for auto-vectorization speculative
3435+
// checks, i.e. fast and slow-paths?
3436+
// Note: Just PeelMainPost is not sufficient, as we could never find the
3437+
// multiversion_if again from the main loop: we need a nicely structured
3438+
// pre-loop, a peeled iteration cannot easily be parsed through.
3439+
phase->maybe_multiversion_for_auto_vectorization_runtime_checks(this, old_new);
3440+
}
34363441

34373442
phase->insert_pre_post_loops(this, old_new, peel_only);
34383443
}

‎src/hotspot/share/opto/loopnode.cpp

+4
Original file line numberDiff line numberDiff line change
@@ -4471,6 +4471,10 @@ void PhaseIdealLoop::eliminate_useless_multiversion_if() {
44714471
IfNode* multiversion_if = head->find_multiversion_if_from_multiversion_fast_main_loop();
44724472
if (multiversion_if != nullptr) {
44734473
useful_multiversioning_opaque_nodes.push(multiversion_if->in(1)->as_OpaqueMultiversioning());
4474+
} else {
4475+
// We could not find the multiversion_if, and would never find it again. Remove the
4476+
// multiversion marking for consistency.
4477+
head->set_no_multiversion();
44744478
}
44754479
}
44764480
}

‎src/hotspot/share/opto/loopopts.cpp

+1-1
Original file line numberDiff line numberDiff line change
@@ -4484,7 +4484,7 @@ PhaseIdealLoop::auto_vectorize(IdealLoopTree* lpt, VSharedData &vshared) {
44844484
return AutoVectorizeStatus::Success;
44854485
}
44864486

4487-
// Just before insert_pre_post_loops, we can multi-version the loop:
4487+
// Just before insert_pre_post_loops, we can multiversion the loop:
44884488
//
44894489
// multiversion_if
44904490
// | |
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,58 @@
1+
/*
2+
* Copyright (c) 2025, Oracle and/or its affiliates. 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.loopopts.superword;
25+
26+
/*
27+
* @test
28+
* @bug 8352587
29+
* @summary Test case where we used to Multiversion a PeelMainPost loop,
30+
* which is useless and triggered an assert later on.
31+
* @run main compiler.loopopts.superword.TestMultiversionWithPeelMainPost
32+
* @run main/othervm -XX:CompileCommand=compileonly,compiler.loopopts.superword.TestMultiversionWithPeelMainPost::test
33+
* -XX:-TieredCompilation -Xcomp
34+
* -XX:PerMethodTrapLimit=0
35+
* compiler.loopopts.superword.TestMultiversionWithPeelMainPost
36+
*/
37+
38+
public class TestMultiversionWithPeelMainPost {
39+
static byte byArr[] = new byte[2];
40+
41+
public static void main(String[] strArr) {
42+
test();
43+
}
44+
45+
static void test() {
46+
int x = 2;
47+
int i = 4;
48+
while (i > 0) {
49+
for (int j = 5; j < 56; j++) {
50+
byArr[1] = 3;
51+
}
52+
for (int j = 256; j > 3; j -= 2) {
53+
x += 3;
54+
}
55+
i--;
56+
}
57+
}
58+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,67 @@
1+
/*
2+
* Copyright (c) 2025, Oracle and/or its affiliates. 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.loopopts.superword;
25+
26+
import compiler.lib.ir_framework.*;
27+
28+
/*
29+
* @test
30+
* @bug 8352587
31+
* @summary IR test to ensure that PeelMainPost cases does not get Multiversioned.
32+
* @library /test/lib /
33+
* @run driver compiler.loopopts.superword.TestPeelMainPostNoMultiversioning
34+
*/
35+
36+
public class TestPeelMainPostNoMultiversioning {
37+
38+
public static void main(String[] args) {
39+
TestFramework framework = new TestFramework(TestPeelMainPostNoMultiversioning.class);
40+
// No traps means we cannot use the predicates version for SuperWord / AutoVectorization,
41+
// and instead use multiversioning directly.
42+
framework.addFlags("-XX:-TieredCompilation", "-XX:PerMethodTrapLimit=0");
43+
framework.setDefaultWarmup(0); // simulates Xcomp
44+
framework.start();
45+
}
46+
47+
public static long value = 1;
48+
public static long multiplicator = 3;
49+
50+
@Test
51+
@IR(counts = {".*multiversion.*", "= 0"},
52+
phase = CompilePhase.PHASEIDEALLOOP1)
53+
@IR(counts = {".*multiversion.*", "= 0"},
54+
phase = CompilePhase.PHASEIDEALLOOP_ITERATIONS)
55+
@IR(counts = {".*multiversion.*", "= 0"},
56+
phase = CompilePhase.PRINT_IDEAL)
57+
// We are checking it for a few phases, just to make sure we are not seeing multiversioning
58+
// at any point.
59+
public static void test() {
60+
long x = value;
61+
long y = multiplicator;
62+
for (int i = 0; i < 10_000; i++) {
63+
x *= y; // No memory load/store -> PeelMainPost
64+
}
65+
value = x;
66+
}
67+
}

1 commit comments

Comments
 (1)

openjdk-notifier[bot] commented on Mar 25, 2025

@openjdk-notifier[bot]
Please sign in to comment.