Skip to content

Commit 6ff4775

Browse files
eastigAndrew Haley
authored and
Andrew Haley
committedSep 7, 2022
8285487: AArch64: Do not generate unneeded trampolines for runtime calls
Reviewed-by: xliu, aph
1 parent d696104 commit 6ff4775

File tree

2 files changed

+136
-16
lines changed

2 files changed

+136
-16
lines changed
 

‎src/hotspot/cpu/aarch64/macroAssembler_aarch64.cpp

+32-16
Original file line numberDiff line numberDiff line change
@@ -849,6 +849,37 @@ void MacroAssembler::call_VM_helper(Register oop_result, address entry_point, in
849849
call_VM_base(oop_result, noreg, noreg, entry_point, number_of_arguments, check_exceptions);
850850
}
851851

852+
// Check the entry target is always reachable from any branch.
853+
static bool is_always_within_branch_range(Address entry) {
854+
const address target = entry.target();
855+
856+
if (!CodeCache::contains(target)) {
857+
// We always use trampolines for callees outside CodeCache.
858+
assert(entry.rspec().type() == relocInfo::runtime_call_type, "non-runtime call of an external target");
859+
return false;
860+
}
861+
862+
if (!MacroAssembler::far_branches()) {
863+
return true;
864+
}
865+
866+
if (entry.rspec().type() == relocInfo::runtime_call_type) {
867+
// Runtime calls are calls of a non-compiled method (stubs, adapters).
868+
// Non-compiled methods stay forever in CodeCache.
869+
// We check whether the longest possible branch is within the branch range.
870+
assert(CodeCache::find_blob(target) != NULL &&
871+
!CodeCache::find_blob(target)->is_compiled(),
872+
"runtime call of compiled method");
873+
const address right_longest_branch_start = CodeCache::high_bound() - NativeInstruction::instruction_size;
874+
const address left_longest_branch_start = CodeCache::low_bound();
875+
const bool is_reachable = Assembler::reachable_from_branch_at(left_longest_branch_start, target) &&
876+
Assembler::reachable_from_branch_at(right_longest_branch_start, target);
877+
return is_reachable;
878+
}
879+
880+
return false;
881+
}
882+
852883
// Maybe emit a call via a trampoline. If the code cache is small
853884
// trampolines won't be emitted.
854885
address MacroAssembler::trampoline_call(Address entry, CodeBuffer* cbuf) {
@@ -859,22 +890,7 @@ address MacroAssembler::trampoline_call(Address entry, CodeBuffer* cbuf) {
859890

860891
address target = entry.target();
861892

862-
// We might need a trampoline if branches are far.
863-
bool need_trampoline = far_branches();
864-
if (!need_trampoline && entry.rspec().type() == relocInfo::runtime_call_type && !CodeCache::contains(target)) {
865-
// If it is a runtime call of an address outside small CodeCache,
866-
// we need to check whether it is in range.
867-
assert(target < CodeCache::low_bound() || target >= CodeCache::high_bound(), "target is inside CodeCache");
868-
// Case 1: -------T-------L====CodeCache====H-------
869-
// ^-------longest branch---|
870-
// Case 2: -------L====CodeCache====H-------T-------
871-
// |-------longest branch ---^
872-
address longest_branch_start = (target < CodeCache::low_bound()) ? CodeCache::high_bound() - NativeInstruction::instruction_size
873-
: CodeCache::low_bound();
874-
need_trampoline = !reachable_from_branch_at(longest_branch_start, target);
875-
}
876-
877-
if (need_trampoline) {
893+
if (!is_always_within_branch_range(entry)) {
878894
if (!in_scratch_emit_size()) {
879895
// We don't want to emit a trampoline if C2 is generating dummy
880896
// code during its branch shortening phase.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,104 @@
1+
/*
2+
* Copyright Amazon.com Inc. 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+
25+
package compiler.c2.aarch64;
26+
27+
import java.util.ArrayList;
28+
import java.util.Iterator;
29+
import jdk.test.lib.process.OutputAnalyzer;
30+
import jdk.test.lib.process.ProcessTools;
31+
32+
/**
33+
* @test TestTrampoline
34+
* @summary Checks that trampolines to runtime code are not generated if they are not needed.
35+
* @bug 8285487
36+
* @library /test/lib
37+
*
38+
* @requires vm.flagless & os.arch=="aarch64" &
39+
* vm.debug == false & vm.compiler2.enabled
40+
*
41+
* @run driver compiler.c2.aarch64.TestTrampoline
42+
*/
43+
44+
public class TestTrampoline {
45+
private final static int ITERATIONS_TO_HEAT_LOOP = 20_000;
46+
47+
public static void main(String[] args) throws Exception {
48+
String testClassName = TestTrampoline.Test.class.getName();
49+
ArrayList<String> command = new ArrayList<String>();
50+
command.add("-XX:+UnlockDiagnosticVMOptions");
51+
command.add("-Xbatch");
52+
command.add("-XX:CompileCommand=print," + testClassName + "::" + "test");
53+
// ReservedCodeCacheSize=130M causes generation of trampolines.
54+
// As the non-nmethod segment is put between other two segments,
55+
// runtime calls will be within 128M range.
56+
// So there is no need for trampolines for runtime calls.
57+
command.add("-XX:ReservedCodeCacheSize=130M");
58+
command.add("-XX:+SegmentedCodeCache");
59+
command.add(testClassName);
60+
ProcessBuilder pb = ProcessTools.createJavaProcessBuilder(command);
61+
OutputAnalyzer analyzer = new OutputAnalyzer(pb.start());
62+
analyzer.shouldHaveExitValue(0);
63+
System.out.println(analyzer.getOutput());
64+
checkOutput(analyzer);
65+
}
66+
67+
private static String skipTo(Iterator<String> iter, String substring) {
68+
while (iter.hasNext()) {
69+
String nextLine = iter.next();
70+
if (nextLine.contains(substring)) {
71+
return nextLine;
72+
}
73+
}
74+
return null;
75+
}
76+
77+
private static void checkOutput(OutputAnalyzer output) {
78+
Iterator<String> iter = output.asLines().listIterator();
79+
80+
String match = skipTo(iter, "Compiled method (c2)");
81+
if (match == null || !match.contains("Test::test")) {
82+
throw new RuntimeException("Missing compiler output for the method 'test'");
83+
}
84+
85+
match = skipTo(iter, "[Stub Code]");
86+
if (match != null && skipTo(iter, "{trampoline_stub}") != null) {
87+
throw new RuntimeException("Found unexpected {trampoline_stub}");
88+
}
89+
}
90+
91+
static class Test {
92+
private static void test(String s, int i) {
93+
if (s.charAt(i) > 128)
94+
throw new RuntimeException();
95+
}
96+
97+
public static void main(String[] args) {
98+
String s = "Returns the char value at the specified index.";
99+
for (int i = 0; i < ITERATIONS_TO_HEAT_LOOP; ++i) {
100+
test(s, i % s.length());
101+
}
102+
}
103+
}
104+
}

0 commit comments

Comments
 (0)
Please sign in to comment.