8226705: [REDO] Deoptimize with handshakes
Reviewed-by: eosterlund, dcubed, dlong, pchilanomate
This commit is contained in:
parent
336b741b1c
commit
35a9f6864a
@ -38,6 +38,7 @@
|
|||||||
#include "memory/universe.hpp"
|
#include "memory/universe.hpp"
|
||||||
#include "oops/compressedOops.hpp"
|
#include "oops/compressedOops.hpp"
|
||||||
#include "oops/method.inline.hpp"
|
#include "oops/method.inline.hpp"
|
||||||
|
#include "runtime/deoptimization.hpp"
|
||||||
#include "runtime/handles.inline.hpp"
|
#include "runtime/handles.inline.hpp"
|
||||||
#include "runtime/os.hpp"
|
#include "runtime/os.hpp"
|
||||||
#include "runtime/safepointVerifiers.hpp"
|
#include "runtime/safepointVerifiers.hpp"
|
||||||
@ -351,7 +352,10 @@ void AOTCodeHeap::publish_aot(const methodHandle& mh, AOTMethodData* method_data
|
|||||||
#ifdef TIERED
|
#ifdef TIERED
|
||||||
mh->set_aot_code(aot);
|
mh->set_aot_code(aot);
|
||||||
#endif
|
#endif
|
||||||
Method::set_code(mh, aot);
|
{
|
||||||
|
MutexLocker pl(CompiledMethod_lock, Mutex::_no_safepoint_check_flag);
|
||||||
|
Method::set_code(mh, aot);
|
||||||
|
}
|
||||||
if (PrintAOT || (PrintCompilation && PrintAOT)) {
|
if (PrintAOT || (PrintCompilation && PrintAOT)) {
|
||||||
PauseNoSafepointVerifier pnsv(&nsv); // aot code is registered already
|
PauseNoSafepointVerifier pnsv(&nsv); // aot code is registered already
|
||||||
aot->print_on(tty, NULL);
|
aot->print_on(tty, NULL);
|
||||||
@ -731,8 +735,7 @@ void AOTCodeHeap::sweep_dependent_methods(int* indexes, int methods_cnt) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (marked > 0) {
|
if (marked > 0) {
|
||||||
VM_Deoptimize op;
|
Deoptimization::deoptimize_all_marked();
|
||||||
VMThread::execute(&op);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -165,7 +165,7 @@ bool AOTCompiledMethod::make_not_entrant_helper(int new_state) {
|
|||||||
|
|
||||||
{
|
{
|
||||||
// Enter critical section. Does not block for safepoint.
|
// Enter critical section. Does not block for safepoint.
|
||||||
MutexLocker pl(Patching_lock, Mutex::_no_safepoint_check_flag);
|
MutexLocker pl(CompiledMethod_lock, Mutex::_no_safepoint_check_flag);
|
||||||
|
|
||||||
if (*_state_adr == new_state) {
|
if (*_state_adr == new_state) {
|
||||||
// another thread already performed this transition so nothing
|
// another thread already performed this transition so nothing
|
||||||
@ -188,12 +188,10 @@ bool AOTCompiledMethod::make_not_entrant_helper(int new_state) {
|
|||||||
#endif
|
#endif
|
||||||
|
|
||||||
// Remove AOTCompiledMethod from method.
|
// Remove AOTCompiledMethod from method.
|
||||||
if (method() != NULL && (method()->code() == this ||
|
if (method() != NULL) {
|
||||||
method()->from_compiled_entry() == verified_entry_point())) {
|
method()->unlink_code(this);
|
||||||
HandleMark hm;
|
|
||||||
method()->clear_code(false /* already owns Patching_lock */);
|
|
||||||
}
|
}
|
||||||
} // leave critical region under Patching_lock
|
} // leave critical region under CompiledMethod_lock
|
||||||
|
|
||||||
|
|
||||||
if (TraceCreateZombies) {
|
if (TraceCreateZombies) {
|
||||||
@ -208,7 +206,6 @@ bool AOTCompiledMethod::make_not_entrant_helper(int new_state) {
|
|||||||
#ifdef TIERED
|
#ifdef TIERED
|
||||||
bool AOTCompiledMethod::make_entrant() {
|
bool AOTCompiledMethod::make_entrant() {
|
||||||
assert(!method()->is_old(), "reviving evolved method!");
|
assert(!method()->is_old(), "reviving evolved method!");
|
||||||
assert(*_state_adr != not_entrant, "%s", method()->has_aot_code() ? "has_aot_code() not cleared" : "caller didn't check has_aot_code()");
|
|
||||||
|
|
||||||
// Make sure the method is not flushed in case of a safepoint in code below.
|
// Make sure the method is not flushed in case of a safepoint in code below.
|
||||||
methodHandle the_method(method());
|
methodHandle the_method(method());
|
||||||
@ -216,9 +213,9 @@ bool AOTCompiledMethod::make_entrant() {
|
|||||||
|
|
||||||
{
|
{
|
||||||
// Enter critical section. Does not block for safepoint.
|
// Enter critical section. Does not block for safepoint.
|
||||||
MutexLocker pl(Patching_lock, Mutex::_no_safepoint_check_flag);
|
MutexLocker pl(CompiledMethod_lock, Mutex::_no_safepoint_check_flag);
|
||||||
|
|
||||||
if (*_state_adr == in_use) {
|
if (*_state_adr == in_use || *_state_adr == not_entrant) {
|
||||||
// another thread already performed this transition so nothing
|
// another thread already performed this transition so nothing
|
||||||
// to do, but return false to indicate this.
|
// to do, but return false to indicate this.
|
||||||
return false;
|
return false;
|
||||||
@ -230,7 +227,7 @@ bool AOTCompiledMethod::make_entrant() {
|
|||||||
|
|
||||||
// Log the transition once
|
// Log the transition once
|
||||||
log_state_change();
|
log_state_change();
|
||||||
} // leave critical region under Patching_lock
|
} // leave critical region under CompiledMethod_lock
|
||||||
|
|
||||||
|
|
||||||
if (TraceCreateZombies) {
|
if (TraceCreateZombies) {
|
||||||
|
@ -1072,7 +1072,10 @@ void ciEnv::register_method(ciMethod* target,
|
|||||||
task()->comp_level(), method_name);
|
task()->comp_level(), method_name);
|
||||||
}
|
}
|
||||||
// Allow the code to be executed
|
// Allow the code to be executed
|
||||||
method->set_code(method, nm);
|
MutexLocker ml(CompiledMethod_lock, Mutex::_no_safepoint_check_flag);
|
||||||
|
if (nm->make_in_use()) {
|
||||||
|
method->set_code(method, nm);
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
LogTarget(Info, nmethod, install) lt;
|
LogTarget(Info, nmethod, install) lt;
|
||||||
if (lt.is_enabled()) {
|
if (lt.is_enabled()) {
|
||||||
@ -1081,9 +1084,11 @@ void ciEnv::register_method(ciMethod* target,
|
|||||||
lt.print("Installing osr method (%d) %s @ %d",
|
lt.print("Installing osr method (%d) %s @ %d",
|
||||||
task()->comp_level(), method_name, entry_bci);
|
task()->comp_level(), method_name, entry_bci);
|
||||||
}
|
}
|
||||||
method->method_holder()->add_osr_nmethod(nm);
|
MutexLocker ml(CompiledMethod_lock, Mutex::_no_safepoint_check_flag);
|
||||||
|
if (nm->make_in_use()) {
|
||||||
|
method->method_holder()->add_osr_nmethod(nm);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
nm->make_in_use();
|
|
||||||
}
|
}
|
||||||
} // safepoints are allowed again
|
} // safepoints are allowed again
|
||||||
|
|
||||||
|
@ -1143,28 +1143,17 @@ void CodeCache::flush_evol_dependents() {
|
|||||||
|
|
||||||
// At least one nmethod has been marked for deoptimization
|
// At least one nmethod has been marked for deoptimization
|
||||||
|
|
||||||
// All this already happens inside a VM_Operation, so we'll do all the work here.
|
Deoptimization::deoptimize_all_marked();
|
||||||
// Stuff copied from VM_Deoptimize and modified slightly.
|
|
||||||
|
|
||||||
// We do not want any GCs to happen while we are in the middle of this VM operation
|
|
||||||
ResourceMark rm;
|
|
||||||
DeoptimizationMarker dm;
|
|
||||||
|
|
||||||
// Deoptimize all activations depending on marked nmethods
|
|
||||||
Deoptimization::deoptimize_dependents();
|
|
||||||
|
|
||||||
// Make the dependent methods not entrant
|
|
||||||
make_marked_nmethods_not_entrant();
|
|
||||||
}
|
}
|
||||||
#endif // INCLUDE_JVMTI
|
#endif // INCLUDE_JVMTI
|
||||||
|
|
||||||
// Deoptimize all methods
|
// Mark methods for deopt (if safe or possible).
|
||||||
void CodeCache::mark_all_nmethods_for_deoptimization() {
|
void CodeCache::mark_all_nmethods_for_deoptimization() {
|
||||||
MutexLocker mu(CodeCache_lock, Mutex::_no_safepoint_check_flag);
|
MutexLocker mu(CodeCache_lock, Mutex::_no_safepoint_check_flag);
|
||||||
CompiledMethodIterator iter(CompiledMethodIterator::only_alive_and_not_unloading);
|
CompiledMethodIterator iter(CompiledMethodIterator::only_alive_and_not_unloading);
|
||||||
while(iter.next()) {
|
while(iter.next()) {
|
||||||
CompiledMethod* nm = iter.method();
|
CompiledMethod* nm = iter.method();
|
||||||
if (!nm->method()->is_method_handle_intrinsic()) {
|
if (!nm->is_native_method()) {
|
||||||
nm->mark_for_deoptimization();
|
nm->mark_for_deoptimization();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -1192,7 +1181,7 @@ void CodeCache::make_marked_nmethods_not_entrant() {
|
|||||||
CompiledMethodIterator iter(CompiledMethodIterator::only_alive_and_not_unloading);
|
CompiledMethodIterator iter(CompiledMethodIterator::only_alive_and_not_unloading);
|
||||||
while(iter.next()) {
|
while(iter.next()) {
|
||||||
CompiledMethod* nm = iter.method();
|
CompiledMethod* nm = iter.method();
|
||||||
if (nm->is_marked_for_deoptimization() && !nm->is_not_entrant()) {
|
if (nm->is_marked_for_deoptimization()) {
|
||||||
nm->make_not_entrant();
|
nm->make_not_entrant();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -1204,17 +1193,12 @@ void CodeCache::flush_dependents_on(InstanceKlass* dependee) {
|
|||||||
|
|
||||||
if (number_of_nmethods_with_dependencies() == 0) return;
|
if (number_of_nmethods_with_dependencies() == 0) return;
|
||||||
|
|
||||||
// CodeCache can only be updated by a thread_in_VM and they will all be
|
|
||||||
// stopped during the safepoint so CodeCache will be safe to update without
|
|
||||||
// holding the CodeCache_lock.
|
|
||||||
|
|
||||||
KlassDepChange changes(dependee);
|
KlassDepChange changes(dependee);
|
||||||
|
|
||||||
// Compute the dependent nmethods
|
// Compute the dependent nmethods
|
||||||
if (mark_for_deoptimization(changes) > 0) {
|
if (mark_for_deoptimization(changes) > 0) {
|
||||||
// At least one nmethod has been marked for deoptimization
|
// At least one nmethod has been marked for deoptimization
|
||||||
VM_Deoptimize op;
|
Deoptimization::deoptimize_all_marked();
|
||||||
VMThread::execute(&op);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1223,26 +1207,9 @@ void CodeCache::flush_dependents_on_method(const methodHandle& m_h) {
|
|||||||
// --- Compile_lock is not held. However we are at a safepoint.
|
// --- Compile_lock is not held. However we are at a safepoint.
|
||||||
assert_locked_or_safepoint(Compile_lock);
|
assert_locked_or_safepoint(Compile_lock);
|
||||||
|
|
||||||
// CodeCache can only be updated by a thread_in_VM and they will all be
|
|
||||||
// stopped dring the safepoint so CodeCache will be safe to update without
|
|
||||||
// holding the CodeCache_lock.
|
|
||||||
|
|
||||||
// Compute the dependent nmethods
|
// Compute the dependent nmethods
|
||||||
if (mark_for_deoptimization(m_h()) > 0) {
|
if (mark_for_deoptimization(m_h()) > 0) {
|
||||||
// At least one nmethod has been marked for deoptimization
|
Deoptimization::deoptimize_all_marked();
|
||||||
|
|
||||||
// All this already happens inside a VM_Operation, so we'll do all the work here.
|
|
||||||
// Stuff copied from VM_Deoptimize and modified slightly.
|
|
||||||
|
|
||||||
// We do not want any GCs to happen while we are in the middle of this VM operation
|
|
||||||
ResourceMark rm;
|
|
||||||
DeoptimizationMarker dm;
|
|
||||||
|
|
||||||
// Deoptimize all activations depending on marked nmethods
|
|
||||||
Deoptimization::deoptimize_dependents();
|
|
||||||
|
|
||||||
// Make the dependent methods not entrant
|
|
||||||
make_marked_nmethods_not_entrant();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -103,6 +103,13 @@ const char* CompiledMethod::state() const {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
//-----------------------------------------------------------------------------
|
||||||
|
void CompiledMethod::mark_for_deoptimization(bool inc_recompile_counts) {
|
||||||
|
MutexLocker ml(CompiledMethod_lock->owned_by_self() ? NULL : CompiledMethod_lock,
|
||||||
|
Mutex::_no_safepoint_check_flag);
|
||||||
|
_mark_for_deoptimization_status = (inc_recompile_counts ? deoptimize : deoptimize_noupdate);
|
||||||
|
}
|
||||||
|
|
||||||
//-----------------------------------------------------------------------------
|
//-----------------------------------------------------------------------------
|
||||||
|
|
||||||
ExceptionCache* CompiledMethod::exception_cache_acquire() const {
|
ExceptionCache* CompiledMethod::exception_cache_acquire() const {
|
||||||
|
@ -244,10 +244,9 @@ public:
|
|||||||
bool is_at_poll_return(address pc);
|
bool is_at_poll_return(address pc);
|
||||||
bool is_at_poll_or_poll_return(address pc);
|
bool is_at_poll_or_poll_return(address pc);
|
||||||
|
|
||||||
bool is_marked_for_deoptimization() const { return _mark_for_deoptimization_status != not_marked; }
|
bool is_marked_for_deoptimization() const { return _mark_for_deoptimization_status != not_marked; }
|
||||||
void mark_for_deoptimization(bool inc_recompile_counts = true) {
|
void mark_for_deoptimization(bool inc_recompile_counts = true);
|
||||||
_mark_for_deoptimization_status = (inc_recompile_counts ? deoptimize : deoptimize_noupdate);
|
|
||||||
}
|
|
||||||
bool update_recompile_counts() const {
|
bool update_recompile_counts() const {
|
||||||
// Update recompile counts when either the update is explicitly requested (deoptimize)
|
// Update recompile counts when either the update is explicitly requested (deoptimize)
|
||||||
// or the nmethod is not marked for deoptimization at all (not_marked).
|
// or the nmethod is not marked for deoptimization at all (not_marked).
|
||||||
|
@ -50,6 +50,7 @@
|
|||||||
#include "oops/oop.inline.hpp"
|
#include "oops/oop.inline.hpp"
|
||||||
#include "prims/jvmtiImpl.hpp"
|
#include "prims/jvmtiImpl.hpp"
|
||||||
#include "runtime/atomic.hpp"
|
#include "runtime/atomic.hpp"
|
||||||
|
#include "runtime/deoptimization.hpp"
|
||||||
#include "runtime/flags/flagSetting.hpp"
|
#include "runtime/flags/flagSetting.hpp"
|
||||||
#include "runtime/frame.inline.hpp"
|
#include "runtime/frame.inline.hpp"
|
||||||
#include "runtime/handles.inline.hpp"
|
#include "runtime/handles.inline.hpp"
|
||||||
@ -476,7 +477,6 @@ nmethod* nmethod::new_native_nmethod(const methodHandle& method,
|
|||||||
debug_only(nm->verify();) // might block
|
debug_only(nm->verify();) // might block
|
||||||
|
|
||||||
nm->log_new_nmethod();
|
nm->log_new_nmethod();
|
||||||
nm->make_in_use();
|
|
||||||
}
|
}
|
||||||
return nm;
|
return nm;
|
||||||
}
|
}
|
||||||
@ -1138,6 +1138,11 @@ void nmethod::inc_decompile_count() {
|
|||||||
|
|
||||||
bool nmethod::try_transition(int new_state_int) {
|
bool nmethod::try_transition(int new_state_int) {
|
||||||
signed char new_state = new_state_int;
|
signed char new_state = new_state_int;
|
||||||
|
#ifdef DEBUG
|
||||||
|
if (new_state != unloaded) {
|
||||||
|
assert_lock_strong(CompiledMethod_lock);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
for (;;) {
|
for (;;) {
|
||||||
signed char old_state = Atomic::load(&_state);
|
signed char old_state = Atomic::load(&_state);
|
||||||
if (old_state >= new_state) {
|
if (old_state >= new_state) {
|
||||||
@ -1193,11 +1198,7 @@ void nmethod::make_unloaded() {
|
|||||||
// have the Method* live here, in case we unload the nmethod because
|
// have the Method* live here, in case we unload the nmethod because
|
||||||
// it is pointing to some oop (other than the Method*) being unloaded.
|
// it is pointing to some oop (other than the Method*) being unloaded.
|
||||||
if (_method != NULL) {
|
if (_method != NULL) {
|
||||||
// OSR methods point to the Method*, but the Method* does not
|
_method->unlink_code(this);
|
||||||
// point back!
|
|
||||||
if (_method->code() == this) {
|
|
||||||
_method->clear_code(); // Break a cycle
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Make the class unloaded - i.e., change state and notify sweeper
|
// Make the class unloaded - i.e., change state and notify sweeper
|
||||||
@ -1281,16 +1282,9 @@ void nmethod::log_state_change() const {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void nmethod::unlink_from_method(bool acquire_lock) {
|
void nmethod::unlink_from_method() {
|
||||||
// We need to check if both the _code and _from_compiled_code_entry_point
|
if (method() != NULL) {
|
||||||
// refer to this nmethod because there is a race in setting these two fields
|
method()->unlink_code(this);
|
||||||
// in Method* as seen in bugid 4947125.
|
|
||||||
// If the vep() points to the zombie nmethod, the memory for the nmethod
|
|
||||||
// could be flushed and the compiler and vtable stubs could still call
|
|
||||||
// through it.
|
|
||||||
if (method() != NULL && (method()->code() == this ||
|
|
||||||
method()->from_compiled_entry() == verified_entry_point())) {
|
|
||||||
method()->clear_code(acquire_lock);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1317,24 +1311,24 @@ bool nmethod::make_not_entrant_or_zombie(int state) {
|
|||||||
|
|
||||||
// during patching, depending on the nmethod state we must notify the GC that
|
// during patching, depending on the nmethod state we must notify the GC that
|
||||||
// code has been unloaded, unregistering it. We cannot do this right while
|
// code has been unloaded, unregistering it. We cannot do this right while
|
||||||
// holding the Patching_lock because we need to use the CodeCache_lock. This
|
// holding the CompiledMethod_lock because we need to use the CodeCache_lock. This
|
||||||
// would be prone to deadlocks.
|
// would be prone to deadlocks.
|
||||||
// This flag is used to remember whether we need to later lock and unregister.
|
// This flag is used to remember whether we need to later lock and unregister.
|
||||||
bool nmethod_needs_unregister = false;
|
bool nmethod_needs_unregister = false;
|
||||||
|
|
||||||
{
|
// invalidate osr nmethod before acquiring the patching lock since
|
||||||
// invalidate osr nmethod before acquiring the patching lock since
|
// they both acquire leaf locks and we don't want a deadlock.
|
||||||
// they both acquire leaf locks and we don't want a deadlock.
|
// This logic is equivalent to the logic below for patching the
|
||||||
// This logic is equivalent to the logic below for patching the
|
// verified entry point of regular methods. We check that the
|
||||||
// verified entry point of regular methods. We check that the
|
// nmethod is in use to ensure that it is invalidated only once.
|
||||||
// nmethod is in use to ensure that it is invalidated only once.
|
if (is_osr_method() && is_in_use()) {
|
||||||
if (is_osr_method() && is_in_use()) {
|
// this effectively makes the osr nmethod not entrant
|
||||||
// this effectively makes the osr nmethod not entrant
|
invalidate_osr_method();
|
||||||
invalidate_osr_method();
|
}
|
||||||
}
|
|
||||||
|
|
||||||
|
{
|
||||||
// Enter critical section. Does not block for safepoint.
|
// Enter critical section. Does not block for safepoint.
|
||||||
MutexLocker pl(Patching_lock, Mutex::_no_safepoint_check_flag);
|
MutexLocker ml(CompiledMethod_lock->owned_by_self() ? NULL : CompiledMethod_lock, Mutex::_no_safepoint_check_flag);
|
||||||
|
|
||||||
if (Atomic::load(&_state) >= state) {
|
if (Atomic::load(&_state) >= state) {
|
||||||
// another thread already performed this transition so nothing
|
// another thread already performed this transition so nothing
|
||||||
@ -1389,8 +1383,9 @@ bool nmethod::make_not_entrant_or_zombie(int state) {
|
|||||||
log_state_change();
|
log_state_change();
|
||||||
|
|
||||||
// Remove nmethod from method.
|
// Remove nmethod from method.
|
||||||
unlink_from_method(false /* already owns Patching_lock */);
|
unlink_from_method();
|
||||||
} // leave critical region under Patching_lock
|
|
||||||
|
} // leave critical region under CompiledMethod_lock
|
||||||
|
|
||||||
#if INCLUDE_JVMCI
|
#if INCLUDE_JVMCI
|
||||||
// Invalidate can't occur while holding the Patching lock
|
// Invalidate can't occur while holding the Patching lock
|
||||||
|
@ -119,7 +119,7 @@ class nmethod : public CompiledMethod {
|
|||||||
// used by jvmti to track if an unload event has been posted for this nmethod.
|
// used by jvmti to track if an unload event has been posted for this nmethod.
|
||||||
bool _unload_reported;
|
bool _unload_reported;
|
||||||
|
|
||||||
// Protected by Patching_lock
|
// Protected by CompiledMethod_lock
|
||||||
volatile signed char _state; // {not_installed, in_use, not_entrant, zombie, unloaded}
|
volatile signed char _state; // {not_installed, in_use, not_entrant, zombie, unloaded}
|
||||||
|
|
||||||
#ifdef ASSERT
|
#ifdef ASSERT
|
||||||
@ -357,7 +357,9 @@ class nmethod : public CompiledMethod {
|
|||||||
void set_rtm_state(RTMState state) { _rtm_state = state; }
|
void set_rtm_state(RTMState state) { _rtm_state = state; }
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
void make_in_use() { _state = in_use; }
|
bool make_in_use() {
|
||||||
|
return try_transition(in_use);
|
||||||
|
}
|
||||||
// Make the nmethod non entrant. The nmethod will continue to be
|
// Make the nmethod non entrant. The nmethod will continue to be
|
||||||
// alive. It is used when an uncommon trap happens. Returns true
|
// alive. It is used when an uncommon trap happens. Returns true
|
||||||
// if this thread changed the state of the nmethod or false if
|
// if this thread changed the state of the nmethod or false if
|
||||||
@ -390,7 +392,7 @@ class nmethod : public CompiledMethod {
|
|||||||
|
|
||||||
int comp_level() const { return _comp_level; }
|
int comp_level() const { return _comp_level; }
|
||||||
|
|
||||||
void unlink_from_method(bool acquire_lock);
|
void unlink_from_method();
|
||||||
|
|
||||||
// Support for oops in scopes and relocs:
|
// Support for oops in scopes and relocs:
|
||||||
// Note: index 0 is reserved for null.
|
// Note: index 0 is reserved for null.
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2018, Oracle and/or its affiliates. All rights reserved.
|
* Copyright (c) 2018, 2019, Oracle and/or its affiliates. All rights reserved.
|
||||||
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||||
*
|
*
|
||||||
* This code is free software; you can redistribute it and/or modify it
|
* This code is free software; you can redistribute it and/or modify it
|
||||||
@ -45,7 +45,7 @@ bool ZBarrierSetNMethod::nmethod_entry_barrier(nmethod* nm) {
|
|||||||
// We don't need to take the lock when unlinking nmethods from
|
// We don't need to take the lock when unlinking nmethods from
|
||||||
// the Method, because it is only concurrently unlinked by
|
// the Method, because it is only concurrently unlinked by
|
||||||
// the entry barrier, which acquires the per nmethod lock.
|
// the entry barrier, which acquires the per nmethod lock.
|
||||||
nm->unlink_from_method(false /* acquire_lock */);
|
nm->unlink_from_method();
|
||||||
|
|
||||||
// We can end up calling nmethods that are unloading
|
// We can end up calling nmethods that are unloading
|
||||||
// since we clear compiled ICs lazily. Returning false
|
// since we clear compiled ICs lazily. Returning false
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2017, 2018, Oracle and/or its affiliates. All rights reserved.
|
* Copyright (c) 2017, 2019, Oracle and/or its affiliates. All rights reserved.
|
||||||
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||||
*
|
*
|
||||||
* This code is free software; you can redistribute it and/or modify it
|
* This code is free software; you can redistribute it and/or modify it
|
||||||
@ -266,10 +266,11 @@ private:
|
|||||||
// handshake separating unlink and purge.
|
// handshake separating unlink and purge.
|
||||||
nm->flush_dependencies(false /* delete_immediately */);
|
nm->flush_dependencies(false /* delete_immediately */);
|
||||||
|
|
||||||
// We don't need to take the lock when unlinking nmethods from
|
// unlink_from_method will take the CompiledMethod_lock.
|
||||||
|
// In this case we don't strictly need it when unlinking nmethods from
|
||||||
// the Method, because it is only concurrently unlinked by
|
// the Method, because it is only concurrently unlinked by
|
||||||
// the entry barrier, which acquires the per nmethod lock.
|
// the entry barrier, which acquires the per nmethod lock.
|
||||||
nm->unlink_from_method(false /* acquire_lock */);
|
nm->unlink_from_method();
|
||||||
|
|
||||||
if (nm->is_osr_method()) {
|
if (nm->is_osr_method()) {
|
||||||
// Invalidate the osr nmethod before the handshake. The nmethod
|
// Invalidate the osr nmethod before the handshake. The nmethod
|
||||||
|
@ -31,6 +31,7 @@
|
|||||||
#include "memory/universe.hpp"
|
#include "memory/universe.hpp"
|
||||||
#include "oops/objArrayKlass.hpp"
|
#include "oops/objArrayKlass.hpp"
|
||||||
#include "oops/typeArrayOop.inline.hpp"
|
#include "oops/typeArrayOop.inline.hpp"
|
||||||
|
#include "runtime/deoptimization.hpp"
|
||||||
#include "runtime/jniHandles.inline.hpp"
|
#include "runtime/jniHandles.inline.hpp"
|
||||||
#include "runtime/javaCalls.hpp"
|
#include "runtime/javaCalls.hpp"
|
||||||
#include "jvmci/jniAccessMark.inline.hpp"
|
#include "jvmci/jniAccessMark.inline.hpp"
|
||||||
@ -1491,8 +1492,7 @@ void JVMCIEnv::invalidate_nmethod_mirror(JVMCIObject mirror, JVMCI_TRAPS) {
|
|||||||
// Invalidating the HotSpotNmethod means we want the nmethod
|
// Invalidating the HotSpotNmethod means we want the nmethod
|
||||||
// to be deoptimized.
|
// to be deoptimized.
|
||||||
nm->mark_for_deoptimization();
|
nm->mark_for_deoptimization();
|
||||||
VM_Deoptimize op;
|
Deoptimization::deoptimize_all_marked();
|
||||||
VMThread::execute(&op);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// A HotSpotNmethod instance can only reference a single nmethod
|
// A HotSpotNmethod instance can only reference a single nmethod
|
||||||
|
@ -1520,7 +1520,10 @@ JVMCI::CodeInstallResult JVMCIRuntime::register_method(JVMCIEnv* JVMCIENV,
|
|||||||
comp_level, method_name, nm->entry_point());
|
comp_level, method_name, nm->entry_point());
|
||||||
}
|
}
|
||||||
// Allow the code to be executed
|
// Allow the code to be executed
|
||||||
method->set_code(method, nm);
|
MutexLocker ml(CompiledMethod_lock, Mutex::_no_safepoint_check_flag);
|
||||||
|
if (nm->make_in_use()) {
|
||||||
|
method->set_code(method, nm);
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
LogTarget(Info, nmethod, install) lt;
|
LogTarget(Info, nmethod, install) lt;
|
||||||
if (lt.is_enabled()) {
|
if (lt.is_enabled()) {
|
||||||
@ -1529,12 +1532,14 @@ JVMCI::CodeInstallResult JVMCIRuntime::register_method(JVMCIEnv* JVMCIENV,
|
|||||||
lt.print("Installing osr method (%d) %s @ %d",
|
lt.print("Installing osr method (%d) %s @ %d",
|
||||||
comp_level, method_name, entry_bci);
|
comp_level, method_name, entry_bci);
|
||||||
}
|
}
|
||||||
InstanceKlass::cast(method->method_holder())->add_osr_nmethod(nm);
|
MutexLocker ml(CompiledMethod_lock, Mutex::_no_safepoint_check_flag);
|
||||||
|
if (nm->make_in_use()) {
|
||||||
|
InstanceKlass::cast(method->method_holder())->add_osr_nmethod(nm);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
assert(!nmethod_mirror.is_hotspot() || data->get_nmethod_mirror(nm, /* phantom_ref */ false) == HotSpotJVMCI::resolve(nmethod_mirror), "must be");
|
assert(!nmethod_mirror.is_hotspot() || data->get_nmethod_mirror(nm, /* phantom_ref */ false) == HotSpotJVMCI::resolve(nmethod_mirror), "must be");
|
||||||
}
|
}
|
||||||
nm->make_in_use();
|
|
||||||
}
|
}
|
||||||
result = nm != NULL ? JVMCI::ok :JVMCI::cache_full;
|
result = nm != NULL ? JVMCI::ok :JVMCI::cache_full;
|
||||||
}
|
}
|
||||||
|
@ -2973,6 +2973,7 @@ void InstanceKlass::adjust_default_methods(bool* trace_name_printed) {
|
|||||||
|
|
||||||
// On-stack replacement stuff
|
// On-stack replacement stuff
|
||||||
void InstanceKlass::add_osr_nmethod(nmethod* n) {
|
void InstanceKlass::add_osr_nmethod(nmethod* n) {
|
||||||
|
assert_lock_strong(CompiledMethod_lock);
|
||||||
#ifndef PRODUCT
|
#ifndef PRODUCT
|
||||||
if (TieredCompilation) {
|
if (TieredCompilation) {
|
||||||
nmethod * prev = lookup_osr_nmethod(n->method(), n->osr_entry_bci(), n->comp_level(), true);
|
nmethod * prev = lookup_osr_nmethod(n->method(), n->osr_entry_bci(), n->comp_level(), true);
|
||||||
@ -2982,8 +2983,6 @@ void InstanceKlass::add_osr_nmethod(nmethod* n) {
|
|||||||
#endif
|
#endif
|
||||||
// only one compilation can be active
|
// only one compilation can be active
|
||||||
{
|
{
|
||||||
// This is a short non-blocking critical region, so the no safepoint check is ok.
|
|
||||||
MutexLocker ml(OsrList_lock, Mutex::_no_safepoint_check_flag);
|
|
||||||
assert(n->is_osr_method(), "wrong kind of nmethod");
|
assert(n->is_osr_method(), "wrong kind of nmethod");
|
||||||
n->set_osr_link(osr_nmethods_head());
|
n->set_osr_link(osr_nmethods_head());
|
||||||
set_osr_nmethods_head(n);
|
set_osr_nmethods_head(n);
|
||||||
@ -3008,7 +3007,8 @@ void InstanceKlass::add_osr_nmethod(nmethod* n) {
|
|||||||
// Remove osr nmethod from the list. Return true if found and removed.
|
// Remove osr nmethod from the list. Return true if found and removed.
|
||||||
bool InstanceKlass::remove_osr_nmethod(nmethod* n) {
|
bool InstanceKlass::remove_osr_nmethod(nmethod* n) {
|
||||||
// This is a short non-blocking critical region, so the no safepoint check is ok.
|
// This is a short non-blocking critical region, so the no safepoint check is ok.
|
||||||
MutexLocker ml(OsrList_lock, Mutex::_no_safepoint_check_flag);
|
MutexLocker ml(CompiledMethod_lock->owned_by_self() ? NULL : CompiledMethod_lock
|
||||||
|
, Mutex::_no_safepoint_check_flag);
|
||||||
assert(n->is_osr_method(), "wrong kind of nmethod");
|
assert(n->is_osr_method(), "wrong kind of nmethod");
|
||||||
nmethod* last = NULL;
|
nmethod* last = NULL;
|
||||||
nmethod* cur = osr_nmethods_head();
|
nmethod* cur = osr_nmethods_head();
|
||||||
@ -3051,8 +3051,8 @@ bool InstanceKlass::remove_osr_nmethod(nmethod* n) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
int InstanceKlass::mark_osr_nmethods(const Method* m) {
|
int InstanceKlass::mark_osr_nmethods(const Method* m) {
|
||||||
// This is a short non-blocking critical region, so the no safepoint check is ok.
|
MutexLocker ml(CompiledMethod_lock->owned_by_self() ? NULL : CompiledMethod_lock,
|
||||||
MutexLocker ml(OsrList_lock, Mutex::_no_safepoint_check_flag);
|
Mutex::_no_safepoint_check_flag);
|
||||||
nmethod* osr = osr_nmethods_head();
|
nmethod* osr = osr_nmethods_head();
|
||||||
int found = 0;
|
int found = 0;
|
||||||
while (osr != NULL) {
|
while (osr != NULL) {
|
||||||
@ -3067,8 +3067,8 @@ int InstanceKlass::mark_osr_nmethods(const Method* m) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
nmethod* InstanceKlass::lookup_osr_nmethod(const Method* m, int bci, int comp_level, bool match_level) const {
|
nmethod* InstanceKlass::lookup_osr_nmethod(const Method* m, int bci, int comp_level, bool match_level) const {
|
||||||
// This is a short non-blocking critical region, so the no safepoint check is ok.
|
MutexLocker ml(CompiledMethod_lock->owned_by_self() ? NULL : CompiledMethod_lock,
|
||||||
MutexLocker ml(OsrList_lock, Mutex::_no_safepoint_check_flag);
|
Mutex::_no_safepoint_check_flag);
|
||||||
nmethod* osr = osr_nmethods_head();
|
nmethod* osr = osr_nmethods_head();
|
||||||
nmethod* best = NULL;
|
nmethod* best = NULL;
|
||||||
while (osr != NULL) {
|
while (osr != NULL) {
|
||||||
|
@ -103,7 +103,7 @@ Method::Method(ConstMethod* xconst, AccessFlags access_flags) {
|
|||||||
// Fix and bury in Method*
|
// Fix and bury in Method*
|
||||||
set_interpreter_entry(NULL); // sets i2i entry and from_int
|
set_interpreter_entry(NULL); // sets i2i entry and from_int
|
||||||
set_adapter_entry(NULL);
|
set_adapter_entry(NULL);
|
||||||
clear_code(false /* don't need a lock */); // from_c/from_i get set to c2i/i2i
|
Method::clear_code(); // from_c/from_i get set to c2i/i2i
|
||||||
|
|
||||||
if (access_flags.is_native()) {
|
if (access_flags.is_native()) {
|
||||||
clear_native_function();
|
clear_native_function();
|
||||||
@ -825,7 +825,7 @@ void Method::clear_native_function() {
|
|||||||
set_native_function(
|
set_native_function(
|
||||||
SharedRuntime::native_method_throw_unsatisfied_link_error_entry(),
|
SharedRuntime::native_method_throw_unsatisfied_link_error_entry(),
|
||||||
!native_bind_event_is_interesting);
|
!native_bind_event_is_interesting);
|
||||||
clear_code();
|
this->unlink_code();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
@ -941,8 +941,7 @@ void Method::set_not_osr_compilable(const char* reason, int comp_level, bool rep
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Revert to using the interpreter and clear out the nmethod
|
// Revert to using the interpreter and clear out the nmethod
|
||||||
void Method::clear_code(bool acquire_lock /* = true */) {
|
void Method::clear_code() {
|
||||||
MutexLocker pl(acquire_lock ? Patching_lock : NULL, Mutex::_no_safepoint_check_flag);
|
|
||||||
// this may be NULL if c2i adapters have not been made yet
|
// this may be NULL if c2i adapters have not been made yet
|
||||||
// Only should happen at allocate time.
|
// Only should happen at allocate time.
|
||||||
if (adapter() == NULL) {
|
if (adapter() == NULL) {
|
||||||
@ -956,6 +955,25 @@ void Method::clear_code(bool acquire_lock /* = true */) {
|
|||||||
_code = NULL;
|
_code = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void Method::unlink_code(CompiledMethod *compare) {
|
||||||
|
MutexLocker ml(CompiledMethod_lock->owned_by_self() ? NULL : CompiledMethod_lock, Mutex::_no_safepoint_check_flag);
|
||||||
|
// We need to check if either the _code or _from_compiled_code_entry_point
|
||||||
|
// refer to this nmethod because there is a race in setting these two fields
|
||||||
|
// in Method* as seen in bugid 4947125.
|
||||||
|
// If the vep() points to the zombie nmethod, the memory for the nmethod
|
||||||
|
// could be flushed and the compiler and vtable stubs could still call
|
||||||
|
// through it.
|
||||||
|
if (code() == compare ||
|
||||||
|
from_compiled_entry() == compare->verified_entry_point()) {
|
||||||
|
clear_code();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
void Method::unlink_code() {
|
||||||
|
MutexLocker ml(CompiledMethod_lock->owned_by_self() ? NULL : CompiledMethod_lock, Mutex::_no_safepoint_check_flag);
|
||||||
|
clear_code();
|
||||||
|
}
|
||||||
|
|
||||||
#if INCLUDE_CDS
|
#if INCLUDE_CDS
|
||||||
// Called by class data sharing to remove any entry points (which are not shared)
|
// Called by class data sharing to remove any entry points (which are not shared)
|
||||||
void Method::unlink_method() {
|
void Method::unlink_method() {
|
||||||
@ -1182,7 +1200,7 @@ bool Method::check_code() const {
|
|||||||
|
|
||||||
// Install compiled code. Instantly it can execute.
|
// Install compiled code. Instantly it can execute.
|
||||||
void Method::set_code(const methodHandle& mh, CompiledMethod *code) {
|
void Method::set_code(const methodHandle& mh, CompiledMethod *code) {
|
||||||
MutexLocker pl(Patching_lock, Mutex::_no_safepoint_check_flag);
|
assert_lock_strong(CompiledMethod_lock);
|
||||||
assert( code, "use clear_code to remove code" );
|
assert( code, "use clear_code to remove code" );
|
||||||
assert( mh->check_code(), "" );
|
assert( mh->check_code(), "" );
|
||||||
|
|
||||||
|
@ -463,7 +463,17 @@ class Method : public Metadata {
|
|||||||
address verified_code_entry();
|
address verified_code_entry();
|
||||||
bool check_code() const; // Not inline to avoid circular ref
|
bool check_code() const; // Not inline to avoid circular ref
|
||||||
CompiledMethod* volatile code() const;
|
CompiledMethod* volatile code() const;
|
||||||
void clear_code(bool acquire_lock = true); // Clear out any compiled code
|
|
||||||
|
// Locks CompiledMethod_lock if not held.
|
||||||
|
void unlink_code(CompiledMethod *compare);
|
||||||
|
// Locks CompiledMethod_lock if not held.
|
||||||
|
void unlink_code();
|
||||||
|
|
||||||
|
private:
|
||||||
|
// Either called with CompiledMethod_lock held or from constructor.
|
||||||
|
void clear_code();
|
||||||
|
|
||||||
|
public:
|
||||||
static void set_code(const methodHandle& mh, CompiledMethod* code);
|
static void set_code(const methodHandle& mh, CompiledMethod* code);
|
||||||
void set_adapter_entry(AdapterHandlerEntry* adapter) {
|
void set_adapter_entry(AdapterHandlerEntry* adapter) {
|
||||||
constMethod()->set_adapter_entry(adapter);
|
constMethod()->set_adapter_entry(adapter);
|
||||||
|
@ -32,6 +32,7 @@
|
|||||||
#include "prims/jvmtiExport.hpp"
|
#include "prims/jvmtiExport.hpp"
|
||||||
#include "prims/jvmtiImpl.hpp"
|
#include "prims/jvmtiImpl.hpp"
|
||||||
#include "prims/jvmtiThreadState.inline.hpp"
|
#include "prims/jvmtiThreadState.inline.hpp"
|
||||||
|
#include "runtime/deoptimization.hpp"
|
||||||
#include "runtime/frame.hpp"
|
#include "runtime/frame.hpp"
|
||||||
#include "runtime/thread.inline.hpp"
|
#include "runtime/thread.inline.hpp"
|
||||||
#include "runtime/threadSMR.hpp"
|
#include "runtime/threadSMR.hpp"
|
||||||
@ -239,8 +240,7 @@ void VM_EnterInterpOnlyMode::doit() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (num_marked > 0) {
|
if (num_marked > 0) {
|
||||||
VM_Deoptimize op;
|
Deoptimization::deoptimize_all_marked();
|
||||||
VMThread::execute(&op);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -42,6 +42,7 @@
|
|||||||
#include "oops/typeArrayOop.inline.hpp"
|
#include "oops/typeArrayOop.inline.hpp"
|
||||||
#include "prims/methodHandles.hpp"
|
#include "prims/methodHandles.hpp"
|
||||||
#include "runtime/compilationPolicy.hpp"
|
#include "runtime/compilationPolicy.hpp"
|
||||||
|
#include "runtime/deoptimization.hpp"
|
||||||
#include "runtime/fieldDescriptor.inline.hpp"
|
#include "runtime/fieldDescriptor.inline.hpp"
|
||||||
#include "runtime/handles.inline.hpp"
|
#include "runtime/handles.inline.hpp"
|
||||||
#include "runtime/interfaceSupport.inline.hpp"
|
#include "runtime/interfaceSupport.inline.hpp"
|
||||||
@ -1109,8 +1110,7 @@ void MethodHandles::flush_dependent_nmethods(Handle call_site, Handle target) {
|
|||||||
}
|
}
|
||||||
if (marked > 0) {
|
if (marked > 0) {
|
||||||
// At least one nmethod has been marked for deoptimization.
|
// At least one nmethod has been marked for deoptimization.
|
||||||
VM_Deoptimize op;
|
Deoptimization::deoptimize_all_marked();
|
||||||
VMThread::execute(&op);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1506,8 +1506,7 @@ JVM_ENTRY(void, MHN_clearCallSiteContext(JNIEnv* env, jobject igcls, jobject con
|
|||||||
}
|
}
|
||||||
if (marked > 0) {
|
if (marked > 0) {
|
||||||
// At least one nmethod has been marked for deoptimization
|
// At least one nmethod has been marked for deoptimization
|
||||||
VM_Deoptimize op;
|
Deoptimization::deoptimize_all_marked();
|
||||||
VMThread::execute(&op);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -820,10 +820,8 @@ WB_ENTRY(jint, WB_DeoptimizeFrames(JNIEnv* env, jobject o, jboolean make_not_ent
|
|||||||
WB_END
|
WB_END
|
||||||
|
|
||||||
WB_ENTRY(void, WB_DeoptimizeAll(JNIEnv* env, jobject o))
|
WB_ENTRY(void, WB_DeoptimizeAll(JNIEnv* env, jobject o))
|
||||||
MutexLocker mu(Compile_lock);
|
|
||||||
CodeCache::mark_all_nmethods_for_deoptimization();
|
CodeCache::mark_all_nmethods_for_deoptimization();
|
||||||
VM_Deoptimize op;
|
Deoptimization::deoptimize_all_marked();
|
||||||
VMThread::execute(&op);
|
|
||||||
WB_END
|
WB_END
|
||||||
|
|
||||||
WB_ENTRY(jint, WB_DeoptimizeMethod(JNIEnv* env, jobject o, jobject method, jboolean is_osr))
|
WB_ENTRY(jint, WB_DeoptimizeMethod(JNIEnv* env, jobject o, jobject method, jboolean is_osr))
|
||||||
@ -840,8 +838,7 @@ WB_ENTRY(jint, WB_DeoptimizeMethod(JNIEnv* env, jobject o, jobject method, jbool
|
|||||||
}
|
}
|
||||||
result += CodeCache::mark_for_deoptimization(mh());
|
result += CodeCache::mark_for_deoptimization(mh());
|
||||||
if (result > 0) {
|
if (result > 0) {
|
||||||
VM_Deoptimize op;
|
Deoptimization::deoptimize_all_marked();
|
||||||
VMThread::execute(&op);
|
|
||||||
}
|
}
|
||||||
return result;
|
return result;
|
||||||
WB_END
|
WB_END
|
||||||
|
@ -726,6 +726,29 @@ void BiasedLocking::walk_stack_and_revoke(oop obj, JavaThread* biased_locker) {
|
|||||||
assert(!obj->mark().has_bias_pattern(), "must not be biased");
|
assert(!obj->mark().has_bias_pattern(), "must not be biased");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void BiasedLocking::revoke_own_lock(Handle obj, TRAPS) {
|
||||||
|
assert(THREAD->is_Java_thread(), "must be called by a JavaThread");
|
||||||
|
JavaThread* thread = (JavaThread*)THREAD;
|
||||||
|
|
||||||
|
markWord mark = obj->mark();
|
||||||
|
|
||||||
|
if (!mark.has_bias_pattern()) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
Klass *k = obj->klass();
|
||||||
|
assert(mark.biased_locker() == thread &&
|
||||||
|
k->prototype_header().bias_epoch() == mark.bias_epoch(), "Revoke failed, unhandled biased lock state");
|
||||||
|
ResourceMark rm;
|
||||||
|
log_info(biasedlocking)("Revoking bias by walking my own stack:");
|
||||||
|
EventBiasedLockSelfRevocation event;
|
||||||
|
BiasedLocking::walk_stack_and_revoke(obj(), (JavaThread*) thread);
|
||||||
|
thread->set_cached_monitor_info(NULL);
|
||||||
|
assert(!obj->mark().has_bias_pattern(), "invariant");
|
||||||
|
if (event.should_commit()) {
|
||||||
|
post_self_revocation_event(&event, k);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void BiasedLocking::revoke(Handle obj, TRAPS) {
|
void BiasedLocking::revoke(Handle obj, TRAPS) {
|
||||||
assert(!SafepointSynchronize::is_at_safepoint(), "must not be called while at safepoint");
|
assert(!SafepointSynchronize::is_at_safepoint(), "must not be called while at safepoint");
|
||||||
@ -864,23 +887,6 @@ void BiasedLocking::revoke_at_safepoint(Handle h_obj) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void BiasedLocking::revoke_at_safepoint(GrowableArray<Handle>* objs) {
|
|
||||||
assert(SafepointSynchronize::is_at_safepoint(), "must only be called while at safepoint");
|
|
||||||
int len = objs->length();
|
|
||||||
for (int i = 0; i < len; i++) {
|
|
||||||
oop obj = (objs->at(i))();
|
|
||||||
HeuristicsResult heuristics = update_heuristics(obj);
|
|
||||||
if (heuristics == HR_SINGLE_REVOKE) {
|
|
||||||
single_revoke_at_safepoint(obj, false, NULL, NULL);
|
|
||||||
} else if ((heuristics == HR_BULK_REBIAS) ||
|
|
||||||
(heuristics == HR_BULK_REVOKE)) {
|
|
||||||
bulk_revoke_at_safepoint(obj, (heuristics == HR_BULK_REBIAS), NULL);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
clean_up_cached_monitor_info();
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
void BiasedLocking::preserve_marks() {
|
void BiasedLocking::preserve_marks() {
|
||||||
if (!UseBiasedLocking)
|
if (!UseBiasedLocking)
|
||||||
return;
|
return;
|
||||||
|
@ -190,12 +190,14 @@ public:
|
|||||||
// This should be called by JavaThreads to revoke the bias of an object
|
// This should be called by JavaThreads to revoke the bias of an object
|
||||||
static void revoke(Handle obj, TRAPS);
|
static void revoke(Handle obj, TRAPS);
|
||||||
|
|
||||||
|
// This must only be called by a JavaThread to revoke the bias of an owned object.
|
||||||
|
static void revoke_own_lock(Handle obj, TRAPS);
|
||||||
|
|
||||||
static void revoke_at_safepoint(Handle obj);
|
static void revoke_at_safepoint(Handle obj);
|
||||||
|
|
||||||
// These are used by deoptimization to ensure that monitors on the stack
|
// These are used by deoptimization to ensure that monitors on the stack
|
||||||
// can be migrated
|
// can be migrated
|
||||||
static void revoke(GrowableArray<Handle>* objs, JavaThread *biaser);
|
static void revoke(GrowableArray<Handle>* objs, JavaThread *biaser);
|
||||||
static void revoke_at_safepoint(GrowableArray<Handle>* objs);
|
|
||||||
|
|
||||||
static void print_counters() { _counters.print(); }
|
static void print_counters() { _counters.print(); }
|
||||||
static BiasedLockingCounters* counters() { return &_counters; }
|
static BiasedLockingCounters* counters() { return &_counters; }
|
||||||
|
@ -157,6 +157,92 @@ JRT_BLOCK_ENTRY(Deoptimization::UnrollBlock*, Deoptimization::fetch_unroll_info(
|
|||||||
return fetch_unroll_info_helper(thread, exec_mode);
|
return fetch_unroll_info_helper(thread, exec_mode);
|
||||||
JRT_END
|
JRT_END
|
||||||
|
|
||||||
|
#if COMPILER2_OR_JVMCI
|
||||||
|
static bool eliminate_allocations(JavaThread* thread, int exec_mode, CompiledMethod* compiled_method,
|
||||||
|
frame& deoptee, RegisterMap& map, GrowableArray<compiledVFrame*>* chunk) {
|
||||||
|
bool realloc_failures = false;
|
||||||
|
assert (chunk->at(0)->scope() != NULL,"expect only compiled java frames");
|
||||||
|
|
||||||
|
GrowableArray<ScopeValue*>* objects = chunk->at(0)->scope()->objects();
|
||||||
|
|
||||||
|
// The flag return_oop() indicates call sites which return oop
|
||||||
|
// in compiled code. Such sites include java method calls,
|
||||||
|
// runtime calls (for example, used to allocate new objects/arrays
|
||||||
|
// on slow code path) and any other calls generated in compiled code.
|
||||||
|
// It is not guaranteed that we can get such information here only
|
||||||
|
// by analyzing bytecode in deoptimized frames. This is why this flag
|
||||||
|
// is set during method compilation (see Compile::Process_OopMap_Node()).
|
||||||
|
// If the previous frame was popped or if we are dispatching an exception,
|
||||||
|
// we don't have an oop result.
|
||||||
|
bool save_oop_result = chunk->at(0)->scope()->return_oop() && !thread->popframe_forcing_deopt_reexecution() && (exec_mode == Deoptimization::Unpack_deopt);
|
||||||
|
Handle return_value;
|
||||||
|
if (save_oop_result) {
|
||||||
|
// Reallocation may trigger GC. If deoptimization happened on return from
|
||||||
|
// call which returns oop we need to save it since it is not in oopmap.
|
||||||
|
oop result = deoptee.saved_oop_result(&map);
|
||||||
|
assert(oopDesc::is_oop_or_null(result), "must be oop");
|
||||||
|
return_value = Handle(thread, result);
|
||||||
|
assert(Universe::heap()->is_in_or_null(result), "must be heap pointer");
|
||||||
|
if (TraceDeoptimization) {
|
||||||
|
ttyLocker ttyl;
|
||||||
|
tty->print_cr("SAVED OOP RESULT " INTPTR_FORMAT " in thread " INTPTR_FORMAT, p2i(result), p2i(thread));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (objects != NULL) {
|
||||||
|
JRT_BLOCK
|
||||||
|
realloc_failures = Deoptimization::realloc_objects(thread, &deoptee, &map, objects, THREAD);
|
||||||
|
JRT_END
|
||||||
|
bool skip_internal = (compiled_method != NULL) && !compiled_method->is_compiled_by_jvmci();
|
||||||
|
Deoptimization::reassign_fields(&deoptee, &map, objects, realloc_failures, skip_internal);
|
||||||
|
#ifndef PRODUCT
|
||||||
|
if (TraceDeoptimization) {
|
||||||
|
ttyLocker ttyl;
|
||||||
|
tty->print_cr("REALLOC OBJECTS in thread " INTPTR_FORMAT, p2i(thread));
|
||||||
|
Deoptimization::print_objects(objects, realloc_failures);
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
}
|
||||||
|
if (save_oop_result) {
|
||||||
|
// Restore result.
|
||||||
|
deoptee.set_saved_oop_result(&map, return_value());
|
||||||
|
}
|
||||||
|
return realloc_failures;
|
||||||
|
}
|
||||||
|
|
||||||
|
static void eliminate_locks(JavaThread* thread, GrowableArray<compiledVFrame*>* chunk, bool realloc_failures) {
|
||||||
|
#ifndef PRODUCT
|
||||||
|
bool first = true;
|
||||||
|
#endif
|
||||||
|
for (int i = 0; i < chunk->length(); i++) {
|
||||||
|
compiledVFrame* cvf = chunk->at(i);
|
||||||
|
assert (cvf->scope() != NULL,"expect only compiled java frames");
|
||||||
|
GrowableArray<MonitorInfo*>* monitors = cvf->monitors();
|
||||||
|
if (monitors->is_nonempty()) {
|
||||||
|
Deoptimization::relock_objects(monitors, thread, realloc_failures);
|
||||||
|
#ifndef PRODUCT
|
||||||
|
if (PrintDeoptimizationDetails) {
|
||||||
|
ttyLocker ttyl;
|
||||||
|
for (int j = 0; j < monitors->length(); j++) {
|
||||||
|
MonitorInfo* mi = monitors->at(j);
|
||||||
|
if (mi->eliminated()) {
|
||||||
|
if (first) {
|
||||||
|
first = false;
|
||||||
|
tty->print_cr("RELOCK OBJECTS in thread " INTPTR_FORMAT, p2i(thread));
|
||||||
|
}
|
||||||
|
if (mi->owner_is_scalar_replaced()) {
|
||||||
|
Klass* k = java_lang_Class::as_Klass(mi->owner_klass());
|
||||||
|
tty->print_cr(" failed reallocation for klass %s", k->external_name());
|
||||||
|
} else {
|
||||||
|
tty->print_cr(" object <" INTPTR_FORMAT "> locked", p2i(mi->owner()));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endif // !PRODUCT
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endif // COMPILER2_OR_JVMCI
|
||||||
|
|
||||||
// This is factored, since it is both called from a JRT_LEAF (deoptimization) and a JRT_ENTRY (uncommon_trap)
|
// This is factored, since it is both called from a JRT_LEAF (deoptimization) and a JRT_ENTRY (uncommon_trap)
|
||||||
Deoptimization::UnrollBlock* Deoptimization::fetch_unroll_info_helper(JavaThread* thread, int exec_mode) {
|
Deoptimization::UnrollBlock* Deoptimization::fetch_unroll_info_helper(JavaThread* thread, int exec_mode) {
|
||||||
@ -201,95 +287,33 @@ Deoptimization::UnrollBlock* Deoptimization::fetch_unroll_info_helper(JavaThread
|
|||||||
bool realloc_failures = false;
|
bool realloc_failures = false;
|
||||||
|
|
||||||
#if COMPILER2_OR_JVMCI
|
#if COMPILER2_OR_JVMCI
|
||||||
|
#if INCLUDE_JVMCI
|
||||||
|
bool jvmci_enabled = true;
|
||||||
|
#else
|
||||||
|
bool jvmci_enabled = false;
|
||||||
|
#endif
|
||||||
|
|
||||||
// Reallocate the non-escaping objects and restore their fields. Then
|
// Reallocate the non-escaping objects and restore their fields. Then
|
||||||
// relock objects if synchronization on them was eliminated.
|
// relock objects if synchronization on them was eliminated.
|
||||||
#if !INCLUDE_JVMCI
|
if (jvmci_enabled || ((DoEscapeAnalysis || EliminateNestedLocks) && EliminateAllocations)) {
|
||||||
if (DoEscapeAnalysis || EliminateNestedLocks) {
|
realloc_failures = eliminate_allocations(thread, exec_mode, cm, deoptee, map, chunk);
|
||||||
if (EliminateAllocations) {
|
}
|
||||||
#endif // INCLUDE_JVMCI
|
#endif // COMPILER2_OR_JVMCI
|
||||||
assert (chunk->at(0)->scope() != NULL,"expect only compiled java frames");
|
|
||||||
GrowableArray<ScopeValue*>* objects = chunk->at(0)->scope()->objects();
|
// Revoke biases, done with in java state.
|
||||||
|
// No safepoints allowed after this
|
||||||
// The flag return_oop() indicates call sites which return oop
|
revoke_from_deopt_handler(thread, deoptee, &map);
|
||||||
// in compiled code. Such sites include java method calls,
|
|
||||||
// runtime calls (for example, used to allocate new objects/arrays
|
// Ensure that no safepoint is taken after pointers have been stored
|
||||||
// on slow code path) and any other calls generated in compiled code.
|
// in fields of rematerialized objects. If a safepoint occurs from here on
|
||||||
// It is not guaranteed that we can get such information here only
|
// out the java state residing in the vframeArray will be missed.
|
||||||
// by analyzing bytecode in deoptimized frames. This is why this flag
|
// Locks may be rebaised in a safepoint.
|
||||||
// is set during method compilation (see Compile::Process_OopMap_Node()).
|
NoSafepointVerifier no_safepoint;
|
||||||
// If the previous frame was popped or if we are dispatching an exception,
|
|
||||||
// we don't have an oop result.
|
#if COMPILER2_OR_JVMCI
|
||||||
bool save_oop_result = chunk->at(0)->scope()->return_oop() && !thread->popframe_forcing_deopt_reexecution() && (exec_mode == Unpack_deopt);
|
if (jvmci_enabled || ((DoEscapeAnalysis || EliminateNestedLocks) && EliminateLocks)) {
|
||||||
Handle return_value;
|
eliminate_locks(thread, chunk, realloc_failures);
|
||||||
if (save_oop_result) {
|
|
||||||
// Reallocation may trigger GC. If deoptimization happened on return from
|
|
||||||
// call which returns oop we need to save it since it is not in oopmap.
|
|
||||||
oop result = deoptee.saved_oop_result(&map);
|
|
||||||
assert(oopDesc::is_oop_or_null(result), "must be oop");
|
|
||||||
return_value = Handle(thread, result);
|
|
||||||
assert(Universe::heap()->is_in_or_null(result), "must be heap pointer");
|
|
||||||
if (TraceDeoptimization) {
|
|
||||||
ttyLocker ttyl;
|
|
||||||
tty->print_cr("SAVED OOP RESULT " INTPTR_FORMAT " in thread " INTPTR_FORMAT, p2i(result), p2i(thread));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (objects != NULL) {
|
|
||||||
JRT_BLOCK
|
|
||||||
realloc_failures = realloc_objects(thread, &deoptee, &map, objects, THREAD);
|
|
||||||
JRT_END
|
|
||||||
bool skip_internal = (cm != NULL) && !cm->is_compiled_by_jvmci();
|
|
||||||
reassign_fields(&deoptee, &map, objects, realloc_failures, skip_internal);
|
|
||||||
#ifndef PRODUCT
|
|
||||||
if (TraceDeoptimization) {
|
|
||||||
ttyLocker ttyl;
|
|
||||||
tty->print_cr("REALLOC OBJECTS in thread " INTPTR_FORMAT, p2i(thread));
|
|
||||||
print_objects(objects, realloc_failures);
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
if (save_oop_result) {
|
|
||||||
// Restore result.
|
|
||||||
deoptee.set_saved_oop_result(&map, return_value());
|
|
||||||
}
|
|
||||||
#if !INCLUDE_JVMCI
|
|
||||||
}
|
|
||||||
if (EliminateLocks) {
|
|
||||||
#endif // INCLUDE_JVMCI
|
|
||||||
#ifndef PRODUCT
|
|
||||||
bool first = true;
|
|
||||||
#endif
|
|
||||||
for (int i = 0; i < chunk->length(); i++) {
|
|
||||||
compiledVFrame* cvf = chunk->at(i);
|
|
||||||
assert (cvf->scope() != NULL,"expect only compiled java frames");
|
|
||||||
GrowableArray<MonitorInfo*>* monitors = cvf->monitors();
|
|
||||||
if (monitors->is_nonempty()) {
|
|
||||||
relock_objects(monitors, thread, realloc_failures);
|
|
||||||
#ifndef PRODUCT
|
|
||||||
if (PrintDeoptimizationDetails) {
|
|
||||||
ttyLocker ttyl;
|
|
||||||
for (int j = 0; j < monitors->length(); j++) {
|
|
||||||
MonitorInfo* mi = monitors->at(j);
|
|
||||||
if (mi->eliminated()) {
|
|
||||||
if (first) {
|
|
||||||
first = false;
|
|
||||||
tty->print_cr("RELOCK OBJECTS in thread " INTPTR_FORMAT, p2i(thread));
|
|
||||||
}
|
|
||||||
if (mi->owner_is_scalar_replaced()) {
|
|
||||||
Klass* k = java_lang_Class::as_Klass(mi->owner_klass());
|
|
||||||
tty->print_cr(" failed reallocation for klass %s", k->external_name());
|
|
||||||
} else {
|
|
||||||
tty->print_cr(" object <" INTPTR_FORMAT "> locked", p2i(mi->owner()));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
#endif // !PRODUCT
|
|
||||||
}
|
|
||||||
}
|
|
||||||
#if !INCLUDE_JVMCI
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
#endif // INCLUDE_JVMCI
|
|
||||||
#endif // COMPILER2_OR_JVMCI
|
#endif // COMPILER2_OR_JVMCI
|
||||||
|
|
||||||
ScopeDesc* trap_scope = chunk->at(0)->scope();
|
ScopeDesc* trap_scope = chunk->at(0)->scope();
|
||||||
@ -305,11 +329,6 @@ Deoptimization::UnrollBlock* Deoptimization::fetch_unroll_info_helper(JavaThread
|
|||||||
guarantee(exceptionObject() != NULL, "exception oop can not be null");
|
guarantee(exceptionObject() != NULL, "exception oop can not be null");
|
||||||
}
|
}
|
||||||
|
|
||||||
// Ensure that no safepoint is taken after pointers have been stored
|
|
||||||
// in fields of rematerialized objects. If a safepoint occurs from here on
|
|
||||||
// out the java state residing in the vframeArray will be missed.
|
|
||||||
NoSafepointVerifier no_safepoint;
|
|
||||||
|
|
||||||
vframeArray* array = create_vframeArray(thread, deoptee, &map, chunk, realloc_failures);
|
vframeArray* array = create_vframeArray(thread, deoptee, &map, chunk, realloc_failures);
|
||||||
#if COMPILER2_OR_JVMCI
|
#if COMPILER2_OR_JVMCI
|
||||||
if (realloc_failures) {
|
if (realloc_failures) {
|
||||||
@ -779,10 +798,33 @@ JRT_LEAF(BasicType, Deoptimization::unpack_frames(JavaThread* thread, int exec_m
|
|||||||
return bt;
|
return bt;
|
||||||
JRT_END
|
JRT_END
|
||||||
|
|
||||||
|
class DeoptimizeMarkedTC : public ThreadClosure {
|
||||||
|
public:
|
||||||
|
virtual void do_thread(Thread* thread) {
|
||||||
|
assert(thread->is_Java_thread(), "must be");
|
||||||
|
JavaThread* jt = (JavaThread*)thread;
|
||||||
|
jt->deoptimize_marked_methods();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
int Deoptimization::deoptimize_dependents() {
|
void Deoptimization::deoptimize_all_marked() {
|
||||||
Threads::deoptimized_wrt_marked_nmethods();
|
ResourceMark rm;
|
||||||
return 0;
|
DeoptimizationMarker dm;
|
||||||
|
|
||||||
|
if (SafepointSynchronize::is_at_safepoint()) {
|
||||||
|
DeoptimizeMarkedTC deopt;
|
||||||
|
// Make the dependent methods not entrant
|
||||||
|
CodeCache::make_marked_nmethods_not_entrant();
|
||||||
|
Threads::java_threads_do(&deopt);
|
||||||
|
} else {
|
||||||
|
// Make the dependent methods not entrant
|
||||||
|
{
|
||||||
|
MutexLocker mu(CodeCache_lock, Mutex::_no_safepoint_check_flag);
|
||||||
|
CodeCache::make_marked_nmethods_not_entrant();
|
||||||
|
}
|
||||||
|
DeoptimizeMarkedTC deopt;
|
||||||
|
Handshake::execute(&deopt);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
Deoptimization::DeoptAction Deoptimization::_unloaded_action
|
Deoptimization::DeoptAction Deoptimization::_unloaded_action
|
||||||
@ -1397,14 +1439,7 @@ static void collect_monitors(compiledVFrame* cvf, GrowableArray<Handle>* objects
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static void get_monitors_from_stack(GrowableArray<Handle>* objects_to_revoke, JavaThread* thread, frame fr, RegisterMap* map) {
|
||||||
void Deoptimization::revoke_biases_of_monitors(JavaThread* thread, frame fr, RegisterMap* map) {
|
|
||||||
if (!UseBiasedLocking) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
GrowableArray<Handle>* objects_to_revoke = new GrowableArray<Handle>();
|
|
||||||
|
|
||||||
// Unfortunately we don't have a RegisterMap available in most of
|
// Unfortunately we don't have a RegisterMap available in most of
|
||||||
// the places we want to call this routine so we need to walk the
|
// the places we want to call this routine so we need to walk the
|
||||||
// stack again to update the register map.
|
// stack again to update the register map.
|
||||||
@ -1428,11 +1463,20 @@ void Deoptimization::revoke_biases_of_monitors(JavaThread* thread, frame fr, Reg
|
|||||||
cvf = compiledVFrame::cast(cvf->sender());
|
cvf = compiledVFrame::cast(cvf->sender());
|
||||||
}
|
}
|
||||||
collect_monitors(cvf, objects_to_revoke);
|
collect_monitors(cvf, objects_to_revoke);
|
||||||
|
}
|
||||||
|
|
||||||
if (SafepointSynchronize::is_at_safepoint()) {
|
void Deoptimization::revoke_from_deopt_handler(JavaThread* thread, frame fr, RegisterMap* map) {
|
||||||
BiasedLocking::revoke_at_safepoint(objects_to_revoke);
|
if (!UseBiasedLocking) {
|
||||||
} else {
|
return;
|
||||||
BiasedLocking::revoke(objects_to_revoke, thread);
|
}
|
||||||
|
GrowableArray<Handle>* objects_to_revoke = new GrowableArray<Handle>();
|
||||||
|
get_monitors_from_stack(objects_to_revoke, thread, fr, map);
|
||||||
|
|
||||||
|
int len = objects_to_revoke->length();
|
||||||
|
for (int i = 0; i < len; i++) {
|
||||||
|
oop obj = (objects_to_revoke->at(i))();
|
||||||
|
BiasedLocking::revoke_own_lock(objects_to_revoke->at(i), thread);
|
||||||
|
assert(!obj->mark().has_bias_pattern(), "biases should be revoked by now");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -1464,10 +1508,6 @@ void Deoptimization::deoptimize_single_frame(JavaThread* thread, frame fr, Deopt
|
|||||||
fr.deoptimize(thread);
|
fr.deoptimize(thread);
|
||||||
}
|
}
|
||||||
|
|
||||||
void Deoptimization::deoptimize(JavaThread* thread, frame fr, RegisterMap *map) {
|
|
||||||
deoptimize(thread, fr, map, Reason_constraint);
|
|
||||||
}
|
|
||||||
|
|
||||||
void Deoptimization::deoptimize(JavaThread* thread, frame fr, RegisterMap *map, DeoptReason reason) {
|
void Deoptimization::deoptimize(JavaThread* thread, frame fr, RegisterMap *map, DeoptReason reason) {
|
||||||
// Deoptimize only if the frame comes from compile code.
|
// Deoptimize only if the frame comes from compile code.
|
||||||
// Do not deoptimize the frame which is already patched
|
// Do not deoptimize the frame which is already patched
|
||||||
@ -1477,11 +1517,7 @@ void Deoptimization::deoptimize(JavaThread* thread, frame fr, RegisterMap *map,
|
|||||||
}
|
}
|
||||||
ResourceMark rm;
|
ResourceMark rm;
|
||||||
DeoptimizationMarker dm;
|
DeoptimizationMarker dm;
|
||||||
if (UseBiasedLocking) {
|
|
||||||
revoke_biases_of_monitors(thread, fr, map);
|
|
||||||
}
|
|
||||||
deoptimize_single_frame(thread, fr, reason);
|
deoptimize_single_frame(thread, fr, reason);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
#if INCLUDE_JVMCI
|
#if INCLUDE_JVMCI
|
||||||
@ -1642,9 +1678,6 @@ JRT_ENTRY(void, Deoptimization::uncommon_trap_inner(JavaThread* thread, jint tra
|
|||||||
{
|
{
|
||||||
ResourceMark rm;
|
ResourceMark rm;
|
||||||
|
|
||||||
// Revoke biases of any monitors in the frame to ensure we can migrate them
|
|
||||||
revoke_biases_of_monitors(thread, fr, ®_map);
|
|
||||||
|
|
||||||
DeoptReason reason = trap_request_reason(trap_request);
|
DeoptReason reason = trap_request_reason(trap_request);
|
||||||
DeoptAction action = trap_request_action(trap_request);
|
DeoptAction action = trap_request_action(trap_request);
|
||||||
#if INCLUDE_JVMCI
|
#if INCLUDE_JVMCI
|
||||||
|
@ -137,13 +137,19 @@ class Deoptimization : AllStatic {
|
|||||||
Unpack_LIMIT = 4
|
Unpack_LIMIT = 4
|
||||||
};
|
};
|
||||||
|
|
||||||
|
static void deoptimize_all_marked();
|
||||||
|
|
||||||
|
private:
|
||||||
// Checks all compiled methods. Invalid methods are deleted and
|
// Checks all compiled methods. Invalid methods are deleted and
|
||||||
// corresponding activations are deoptimized.
|
// corresponding activations are deoptimized.
|
||||||
static int deoptimize_dependents();
|
static int deoptimize_dependents();
|
||||||
|
|
||||||
|
// Revoke biased locks at deopt.
|
||||||
|
static void revoke_from_deopt_handler(JavaThread* thread, frame fr, RegisterMap* map);
|
||||||
|
|
||||||
|
public:
|
||||||
// Deoptimizes a frame lazily. nmethod gets patched deopt happens on return to the frame
|
// Deoptimizes a frame lazily. nmethod gets patched deopt happens on return to the frame
|
||||||
static void deoptimize(JavaThread* thread, frame fr, RegisterMap *reg_map);
|
static void deoptimize(JavaThread* thread, frame fr, RegisterMap *reg_map, DeoptReason reason = Reason_constraint);
|
||||||
static void deoptimize(JavaThread* thread, frame fr, RegisterMap *reg_map, DeoptReason reason);
|
|
||||||
|
|
||||||
#if INCLUDE_JVMCI
|
#if INCLUDE_JVMCI
|
||||||
static address deoptimize_for_missing_exception_handler(CompiledMethod* cm);
|
static address deoptimize_for_missing_exception_handler(CompiledMethod* cm);
|
||||||
@ -154,12 +160,8 @@ class Deoptimization : AllStatic {
|
|||||||
// Does the actual work for deoptimizing a single frame
|
// Does the actual work for deoptimizing a single frame
|
||||||
static void deoptimize_single_frame(JavaThread* thread, frame fr, DeoptReason reason);
|
static void deoptimize_single_frame(JavaThread* thread, frame fr, DeoptReason reason);
|
||||||
|
|
||||||
// Helper function to revoke biases of all monitors in frame if UseBiasedLocking
|
|
||||||
// is enabled
|
|
||||||
static void revoke_biases_of_monitors(JavaThread* thread, frame fr, RegisterMap* map);
|
|
||||||
|
|
||||||
#if COMPILER2_OR_JVMCI
|
#if COMPILER2_OR_JVMCI
|
||||||
JVMCI_ONLY(public:)
|
public:
|
||||||
|
|
||||||
// Support for restoring non-escaping objects
|
// Support for restoring non-escaping objects
|
||||||
static bool realloc_objects(JavaThread* thread, frame* fr, RegisterMap* reg_map, GrowableArray<ScopeValue*>* objects, TRAPS);
|
static bool realloc_objects(JavaThread* thread, frame* fr, RegisterMap* reg_map, GrowableArray<ScopeValue*>* objects, TRAPS);
|
||||||
|
@ -64,7 +64,7 @@ class Mutex : public CHeapObj<mtSynchronizer> {
|
|||||||
event,
|
event,
|
||||||
access = event + 1,
|
access = event + 1,
|
||||||
tty = access + 2,
|
tty = access + 2,
|
||||||
special = tty + 1,
|
special = tty + 2,
|
||||||
suspend_resume = special + 1,
|
suspend_resume = special + 1,
|
||||||
oopstorage = suspend_resume + 2,
|
oopstorage = suspend_resume + 2,
|
||||||
leaf = oopstorage + 2,
|
leaf = oopstorage + 2,
|
||||||
|
@ -39,6 +39,7 @@
|
|||||||
// Consider using GCC's __read_mostly.
|
// Consider using GCC's __read_mostly.
|
||||||
|
|
||||||
Mutex* Patching_lock = NULL;
|
Mutex* Patching_lock = NULL;
|
||||||
|
Mutex* CompiledMethod_lock = NULL;
|
||||||
Monitor* SystemDictionary_lock = NULL;
|
Monitor* SystemDictionary_lock = NULL;
|
||||||
Mutex* ProtectionDomainSet_lock = NULL;
|
Mutex* ProtectionDomainSet_lock = NULL;
|
||||||
Mutex* SharedDictionary_lock = NULL;
|
Mutex* SharedDictionary_lock = NULL;
|
||||||
@ -93,7 +94,6 @@ Monitor* BeforeExit_lock = NULL;
|
|||||||
Monitor* Notify_lock = NULL;
|
Monitor* Notify_lock = NULL;
|
||||||
Mutex* ProfilePrint_lock = NULL;
|
Mutex* ProfilePrint_lock = NULL;
|
||||||
Mutex* ExceptionCache_lock = NULL;
|
Mutex* ExceptionCache_lock = NULL;
|
||||||
Mutex* OsrList_lock = NULL;
|
|
||||||
Mutex* NMethodSweeperStats_lock = NULL;
|
Mutex* NMethodSweeperStats_lock = NULL;
|
||||||
#ifndef PRODUCT
|
#ifndef PRODUCT
|
||||||
Mutex* FullGCALot_lock = NULL;
|
Mutex* FullGCALot_lock = NULL;
|
||||||
@ -233,6 +233,7 @@ void mutex_init() {
|
|||||||
def(ClassLoaderDataGraph_lock , PaddedMutex , nonleaf, true, Monitor::_safepoint_check_always);
|
def(ClassLoaderDataGraph_lock , PaddedMutex , nonleaf, true, Monitor::_safepoint_check_always);
|
||||||
|
|
||||||
def(Patching_lock , PaddedMutex , special, true, Monitor::_safepoint_check_never); // used for safepointing and code patching.
|
def(Patching_lock , PaddedMutex , special, true, Monitor::_safepoint_check_never); // used for safepointing and code patching.
|
||||||
|
def(CompiledMethod_lock , PaddedMutex , special-1, true, Monitor::_safepoint_check_never);
|
||||||
def(Service_lock , PaddedMonitor, special, true, Monitor::_safepoint_check_never); // used for service thread operations
|
def(Service_lock , PaddedMonitor, special, true, Monitor::_safepoint_check_never); // used for service thread operations
|
||||||
def(JmethodIdCreation_lock , PaddedMutex , leaf, true, Monitor::_safepoint_check_always); // used for creating jmethodIDs.
|
def(JmethodIdCreation_lock , PaddedMutex , leaf, true, Monitor::_safepoint_check_always); // used for creating jmethodIDs.
|
||||||
|
|
||||||
@ -248,7 +249,6 @@ void mutex_init() {
|
|||||||
def(SymbolArena_lock , PaddedMutex , leaf+2, true, Monitor::_safepoint_check_never);
|
def(SymbolArena_lock , PaddedMutex , leaf+2, true, Monitor::_safepoint_check_never);
|
||||||
def(ProfilePrint_lock , PaddedMutex , leaf, false, Monitor::_safepoint_check_always); // serial profile printing
|
def(ProfilePrint_lock , PaddedMutex , leaf, false, Monitor::_safepoint_check_always); // serial profile printing
|
||||||
def(ExceptionCache_lock , PaddedMutex , leaf, false, Monitor::_safepoint_check_always); // serial profile printing
|
def(ExceptionCache_lock , PaddedMutex , leaf, false, Monitor::_safepoint_check_always); // serial profile printing
|
||||||
def(OsrList_lock , PaddedMutex , leaf, true, Monitor::_safepoint_check_never);
|
|
||||||
def(Debug1_lock , PaddedMutex , leaf, true, Monitor::_safepoint_check_never);
|
def(Debug1_lock , PaddedMutex , leaf, true, Monitor::_safepoint_check_never);
|
||||||
#ifndef PRODUCT
|
#ifndef PRODUCT
|
||||||
def(FullGCALot_lock , PaddedMutex , leaf, false, Monitor::_safepoint_check_always); // a lock to make FullGCALot MT safe
|
def(FullGCALot_lock , PaddedMutex , leaf, false, Monitor::_safepoint_check_always); // a lock to make FullGCALot MT safe
|
||||||
|
@ -32,6 +32,7 @@
|
|||||||
// Mutexes used in the VM.
|
// Mutexes used in the VM.
|
||||||
|
|
||||||
extern Mutex* Patching_lock; // a lock used to guard code patching of compiled code
|
extern Mutex* Patching_lock; // a lock used to guard code patching of compiled code
|
||||||
|
extern Mutex* CompiledMethod_lock; // a lock used to guard a compiled method and OSR queues
|
||||||
extern Monitor* SystemDictionary_lock; // a lock on the system dictionary
|
extern Monitor* SystemDictionary_lock; // a lock on the system dictionary
|
||||||
extern Mutex* ProtectionDomainSet_lock; // a lock on the pd_set list in the system dictionary
|
extern Mutex* ProtectionDomainSet_lock; // a lock on the pd_set list in the system dictionary
|
||||||
extern Mutex* SharedDictionary_lock; // a lock on the CDS shared dictionary
|
extern Mutex* SharedDictionary_lock; // a lock on the CDS shared dictionary
|
||||||
@ -90,7 +91,6 @@ extern Monitor* BeforeExit_lock; // a lock used to guard cleanup
|
|||||||
extern Monitor* Notify_lock; // a lock used to synchronize the start-up of the vm
|
extern Monitor* Notify_lock; // a lock used to synchronize the start-up of the vm
|
||||||
extern Mutex* ProfilePrint_lock; // a lock used to serialize the printing of profiles
|
extern Mutex* ProfilePrint_lock; // a lock used to serialize the printing of profiles
|
||||||
extern Mutex* ExceptionCache_lock; // a lock used to synchronize exception cache updates
|
extern Mutex* ExceptionCache_lock; // a lock used to synchronize exception cache updates
|
||||||
extern Mutex* OsrList_lock; // a lock used to serialize access to OSR queues
|
|
||||||
extern Mutex* NMethodSweeperStats_lock; // a lock used to serialize access to sweeper statistics
|
extern Mutex* NMethodSweeperStats_lock; // a lock used to serialize access to sweeper statistics
|
||||||
|
|
||||||
#ifndef PRODUCT
|
#ifndef PRODUCT
|
||||||
|
@ -2902,7 +2902,12 @@ void AdapterHandlerLibrary::create_native_wrapper(const methodHandle& method) {
|
|||||||
nm = SharedRuntime::generate_native_wrapper(&_masm, method, compile_id, sig_bt, regs, ret_type, critical_entry);
|
nm = SharedRuntime::generate_native_wrapper(&_masm, method, compile_id, sig_bt, regs, ret_type, critical_entry);
|
||||||
|
|
||||||
if (nm != NULL) {
|
if (nm != NULL) {
|
||||||
method->set_code(method, nm);
|
{
|
||||||
|
MutexLocker pl(CompiledMethod_lock, Mutex::_no_safepoint_check_flag);
|
||||||
|
if (nm->make_in_use()) {
|
||||||
|
method->set_code(method, nm);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
DirectiveSet* directive = DirectivesStack::getDefaultDirective(CompileBroker::compiler(CompLevel_simple));
|
DirectiveSet* directive = DirectivesStack::getDefaultDirective(CompileBroker::compiler(CompLevel_simple));
|
||||||
if (directive->PrintAssemblyOption) {
|
if (directive->PrintAssemblyOption) {
|
||||||
|
@ -2894,7 +2894,7 @@ void JavaThread::make_zombies() {
|
|||||||
#endif // PRODUCT
|
#endif // PRODUCT
|
||||||
|
|
||||||
|
|
||||||
void JavaThread::deoptimized_wrt_marked_nmethods() {
|
void JavaThread::deoptimize_marked_methods() {
|
||||||
if (!has_last_Java_frame()) return;
|
if (!has_last_Java_frame()) return;
|
||||||
// BiasedLocking needs an updated RegisterMap for the revoke monitors pass
|
// BiasedLocking needs an updated RegisterMap for the revoke monitors pass
|
||||||
StackFrameStream fst(this, UseBiasedLocking);
|
StackFrameStream fst(this, UseBiasedLocking);
|
||||||
@ -2905,7 +2905,6 @@ void JavaThread::deoptimized_wrt_marked_nmethods() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
// If the caller is a NamedThread, then remember, in the current scope,
|
// If the caller is a NamedThread, then remember, in the current scope,
|
||||||
// the given JavaThread in its _processed_thread field.
|
// the given JavaThread in its _processed_thread field.
|
||||||
class RememberProcessedThread: public StackObj {
|
class RememberProcessedThread: public StackObj {
|
||||||
@ -4638,13 +4637,6 @@ void Threads::metadata_handles_do(void f(Metadata*)) {
|
|||||||
threads_do(&handles_closure);
|
threads_do(&handles_closure);
|
||||||
}
|
}
|
||||||
|
|
||||||
void Threads::deoptimized_wrt_marked_nmethods() {
|
|
||||||
ALL_JAVA_THREADS(p) {
|
|
||||||
p->deoptimized_wrt_marked_nmethods();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
// Get count Java threads that are waiting to enter the specified monitor.
|
// Get count Java threads that are waiting to enter the specified monitor.
|
||||||
GrowableArray<JavaThread*>* Threads::get_pending_threads(ThreadsList * t_list,
|
GrowableArray<JavaThread*>* Threads::get_pending_threads(ThreadsList * t_list,
|
||||||
int count,
|
int count,
|
||||||
|
@ -1886,7 +1886,7 @@ class JavaThread: public Thread {
|
|||||||
void deoptimize();
|
void deoptimize();
|
||||||
void make_zombies();
|
void make_zombies();
|
||||||
|
|
||||||
void deoptimized_wrt_marked_nmethods();
|
void deoptimize_marked_methods();
|
||||||
|
|
||||||
public:
|
public:
|
||||||
// Returns the running thread as a JavaThread
|
// Returns the running thread as a JavaThread
|
||||||
|
@ -445,6 +445,7 @@ void TieredThresholdPolicy::compile(const methodHandle& mh, int bci, CompLevel l
|
|||||||
if (mh->has_compiled_code()) {
|
if (mh->has_compiled_code()) {
|
||||||
mh->code()->make_not_entrant();
|
mh->code()->make_not_entrant();
|
||||||
}
|
}
|
||||||
|
MutexLocker pl(CompiledMethod_lock, Mutex::_no_safepoint_check_flag);
|
||||||
Method::set_code(mh, mh->aot_code());
|
Method::set_code(mh, mh->aot_code());
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -115,18 +115,6 @@ void VM_ClearICs::doit() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
void VM_Deoptimize::doit() {
|
|
||||||
// We do not want any GCs to happen while we are in the middle of this VM operation
|
|
||||||
ResourceMark rm;
|
|
||||||
DeoptimizationMarker dm;
|
|
||||||
|
|
||||||
// Deoptimize all activations depending on marked nmethods
|
|
||||||
Deoptimization::deoptimize_dependents();
|
|
||||||
|
|
||||||
// Make the dependent methods not entrant
|
|
||||||
CodeCache::make_marked_nmethods_not_entrant();
|
|
||||||
}
|
|
||||||
|
|
||||||
void VM_MarkActiveNMethods::doit() {
|
void VM_MarkActiveNMethods::doit() {
|
||||||
NMethodSweeper::mark_active_nmethods();
|
NMethodSweeper::mark_active_nmethods();
|
||||||
}
|
}
|
||||||
|
@ -49,7 +49,6 @@
|
|||||||
template(ClearICs) \
|
template(ClearICs) \
|
||||||
template(ForceSafepoint) \
|
template(ForceSafepoint) \
|
||||||
template(ForceAsyncSafepoint) \
|
template(ForceAsyncSafepoint) \
|
||||||
template(Deoptimize) \
|
|
||||||
template(DeoptimizeFrame) \
|
template(DeoptimizeFrame) \
|
||||||
template(DeoptimizeAll) \
|
template(DeoptimizeAll) \
|
||||||
template(ZombieAll) \
|
template(ZombieAll) \
|
||||||
@ -318,14 +317,6 @@ class VM_GTestExecuteAtSafepoint: public VM_Operation {
|
|||||||
VM_GTestExecuteAtSafepoint() {}
|
VM_GTestExecuteAtSafepoint() {}
|
||||||
};
|
};
|
||||||
|
|
||||||
class VM_Deoptimize: public VM_Operation {
|
|
||||||
public:
|
|
||||||
VM_Deoptimize() {}
|
|
||||||
VMOp_Type type() const { return VMOp_Deoptimize; }
|
|
||||||
void doit();
|
|
||||||
bool allow_nested_vm_operations() const { return true; }
|
|
||||||
};
|
|
||||||
|
|
||||||
class VM_MarkActiveNMethods: public VM_Operation {
|
class VM_MarkActiveNMethods: public VM_Operation {
|
||||||
public:
|
public:
|
||||||
VM_MarkActiveNMethods() {}
|
VM_MarkActiveNMethods() {}
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2006, 2018, Oracle and/or its affiliates. All rights reserved.
|
* Copyright (c) 2006, 2019, Oracle and/or its affiliates. All rights reserved.
|
||||||
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||||
*
|
*
|
||||||
* This code is free software; you can redistribute it and/or modify it
|
* This code is free software; you can redistribute it and/or modify it
|
||||||
@ -33,23 +33,6 @@
|
|||||||
|
|
||||||
#ifdef SOLARIS
|
#ifdef SOLARIS
|
||||||
|
|
||||||
class VM_DeoptimizeTheWorld : public VM_Operation {
|
|
||||||
public:
|
|
||||||
VMOp_Type type() const {
|
|
||||||
return VMOp_DeoptimizeTheWorld;
|
|
||||||
}
|
|
||||||
void doit() {
|
|
||||||
CodeCache::mark_all_nmethods_for_deoptimization();
|
|
||||||
ResourceMark rm;
|
|
||||||
DeoptimizationMarker dm;
|
|
||||||
// Deoptimize all activations depending on marked methods
|
|
||||||
Deoptimization::deoptimize_dependents();
|
|
||||||
|
|
||||||
// Mark the dependent methods non entrant
|
|
||||||
CodeCache::make_marked_nmethods_not_entrant();
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
static void set_bool_flag(const char* name, bool value) {
|
static void set_bool_flag(const char* name, bool value) {
|
||||||
JVMFlag* flag = JVMFlag::find_flag(name);
|
JVMFlag* flag = JVMFlag::find_flag(name);
|
||||||
JVMFlag::boolAtPut(flag, &value, JVMFlag::ATTACH_ON_DEMAND);
|
JVMFlag::boolAtPut(flag, &value, JVMFlag::ATTACH_ON_DEMAND);
|
||||||
@ -74,8 +57,8 @@ void DTrace::enable_dprobes(int probes) {
|
|||||||
|
|
||||||
if (changed) {
|
if (changed) {
|
||||||
// one or more flags changed, need to deoptimize
|
// one or more flags changed, need to deoptimize
|
||||||
VM_DeoptimizeTheWorld op;
|
CodeCache::mark_all_nmethods_for_deoptimization();
|
||||||
VMThread::execute(&op);
|
Deoptimization::deoptimize_all_marked();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -97,8 +80,8 @@ void DTrace::disable_dprobes(int probes) {
|
|||||||
}
|
}
|
||||||
if (changed) {
|
if (changed) {
|
||||||
// one or more flags changed, need to deoptimize
|
// one or more flags changed, need to deoptimize
|
||||||
VM_DeoptimizeTheWorld op;
|
CodeCache::mark_all_nmethods_for_deoptimization();
|
||||||
VMThread::execute(&op);
|
Deoptimization::deoptimize_all_marked();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -0,0 +1,64 @@
|
|||||||
|
/*
|
||||||
|
* Copyright (c) 2019, Oracle and/or its affiliates. All rights reserved.
|
||||||
|
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
|
||||||
|
*
|
||||||
|
* This code is free software; you can redistribute it and/or modify it
|
||||||
|
* under the terms of the GNU General Public License version 2 only, as
|
||||||
|
* published by the Free Software Foundation.
|
||||||
|
*
|
||||||
|
* This code is distributed in the hope that it will be useful, but WITHOUT
|
||||||
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
||||||
|
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
|
||||||
|
* version 2 for more details (a copy is included in the LICENSE file that
|
||||||
|
* accompanied this code).
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License version
|
||||||
|
* 2 along with this work; if not, write to the Free Software Foundation,
|
||||||
|
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
|
||||||
|
*
|
||||||
|
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
|
||||||
|
* or visit www.oracle.com if you need additional information or have any
|
||||||
|
* questions.
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
* @test UnexpectedDeoptimizationAllTest
|
||||||
|
* @key stress
|
||||||
|
* @summary stressing code cache by forcing unexpected deoptimizations of all methods
|
||||||
|
* @library /test/lib /
|
||||||
|
* @modules java.base/jdk.internal.misc
|
||||||
|
* java.management
|
||||||
|
*
|
||||||
|
* @build sun.hotspot.WhiteBox compiler.codecache.stress.Helper compiler.codecache.stress.TestCaseImpl
|
||||||
|
* @run driver ClassFileInstaller sun.hotspot.WhiteBox
|
||||||
|
* sun.hotspot.WhiteBox$WhiteBoxPermission
|
||||||
|
* @run main/othervm -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions
|
||||||
|
* -XX:+WhiteBoxAPI -XX:-DeoptimizeRandom
|
||||||
|
* -XX:CompileCommand=dontinline,compiler.codecache.stress.Helper$TestCase::method
|
||||||
|
* -XX:-SegmentedCodeCache
|
||||||
|
* compiler.codecache.stress.UnexpectedDeoptimizationAllTest
|
||||||
|
* @run main/othervm -Xbootclasspath/a:. -XX:+UnlockDiagnosticVMOptions
|
||||||
|
* -XX:+WhiteBoxAPI -XX:-DeoptimizeRandom
|
||||||
|
* -XX:CompileCommand=dontinline,compiler.codecache.stress.Helper$TestCase::method
|
||||||
|
* -XX:+SegmentedCodeCache
|
||||||
|
* compiler.codecache.stress.UnexpectedDeoptimizationAllTest
|
||||||
|
*/
|
||||||
|
|
||||||
|
package compiler.codecache.stress;
|
||||||
|
|
||||||
|
public class UnexpectedDeoptimizationAllTest implements Runnable {
|
||||||
|
|
||||||
|
public static void main(String[] args) {
|
||||||
|
new CodeCacheStressRunner(new UnexpectedDeoptimizationAllTest()).runTest();
|
||||||
|
}
|
||||||
|
|
||||||
|
@Override
|
||||||
|
public void run() {
|
||||||
|
Helper.WHITE_BOX.deoptimizeAll();
|
||||||
|
try {
|
||||||
|
Thread.sleep(10);
|
||||||
|
} catch (Exception e) {
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user