8314225: SIGSEGV in JavaThread::is_lock_owned
Reviewed-by: dlong, dholmes
This commit is contained in:
parent
ac86f59e4f
commit
ad0b54d429
@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 2014, 2023, Oracle and/or its affiliates. All rights reserved.
|
* Copyright (c) 2014, 2024, 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
|
||||||
@ -248,12 +248,6 @@ bool ReferenceToThreadRootClosure::do_thread_stack_detailed(JavaThread* jt) {
|
|||||||
ReferenceLocateClosure rcl(_callback, OldObjectRoot::_threads, OldObjectRoot::_stack_variable, jt);
|
ReferenceLocateClosure rcl(_callback, OldObjectRoot::_threads, OldObjectRoot::_stack_variable, jt);
|
||||||
|
|
||||||
if (jt->has_last_Java_frame()) {
|
if (jt->has_last_Java_frame()) {
|
||||||
// Traverse the monitor chunks
|
|
||||||
MonitorChunk* chunk = jt->monitor_chunks();
|
|
||||||
for (; chunk != nullptr; chunk = chunk->next()) {
|
|
||||||
chunk->oops_do(&rcl);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (rcl.complete()) {
|
if (rcl.complete()) {
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
@ -1736,7 +1736,7 @@ void Deoptimization::pop_frames_failed_reallocs(JavaThread* thread, vframeArray*
|
|||||||
ObjectSynchronizer::exit(src->obj(), src->lock(), thread);
|
ObjectSynchronizer::exit(src->obj(), src->lock(), thread);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
array->element(i)->free_monitors(thread);
|
array->element(i)->free_monitors();
|
||||||
#ifdef ASSERT
|
#ifdef ASSERT
|
||||||
array->element(i)->set_removed_monitors();
|
array->element(i)->set_removed_monitors();
|
||||||
#endif
|
#endif
|
||||||
|
@ -430,8 +430,6 @@ JavaThread::JavaThread() :
|
|||||||
_active_handles(nullptr),
|
_active_handles(nullptr),
|
||||||
_free_handle_block(nullptr),
|
_free_handle_block(nullptr),
|
||||||
|
|
||||||
_monitor_chunks(nullptr),
|
|
||||||
|
|
||||||
_suspend_flags(0),
|
_suspend_flags(0),
|
||||||
|
|
||||||
_thread_state(_thread_new),
|
_thread_state(_thread_new),
|
||||||
@ -1050,13 +1048,7 @@ JavaThread* JavaThread::active() {
|
|||||||
|
|
||||||
bool JavaThread::is_lock_owned(address adr) const {
|
bool JavaThread::is_lock_owned(address adr) const {
|
||||||
assert(LockingMode != LM_LIGHTWEIGHT, "should not be called with new lightweight locking");
|
assert(LockingMode != LM_LIGHTWEIGHT, "should not be called with new lightweight locking");
|
||||||
if (Thread::is_lock_owned(adr)) return true;
|
return is_in_full_stack(adr);
|
||||||
|
|
||||||
for (MonitorChunk* chunk = monitor_chunks(); chunk != nullptr; chunk = chunk->next()) {
|
|
||||||
if (chunk->contains(adr)) return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
oop JavaThread::exception_oop() const {
|
oop JavaThread::exception_oop() const {
|
||||||
@ -1067,22 +1059,6 @@ void JavaThread::set_exception_oop(oop o) {
|
|||||||
Atomic::store(&_exception_oop, o);
|
Atomic::store(&_exception_oop, o);
|
||||||
}
|
}
|
||||||
|
|
||||||
void JavaThread::add_monitor_chunk(MonitorChunk* chunk) {
|
|
||||||
chunk->set_next(monitor_chunks());
|
|
||||||
set_monitor_chunks(chunk);
|
|
||||||
}
|
|
||||||
|
|
||||||
void JavaThread::remove_monitor_chunk(MonitorChunk* chunk) {
|
|
||||||
guarantee(monitor_chunks() != nullptr, "must be non empty");
|
|
||||||
if (monitor_chunks() == chunk) {
|
|
||||||
set_monitor_chunks(chunk->next());
|
|
||||||
} else {
|
|
||||||
MonitorChunk* prev = monitor_chunks();
|
|
||||||
while (prev->next() != chunk) prev = prev->next();
|
|
||||||
prev->set_next(chunk->next());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
void JavaThread::handle_special_runtime_exit_condition() {
|
void JavaThread::handle_special_runtime_exit_condition() {
|
||||||
if (is_obj_deopt_suspend()) {
|
if (is_obj_deopt_suspend()) {
|
||||||
frame_anchor()->make_walkable();
|
frame_anchor()->make_walkable();
|
||||||
@ -1408,13 +1384,6 @@ void JavaThread::oops_do_no_frames(OopClosure* f, NMethodClosure* cf) {
|
|||||||
|
|
||||||
DEBUG_ONLY(verify_frame_info();)
|
DEBUG_ONLY(verify_frame_info();)
|
||||||
|
|
||||||
if (has_last_Java_frame()) {
|
|
||||||
// Traverse the monitor chunks
|
|
||||||
for (MonitorChunk* chunk = monitor_chunks(); chunk != nullptr; chunk = chunk->next()) {
|
|
||||||
chunk->oops_do(f);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
assert(vframe_array_head() == nullptr, "deopt in progress at a safepoint!");
|
assert(vframe_array_head() == nullptr, "deopt in progress at a safepoint!");
|
||||||
// If we have deferred set_locals there might be oops waiting to be
|
// If we have deferred set_locals there might be oops waiting to be
|
||||||
// written
|
// written
|
||||||
|
@ -193,10 +193,6 @@ class JavaThread: public Thread {
|
|||||||
void pop_jni_handle_block();
|
void pop_jni_handle_block();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
MonitorChunk* _monitor_chunks; // Contains the off stack monitors
|
|
||||||
// allocated during deoptimization
|
|
||||||
// and by JNI_MonitorEnter/Exit
|
|
||||||
|
|
||||||
enum SuspendFlags {
|
enum SuspendFlags {
|
||||||
// NOTE: avoid using the sign-bit as cc generates different test code
|
// NOTE: avoid using the sign-bit as cc generates different test code
|
||||||
// when the sign-bit is used, and sometimes incorrectly - see CR 6398077
|
// when the sign-bit is used, and sometimes incorrectly - see CR 6398077
|
||||||
@ -679,7 +675,7 @@ private:
|
|||||||
return (_suspend_flags & (_obj_deopt JFR_ONLY(| _trace_flag))) != 0;
|
return (_suspend_flags & (_obj_deopt JFR_ONLY(| _trace_flag))) != 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Fast-locking support
|
// Stack-locking support (not for LM_LIGHTWEIGHT)
|
||||||
bool is_lock_owned(address adr) const;
|
bool is_lock_owned(address adr) const;
|
||||||
|
|
||||||
// Accessors for vframe array top
|
// Accessors for vframe array top
|
||||||
@ -881,13 +877,7 @@ private:
|
|||||||
int depth_first_number() { return _depth_first_number; }
|
int depth_first_number() { return _depth_first_number; }
|
||||||
void set_depth_first_number(int dfn) { _depth_first_number = dfn; }
|
void set_depth_first_number(int dfn) { _depth_first_number = dfn; }
|
||||||
|
|
||||||
private:
|
|
||||||
void set_monitor_chunks(MonitorChunk* monitor_chunks) { _monitor_chunks = monitor_chunks; }
|
|
||||||
|
|
||||||
public:
|
public:
|
||||||
MonitorChunk* monitor_chunks() const { return _monitor_chunks; }
|
|
||||||
void add_monitor_chunk(MonitorChunk* chunk);
|
|
||||||
void remove_monitor_chunk(MonitorChunk* chunk);
|
|
||||||
bool in_deopt_handler() const { return _in_deopt_handler > 0; }
|
bool in_deopt_handler() const { return _in_deopt_handler > 0; }
|
||||||
void inc_in_deopt_handler() { _in_deopt_handler++; }
|
void inc_in_deopt_handler() { _in_deopt_handler++; }
|
||||||
void dec_in_deopt_handler() {
|
void dec_in_deopt_handler() {
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 1997, 2023, Oracle and/or its affiliates. All rights reserved.
|
* Copyright (c) 1997, 2024, 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
|
||||||
@ -30,7 +30,6 @@
|
|||||||
MonitorChunk::MonitorChunk(int number_on_monitors) {
|
MonitorChunk::MonitorChunk(int number_on_monitors) {
|
||||||
_number_of_monitors = number_on_monitors;
|
_number_of_monitors = number_on_monitors;
|
||||||
_monitors = NEW_C_HEAP_ARRAY(BasicObjectLock, number_on_monitors, mtSynchronizer);
|
_monitors = NEW_C_HEAP_ARRAY(BasicObjectLock, number_on_monitors, mtSynchronizer);
|
||||||
_next = nullptr;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 1997, 2022, Oracle and/or its affiliates. All rights reserved.
|
* Copyright (c) 1997, 2024, 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
|
||||||
@ -38,23 +38,17 @@ class MonitorChunk: public CHeapObj<mtSynchronizer> {
|
|||||||
int _number_of_monitors;
|
int _number_of_monitors;
|
||||||
BasicObjectLock* _monitors;
|
BasicObjectLock* _monitors;
|
||||||
BasicObjectLock* monitors() const { return _monitors; }
|
BasicObjectLock* monitors() const { return _monitors; }
|
||||||
MonitorChunk* _next;
|
|
||||||
public:
|
public:
|
||||||
// Constructor
|
// Constructor
|
||||||
MonitorChunk(int number_on_monitors);
|
MonitorChunk(int number_on_monitors);
|
||||||
~MonitorChunk();
|
~MonitorChunk();
|
||||||
|
|
||||||
// link operations
|
|
||||||
MonitorChunk* next() const { return _next; }
|
|
||||||
void set_next(MonitorChunk* next) { _next = next; }
|
|
||||||
|
|
||||||
// Returns the number of monitors
|
// Returns the number of monitors
|
||||||
int number_of_monitors() const { return _number_of_monitors; }
|
int number_of_monitors() const { return _number_of_monitors; }
|
||||||
|
|
||||||
// Returns the index'th monitor
|
// Returns the index'th monitor
|
||||||
BasicObjectLock* at(int index) { assert(index >= 0 && index < number_of_monitors(), "out of bounds check"); return &monitors()[index]; }
|
BasicObjectLock* at(int index) { assert(index >= 0 && index < number_of_monitors(), "out of bounds check"); return &monitors()[index]; }
|
||||||
|
|
||||||
|
|
||||||
// Memory management
|
// Memory management
|
||||||
void oops_do(OopClosure* f);
|
void oops_do(OopClosure* f);
|
||||||
|
|
||||||
|
@ -1056,7 +1056,9 @@ intptr_t ObjectSynchronizer::FastHashCode(Thread* current, oop obj) {
|
|||||||
}
|
}
|
||||||
// Fall thru so we only have one place that installs the hash in
|
// Fall thru so we only have one place that installs the hash in
|
||||||
// the ObjectMonitor.
|
// the ObjectMonitor.
|
||||||
} else if (LockingMode == LM_LEGACY && mark.has_locker() && current->is_lock_owned((address)mark.locker())) {
|
} else if (LockingMode == LM_LEGACY && mark.has_locker()
|
||||||
|
&& current->is_Java_thread()
|
||||||
|
&& JavaThread::cast(current)->is_lock_owned((address)mark.locker())) {
|
||||||
// This is a stack-lock owned by the calling thread so fetch the
|
// This is a stack-lock owned by the calling thread so fetch the
|
||||||
// displaced markWord from the BasicLock on the stack.
|
// displaced markWord from the BasicLock on the stack.
|
||||||
temp = mark.displaced_mark_helper();
|
temp = mark.displaced_mark_helper();
|
||||||
|
@ -527,16 +527,6 @@ void Thread::print_owned_locks_on(outputStream* st) const {
|
|||||||
}
|
}
|
||||||
#endif // ASSERT
|
#endif // ASSERT
|
||||||
|
|
||||||
// We had to move these methods here, because vm threads get into ObjectSynchronizer::enter
|
|
||||||
// However, there is a note in JavaThread::is_lock_owned() about the VM threads not being
|
|
||||||
// used for compilation in the future. If that change is made, the need for these methods
|
|
||||||
// should be revisited, and they should be removed if possible.
|
|
||||||
|
|
||||||
bool Thread::is_lock_owned(address adr) const {
|
|
||||||
assert(LockingMode != LM_LIGHTWEIGHT, "should not be called with new lightweight locking");
|
|
||||||
return is_in_full_stack(adr);
|
|
||||||
}
|
|
||||||
|
|
||||||
bool Thread::set_as_starting_thread() {
|
bool Thread::set_as_starting_thread() {
|
||||||
assert(_starting_thread == nullptr, "already initialized: "
|
assert(_starting_thread == nullptr, "already initialized: "
|
||||||
"_starting_thread=" INTPTR_FORMAT, p2i(_starting_thread));
|
"_starting_thread=" INTPTR_FORMAT, p2i(_starting_thread));
|
||||||
|
@ -475,9 +475,6 @@ class Thread: public ThreadShadow {
|
|||||||
}
|
}
|
||||||
|
|
||||||
public:
|
public:
|
||||||
// Used by fast lock support
|
|
||||||
virtual bool is_lock_owned(address adr) const;
|
|
||||||
|
|
||||||
// Check if address is within the given range of this thread's
|
// Check if address is within the given range of this thread's
|
||||||
// stack: stack_base() > adr >= limit
|
// stack: stack_base() > adr >= limit
|
||||||
bool is_in_stack_range_incl(address adr, address limit) const {
|
bool is_in_stack_range_incl(address adr, address limit) const {
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 1997, 2023, Oracle and/or its affiliates. All rights reserved.
|
* Copyright (c) 1997, 2024, 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
|
||||||
@ -37,6 +37,7 @@
|
|||||||
#include "runtime/handles.inline.hpp"
|
#include "runtime/handles.inline.hpp"
|
||||||
#include "runtime/monitorChunk.hpp"
|
#include "runtime/monitorChunk.hpp"
|
||||||
#include "runtime/sharedRuntime.hpp"
|
#include "runtime/sharedRuntime.hpp"
|
||||||
|
#include "runtime/synchronizer.hpp"
|
||||||
#include "runtime/vframe.hpp"
|
#include "runtime/vframe.hpp"
|
||||||
#include "runtime/vframeArray.hpp"
|
#include "runtime/vframeArray.hpp"
|
||||||
#include "runtime/vframe_hp.hpp"
|
#include "runtime/vframe_hp.hpp"
|
||||||
@ -48,11 +49,10 @@
|
|||||||
|
|
||||||
int vframeArrayElement:: bci(void) const { return (_bci == SynchronizationEntryBCI ? 0 : _bci); }
|
int vframeArrayElement:: bci(void) const { return (_bci == SynchronizationEntryBCI ? 0 : _bci); }
|
||||||
|
|
||||||
void vframeArrayElement::free_monitors(JavaThread* jt) {
|
void vframeArrayElement::free_monitors() {
|
||||||
if (_monitors != nullptr) {
|
if (_monitors != nullptr) {
|
||||||
MonitorChunk* chunk = _monitors;
|
MonitorChunk* chunk = _monitors;
|
||||||
_monitors = nullptr;
|
_monitors = nullptr;
|
||||||
jt->remove_monitor_chunk(chunk);
|
|
||||||
delete chunk;
|
delete chunk;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -72,7 +72,7 @@ void vframeArrayElement::fill_in(compiledVFrame* vf, bool realloc_failures) {
|
|||||||
int index;
|
int index;
|
||||||
|
|
||||||
{
|
{
|
||||||
Thread* current_thread = Thread::current();
|
JavaThread* current_thread = JavaThread::current();
|
||||||
ResourceMark rm(current_thread);
|
ResourceMark rm(current_thread);
|
||||||
HandleMark hm(current_thread);
|
HandleMark hm(current_thread);
|
||||||
|
|
||||||
@ -85,7 +85,6 @@ void vframeArrayElement::fill_in(compiledVFrame* vf, bool realloc_failures) {
|
|||||||
|
|
||||||
// Allocate monitor chunk
|
// Allocate monitor chunk
|
||||||
_monitors = new MonitorChunk(list->length());
|
_monitors = new MonitorChunk(list->length());
|
||||||
vf->thread()->add_monitor_chunk(_monitors);
|
|
||||||
|
|
||||||
// Migrate the BasicLocks from the stack to the monitor chunk
|
// Migrate the BasicLocks from the stack to the monitor chunk
|
||||||
for (index = 0; index < list->length(); index++) {
|
for (index = 0; index < list->length(); index++) {
|
||||||
@ -95,9 +94,16 @@ void vframeArrayElement::fill_in(compiledVFrame* vf, bool realloc_failures) {
|
|||||||
if (monitor->owner_is_scalar_replaced()) {
|
if (monitor->owner_is_scalar_replaced()) {
|
||||||
dest->set_obj(nullptr);
|
dest->set_obj(nullptr);
|
||||||
} else {
|
} else {
|
||||||
assert(monitor->owner() == nullptr || !monitor->owner()->is_unlocked(), "object must be null or locked");
|
assert(monitor->owner() != nullptr, "monitor owner must not be null");
|
||||||
|
assert(!monitor->owner()->is_unlocked(), "monitor must be locked");
|
||||||
dest->set_obj(monitor->owner());
|
dest->set_obj(monitor->owner());
|
||||||
|
assert(ObjectSynchronizer::current_thread_holds_lock(current_thread, Handle(current_thread, dest->obj())),
|
||||||
|
"should be held, before move_to");
|
||||||
|
|
||||||
monitor->lock()->move_to(monitor->owner(), dest->lock());
|
monitor->lock()->move_to(monitor->owner(), dest->lock());
|
||||||
|
|
||||||
|
assert(ObjectSynchronizer::current_thread_holds_lock(current_thread, Handle(current_thread, dest->obj())),
|
||||||
|
"should be held, after move_to");
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -308,7 +314,11 @@ void vframeArrayElement::unpack_on_stack(int caller_actual_parameters,
|
|||||||
top = iframe()->previous_monitor_in_interpreter_frame(top);
|
top = iframe()->previous_monitor_in_interpreter_frame(top);
|
||||||
BasicObjectLock* src = _monitors->at(index);
|
BasicObjectLock* src = _monitors->at(index);
|
||||||
top->set_obj(src->obj());
|
top->set_obj(src->obj());
|
||||||
|
assert(src->obj() != nullptr || ObjectSynchronizer::current_thread_holds_lock(thread, Handle(thread, src->obj())),
|
||||||
|
"should be held, before move_to");
|
||||||
src->lock()->move_to(src->obj(), top->lock());
|
src->lock()->move_to(src->obj(), top->lock());
|
||||||
|
assert(src->obj() != nullptr || ObjectSynchronizer::current_thread_holds_lock(thread, Handle(thread, src->obj())),
|
||||||
|
"should be held, after move_to");
|
||||||
}
|
}
|
||||||
if (ProfileInterpreter) {
|
if (ProfileInterpreter) {
|
||||||
iframe()->interpreter_frame_set_mdp(0); // clear out the mdp.
|
iframe()->interpreter_frame_set_mdp(0); // clear out the mdp.
|
||||||
@ -649,9 +659,8 @@ void vframeArray::unpack_to_stack(frame &unpack_frame, int exec_mode, int caller
|
|||||||
}
|
}
|
||||||
|
|
||||||
void vframeArray::deallocate_monitor_chunks() {
|
void vframeArray::deallocate_monitor_chunks() {
|
||||||
JavaThread* jt = JavaThread::current();
|
|
||||||
for (int index = 0; index < frames(); index++ ) {
|
for (int index = 0; index < frames(); index++ ) {
|
||||||
element(index)->free_monitors(jt);
|
element(index)->free_monitors();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
/*
|
/*
|
||||||
* Copyright (c) 1997, 2021, Oracle and/or its affiliates. All rights reserved.
|
* Copyright (c) 1997, 2024, 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
|
||||||
@ -77,7 +77,7 @@ class vframeArrayElement {
|
|||||||
|
|
||||||
MonitorChunk* monitors(void) const { return _monitors; }
|
MonitorChunk* monitors(void) const { return _monitors; }
|
||||||
|
|
||||||
void free_monitors(JavaThread* jt);
|
void free_monitors();
|
||||||
|
|
||||||
StackValueCollection* locals(void) const { return _locals; }
|
StackValueCollection* locals(void) const { return _locals; }
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user