8265360: several compiler/whitebox tests fail with "private compiler.whitebox.SimpleTestCaseHelper(int) must be compiled"
Reviewed-by: kvn, thartmann
This commit is contained in:
parent
ffca23a531
commit
4434c7df03
@ -191,7 +191,7 @@ CompileTask* CompilationPolicy::select_task_helper(CompileQueue* compile_queue)
|
||||
|
||||
// Simple methods are as good being compiled with C1 as C2.
|
||||
// Determine if a given method is such a case.
|
||||
bool CompilationPolicy::is_trivial(Method* method) {
|
||||
bool CompilationPolicy::is_trivial(const methodHandle& method) {
|
||||
if (method->is_accessor() ||
|
||||
method->is_constant_getter()) {
|
||||
return true;
|
||||
@ -245,7 +245,7 @@ public:
|
||||
}
|
||||
}
|
||||
|
||||
static bool apply(int i, int b, CompLevel cur_level, const methodHandle& method) {
|
||||
static bool apply(const methodHandle& method, CompLevel cur_level, int i, int b) {
|
||||
double k = 1;
|
||||
switch(cur_level) {
|
||||
case CompLevel_none:
|
||||
@ -285,7 +285,7 @@ public:
|
||||
}
|
||||
}
|
||||
|
||||
static bool apply(int i, int b, CompLevel cur_level, const methodHandle& method) {
|
||||
static bool apply(const methodHandle& method, CompLevel cur_level, int i, int b) {
|
||||
double k = 1;
|
||||
switch(cur_level) {
|
||||
case CompLevel_none:
|
||||
@ -596,7 +596,7 @@ CompLevel CompilationPolicy::initial_compile_level(const methodHandle& method) {
|
||||
}
|
||||
|
||||
// Set carry flags on the counters if necessary
|
||||
void CompilationPolicy::handle_counter_overflow(Method* method) {
|
||||
void CompilationPolicy::handle_counter_overflow(const methodHandle& method) {
|
||||
MethodCounters *mcs = method->method_counters();
|
||||
if (mcs != NULL) {
|
||||
mcs->invocation_counter()->set_carry_on_overflow();
|
||||
@ -619,21 +619,25 @@ CompileTask* CompilationPolicy::select_task(CompileQueue* compile_queue) {
|
||||
// Iterate through the queue and find a method with a maximum rate.
|
||||
for (CompileTask* task = compile_queue->first(); task != NULL;) {
|
||||
CompileTask* next_task = task->next();
|
||||
Method* method = task->method();
|
||||
// If a method was unloaded or has been stale for some time, remove it from the queue.
|
||||
// Blocking tasks and tasks submitted from whitebox API don't become stale
|
||||
if (task->is_unloaded() || (task->can_become_stale() && is_stale(t, TieredCompileTaskTimeout, method) && !is_old(method))) {
|
||||
if (!task->is_unloaded()) {
|
||||
if (PrintTieredEvents) {
|
||||
print_event(REMOVE_FROM_QUEUE, method, method, task->osr_bci(), (CompLevel) task->comp_level());
|
||||
}
|
||||
method->clear_queued_for_compilation();
|
||||
}
|
||||
if (task->is_unloaded()) {
|
||||
compile_queue->remove_and_mark_stale(task);
|
||||
task = next_task;
|
||||
continue;
|
||||
}
|
||||
update_rate(t, method);
|
||||
Method* method = task->method();
|
||||
methodHandle mh(Thread::current(), method);
|
||||
if (task->can_become_stale() && is_stale(t, TieredCompileTaskTimeout, mh) && !is_old(mh)) {
|
||||
if (PrintTieredEvents) {
|
||||
print_event(REMOVE_FROM_QUEUE, method, method, task->osr_bci(), (CompLevel) task->comp_level());
|
||||
}
|
||||
method->clear_queued_for_compilation();
|
||||
compile_queue->remove_and_mark_stale(task);
|
||||
task = next_task;
|
||||
continue;
|
||||
}
|
||||
update_rate(t, mh);
|
||||
if (max_task == NULL || compare_methods(method, max_method)) {
|
||||
// Select a method with the highest rate
|
||||
max_task = task;
|
||||
@ -711,9 +715,9 @@ nmethod* CompilationPolicy::event(const methodHandle& method, const methodHandle
|
||||
return NULL;
|
||||
}
|
||||
|
||||
handle_counter_overflow(method());
|
||||
handle_counter_overflow(method);
|
||||
if (method() != inlinee()) {
|
||||
handle_counter_overflow(inlinee());
|
||||
handle_counter_overflow(inlinee);
|
||||
}
|
||||
|
||||
if (bci == InvocationEntryBci) {
|
||||
@ -792,43 +796,43 @@ void CompilationPolicy::compile(const methodHandle& mh, int bci, CompLevel level
|
||||
print_event(COMPILE, mh(), mh(), bci, level);
|
||||
}
|
||||
int hot_count = (bci == InvocationEntryBci) ? mh->invocation_count() : mh->backedge_count();
|
||||
update_rate(nanos_to_millis(os::javaTimeNanos()), mh());
|
||||
update_rate(nanos_to_millis(os::javaTimeNanos()), mh);
|
||||
CompileBroker::compile_method(mh, bci, level, mh, hot_count, CompileTask::Reason_Tiered, THREAD);
|
||||
}
|
||||
}
|
||||
|
||||
// update_rate() is called from select_task() while holding a compile queue lock.
|
||||
void CompilationPolicy::update_rate(jlong t, Method* m) {
|
||||
void CompilationPolicy::update_rate(jlong t, const methodHandle& method) {
|
||||
// Skip update if counters are absent.
|
||||
// Can't allocate them since we are holding compile queue lock.
|
||||
if (m->method_counters() == NULL) return;
|
||||
if (method->method_counters() == NULL) return;
|
||||
|
||||
if (is_old(m)) {
|
||||
if (is_old(method)) {
|
||||
// We don't remove old methods from the queue,
|
||||
// so we can just zero the rate.
|
||||
m->set_rate(0);
|
||||
method->set_rate(0);
|
||||
return;
|
||||
}
|
||||
|
||||
// We don't update the rate if we've just came out of a safepoint.
|
||||
// delta_s is the time since last safepoint in milliseconds.
|
||||
jlong delta_s = t - SafepointTracing::end_of_last_safepoint_ms();
|
||||
jlong delta_t = t - (m->prev_time() != 0 ? m->prev_time() : start_time()); // milliseconds since the last measurement
|
||||
jlong delta_t = t - (method->prev_time() != 0 ? method->prev_time() : start_time()); // milliseconds since the last measurement
|
||||
// How many events were there since the last time?
|
||||
int event_count = m->invocation_count() + m->backedge_count();
|
||||
int delta_e = event_count - m->prev_event_count();
|
||||
int event_count = method->invocation_count() + method->backedge_count();
|
||||
int delta_e = event_count - method->prev_event_count();
|
||||
|
||||
// We should be running for at least 1ms.
|
||||
if (delta_s >= TieredRateUpdateMinTime) {
|
||||
// And we must've taken the previous point at least 1ms before.
|
||||
if (delta_t >= TieredRateUpdateMinTime && delta_e > 0) {
|
||||
m->set_prev_time(t);
|
||||
m->set_prev_event_count(event_count);
|
||||
m->set_rate((float)delta_e / (float)delta_t); // Rate is events per millisecond
|
||||
method->set_prev_time(t);
|
||||
method->set_prev_event_count(event_count);
|
||||
method->set_rate((float)delta_e / (float)delta_t); // Rate is events per millisecond
|
||||
} else {
|
||||
if (delta_t > TieredRateUpdateMaxTime && delta_e == 0) {
|
||||
// If nothing happened for 25ms, zero the rate. Don't modify prev values.
|
||||
m->set_rate(0);
|
||||
method->set_rate(0);
|
||||
}
|
||||
}
|
||||
}
|
||||
@ -836,12 +840,12 @@ void CompilationPolicy::update_rate(jlong t, Method* m) {
|
||||
|
||||
// Check if this method has been stale for a given number of milliseconds.
|
||||
// See select_task().
|
||||
bool CompilationPolicy::is_stale(jlong t, jlong timeout, Method* m) {
|
||||
bool CompilationPolicy::is_stale(jlong t, jlong timeout, const methodHandle& method) {
|
||||
jlong delta_s = t - SafepointTracing::end_of_last_safepoint_ms();
|
||||
jlong delta_t = t - m->prev_time();
|
||||
jlong delta_t = t - method->prev_time();
|
||||
if (delta_t > timeout && delta_s > timeout) {
|
||||
int event_count = m->invocation_count() + m->backedge_count();
|
||||
int delta_e = event_count - m->prev_event_count();
|
||||
int event_count = method->invocation_count() + method->backedge_count();
|
||||
int delta_e = event_count - method->prev_event_count();
|
||||
// Return true if there were no events.
|
||||
return delta_e == 0;
|
||||
}
|
||||
@ -850,13 +854,16 @@ bool CompilationPolicy::is_stale(jlong t, jlong timeout, Method* m) {
|
||||
|
||||
// We don't remove old methods from the compile queue even if they have
|
||||
// very low activity. See select_task().
|
||||
bool CompilationPolicy::is_old(Method* method) {
|
||||
return method->invocation_count() > 50000 || method->backedge_count() > 500000;
|
||||
bool CompilationPolicy::is_old(const methodHandle& method) {
|
||||
int i = method->invocation_count();
|
||||
int b = method->backedge_count();
|
||||
double k = TieredOldPercentage / 100.0;
|
||||
|
||||
return CallPredicate::apply_scaled(method, CompLevel_none, i, b, k) || LoopPredicate::apply_scaled(method, CompLevel_none, i, b, k);
|
||||
}
|
||||
|
||||
double CompilationPolicy::weight(Method* method) {
|
||||
return (double)(method->rate() + 1) *
|
||||
(method->invocation_count() + 1) * (method->backedge_count() + 1);
|
||||
return (double)(method->rate() + 1) * (method->invocation_count() + 1) * (method->backedge_count() + 1);
|
||||
}
|
||||
|
||||
// Apply heuristics and return true if x should be compiled before y
|
||||
@ -909,13 +916,16 @@ bool CompilationPolicy::should_create_mdo(const methodHandle& method, CompLevel
|
||||
if (cur_level != CompLevel_none || force_comp_at_level_simple(method) || CompilationModeFlag::quick_only() || !ProfileInterpreter) {
|
||||
return false;
|
||||
}
|
||||
if (is_old(method)) {
|
||||
return true;
|
||||
}
|
||||
int i = method->invocation_count();
|
||||
int b = method->backedge_count();
|
||||
double k = Tier0ProfilingStartPercentage / 100.0;
|
||||
|
||||
// If the top level compiler is not keeping up, delay profiling.
|
||||
if (CompileBroker::queue_size(CompLevel_full_optimization) <= Tier0Delay * compiler_count(CompLevel_full_optimization)) {
|
||||
return CallPredicate::apply_scaled(method, CompLevel_full_profile, i, b, k) || LoopPredicate::apply_scaled(method, CompLevel_full_profile, i, b, k);
|
||||
return CallPredicate::apply_scaled(method, CompLevel_none, i, b, k) || LoopPredicate::apply_scaled(method, CompLevel_none, i, b, k);
|
||||
}
|
||||
return false;
|
||||
}
|
||||
@ -1005,7 +1015,7 @@ CompLevel CompilationPolicy::common(const methodHandle& method, CompLevel cur_le
|
||||
if (force_comp_at_level_simple(method)) {
|
||||
next_level = CompLevel_simple;
|
||||
} else {
|
||||
if (is_trivial(method())) {
|
||||
if (is_trivial(method)) {
|
||||
next_level = CompilationModeFlag::disable_intermediate() ? CompLevel_full_optimization : CompLevel_simple;
|
||||
} else {
|
||||
switch(cur_level) {
|
||||
@ -1014,7 +1024,7 @@ CompLevel CompilationPolicy::common(const methodHandle& method, CompLevel cur_le
|
||||
// If we were at full profile level, would we switch to full opt?
|
||||
if (common<Predicate>(method, CompLevel_full_profile, disable_feedback) == CompLevel_full_optimization) {
|
||||
next_level = CompLevel_full_optimization;
|
||||
} else if (!CompilationModeFlag::disable_intermediate() && Predicate::apply(i, b, cur_level, method)) {
|
||||
} else if (!CompilationModeFlag::disable_intermediate() && Predicate::apply(method, cur_level, i, b)) {
|
||||
// C1-generated fully profiled code is about 30% slower than the limited profile
|
||||
// code that has only invocation and backedge counters. The observation is that
|
||||
// if C2 queue is large enough we can spend too much time in the fully profiled code
|
||||
@ -1040,7 +1050,7 @@ CompLevel CompilationPolicy::common(const methodHandle& method, CompLevel cur_le
|
||||
if (mdo->would_profile()) {
|
||||
if (disable_feedback || (CompileBroker::queue_size(CompLevel_full_optimization) <=
|
||||
Tier3DelayOff * compiler_count(CompLevel_full_optimization) &&
|
||||
Predicate::apply(i, b, cur_level, method))) {
|
||||
Predicate::apply(method, cur_level, i, b))) {
|
||||
next_level = CompLevel_full_profile;
|
||||
}
|
||||
} else {
|
||||
@ -1050,7 +1060,7 @@ CompLevel CompilationPolicy::common(const methodHandle& method, CompLevel cur_le
|
||||
// If there is no MDO we need to profile
|
||||
if (disable_feedback || (CompileBroker::queue_size(CompLevel_full_optimization) <=
|
||||
Tier3DelayOff * compiler_count(CompLevel_full_optimization) &&
|
||||
Predicate::apply(i, b, cur_level, method))) {
|
||||
Predicate::apply(method, cur_level, i, b))) {
|
||||
next_level = CompLevel_full_profile;
|
||||
}
|
||||
}
|
||||
@ -1063,7 +1073,7 @@ CompLevel CompilationPolicy::common(const methodHandle& method, CompLevel cur_le
|
||||
if (mdo->would_profile() || CompilationModeFlag::disable_intermediate()) {
|
||||
int mdo_i = mdo->invocation_count_delta();
|
||||
int mdo_b = mdo->backedge_count_delta();
|
||||
if (Predicate::apply(mdo_i, mdo_b, cur_level, method)) {
|
||||
if (Predicate::apply(method, cur_level, mdo_i, mdo_b)) {
|
||||
next_level = CompLevel_full_optimization;
|
||||
}
|
||||
} else {
|
||||
@ -1083,7 +1093,7 @@ CompLevel CompilationPolicy::common(const methodHandle& method, CompLevel cur_le
|
||||
// Determine if a method should be compiled with a normal entry point at a different level.
|
||||
CompLevel CompilationPolicy::call_event(const methodHandle& method, CompLevel cur_level, Thread* thread) {
|
||||
CompLevel osr_level = MIN2((CompLevel) method->highest_osr_comp_level(), common<LoopPredicate>(method, cur_level, true));
|
||||
CompLevel next_level = common<CallPredicate>(method, cur_level);
|
||||
CompLevel next_level = common<CallPredicate>(method, cur_level, is_old(method));
|
||||
|
||||
// If OSR method level is greater than the regular method level, the levels should be
|
||||
// equalized by raising the regular method level in order to avoid OSRs during each
|
||||
|
@ -169,7 +169,7 @@ class CompilationPolicy : AllStatic {
|
||||
static double _increase_threshold_at_ratio;
|
||||
|
||||
// Set carry flags in the counters (in Method* and MDO).
|
||||
inline static void handle_counter_overflow(Method* method);
|
||||
inline static void handle_counter_overflow(const methodHandle& method);
|
||||
// Verify that a level is consistent with the compilation mode
|
||||
static bool verify_level(CompLevel level);
|
||||
// Clamp the request level according to various constraints.
|
||||
@ -188,17 +188,17 @@ class CompilationPolicy : AllStatic {
|
||||
// Has a method been long around?
|
||||
// We don't remove old methods from the compile queue even if they have
|
||||
// very low activity (see select_task()).
|
||||
inline static bool is_old(Method* method);
|
||||
inline static bool is_old(const methodHandle& method);
|
||||
// Was a given method inactive for a given number of milliseconds.
|
||||
// If it is, we would remove it from the queue (see select_task()).
|
||||
inline static bool is_stale(jlong t, jlong timeout, Method* m);
|
||||
inline static bool is_stale(jlong t, jlong timeout, const methodHandle& method);
|
||||
// Compute the weight of the method for the compilation scheduling
|
||||
inline static double weight(Method* method);
|
||||
// Apply heuristics and return true if x should be compiled before y
|
||||
inline static bool compare_methods(Method* x, Method* y);
|
||||
// Compute event rate for a given method. The rate is the number of event (invocations + backedges)
|
||||
// per millisecond.
|
||||
inline static void update_rate(jlong t, Method* m);
|
||||
inline static void update_rate(jlong t, const methodHandle& method);
|
||||
// Compute threshold scaling coefficient
|
||||
inline static double threshold_scale(CompLevel level, int feedback_k);
|
||||
// If a method is old enough and is still in the interpreter we would want to
|
||||
@ -219,7 +219,7 @@ class CompilationPolicy : AllStatic {
|
||||
static void compile(const methodHandle& mh, int bci, CompLevel level, TRAPS);
|
||||
// Simple methods are as good being compiled with C1 as C2.
|
||||
// This function tells if it's such a function.
|
||||
inline static bool is_trivial(Method* method);
|
||||
inline static bool is_trivial(const methodHandle& method);
|
||||
// Force method to be compiled at CompLevel_simple?
|
||||
inline static bool force_comp_at_level_simple(const methodHandle& method);
|
||||
|
||||
|
@ -267,12 +267,11 @@ void CompilerConfig::set_legacy_emulation_flags() {
|
||||
FLAG_SET_ERGO(Tier0BackedgeNotifyFreqLog, MAX2<intx>(10, osr_threshold_log));
|
||||
}
|
||||
// Adjust the tiered policy flags to approximate the legacy behavior.
|
||||
if (CompilerConfig::is_c1_only()) {
|
||||
FLAG_SET_ERGO(Tier3InvocationThreshold, threshold);
|
||||
FLAG_SET_ERGO(Tier3MinInvocationThreshold, threshold);
|
||||
FLAG_SET_ERGO(Tier3CompileThreshold, threshold);
|
||||
FLAG_SET_ERGO(Tier3BackEdgeThreshold, osr_threshold);
|
||||
} else {
|
||||
FLAG_SET_ERGO(Tier3InvocationThreshold, threshold);
|
||||
FLAG_SET_ERGO(Tier3MinInvocationThreshold, threshold);
|
||||
FLAG_SET_ERGO(Tier3CompileThreshold, threshold);
|
||||
FLAG_SET_ERGO(Tier3BackEdgeThreshold, osr_threshold);
|
||||
if (CompilerConfig::is_c2_or_jvmci_compiler_only()) {
|
||||
FLAG_SET_ERGO(Tier4InvocationThreshold, threshold);
|
||||
FLAG_SET_ERGO(Tier4MinInvocationThreshold, threshold);
|
||||
FLAG_SET_ERGO(Tier4CompileThreshold, threshold);
|
||||
@ -338,6 +337,20 @@ void CompilerConfig::set_compilation_policy_flags() {
|
||||
if (FLAG_IS_DEFAULT(Tier4BackEdgeThreshold)) {
|
||||
FLAG_SET_DEFAULT(Tier4BackEdgeThreshold, 15000);
|
||||
}
|
||||
|
||||
if (FLAG_IS_DEFAULT(Tier3InvocationThreshold)) {
|
||||
FLAG_SET_DEFAULT(Tier3InvocationThreshold, Tier4InvocationThreshold);
|
||||
}
|
||||
if (FLAG_IS_DEFAULT(Tier3MinInvocationThreshold)) {
|
||||
FLAG_SET_DEFAULT(Tier3MinInvocationThreshold, Tier4MinInvocationThreshold);
|
||||
}
|
||||
if (FLAG_IS_DEFAULT(Tier3CompileThreshold)) {
|
||||
FLAG_SET_DEFAULT(Tier3CompileThreshold, Tier4CompileThreshold);
|
||||
}
|
||||
if (FLAG_IS_DEFAULT(Tier3BackEdgeThreshold)) {
|
||||
FLAG_SET_DEFAULT(Tier3BackEdgeThreshold, Tier4BackEdgeThreshold);
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
// Scale tiered compilation thresholds.
|
||||
|
@ -217,6 +217,12 @@
|
||||
"do not start profiling in the interpreter") \
|
||||
range(0, max_jint) \
|
||||
\
|
||||
product(intx, TieredOldPercentage, 1000, DIAGNOSTIC, \
|
||||
"Percentage over tier 3 thresholds after which a method is " \
|
||||
"considered old (turns off parts of prioritization based on " \
|
||||
"compile queue length)") \
|
||||
range(0, max_jint) \
|
||||
\
|
||||
product(intx, Tier3DelayOn, 5, \
|
||||
"If C2 queue size grows over this amount per compiler thread " \
|
||||
"stop compiling at tier 3 and start compiling at tier 2") \
|
||||
@ -248,8 +254,7 @@
|
||||
\
|
||||
product(intx, Tier0ProfilingStartPercentage, 200, \
|
||||
"Start profiling in interpreter if the counters exceed the " \
|
||||
"specified percentage of tier 3 thresholds (tier 4 thresholds " \
|
||||
"with CompilationMode=high-only|high-only-quick-internal)") \
|
||||
"specified percentage of tier 3 thresholds") \
|
||||
range(0, max_jint) \
|
||||
\
|
||||
product(uintx, IncreaseFirstTierCompileThresholdAt, 50, \
|
||||
|
@ -67,10 +67,6 @@ compiler/rtm/print/TestPrintPreciseRTMLockingStatistics.java 8183263 generic-x64
|
||||
|
||||
compiler/c2/Test8004741.java 8235801 generic-all
|
||||
|
||||
compiler/whitebox/ClearMethodStateTest.java 8265360 macosx-all
|
||||
compiler/whitebox/EnqueueMethodForCompilationTest.java 8265360 macosx-all
|
||||
compiler/whitebox/MakeMethodNotCompilableTest.java 8265360 macosx-all
|
||||
|
||||
compiler/codecache/jmx/PoolsIndependenceTest.java 8264632 macosx-all
|
||||
compiler/codecache/TestStressCodeBuffers.java 8272094 generic-aarch64
|
||||
|
||||
|
Loading…
x
Reference in New Issue
Block a user