8256641: CDS VM operations do not lock the heap

Reviewed-by: kbarrett, iklam
This commit is contained in:
Thomas Schatzl 2020-12-11 18:14:37 +00:00
parent 58dca9253d
commit bacf22b907
10 changed files with 85 additions and 35 deletions

View File

@ -472,6 +472,7 @@ bool G1HeapVerifier::should_verify(G1VerifyType type) {
void G1HeapVerifier::verify(VerifyOption vo) {
assert_at_safepoint_on_vm_thread();
assert(Heap_lock->is_locked(), "heap must be locked");
log_debug(gc, verify)("Roots");
VerifyRootsClosure rootsCl(vo);

View File

@ -46,6 +46,20 @@
#include "gc/g1/g1Policy.hpp"
#endif // INCLUDE_G1GC
bool VM_GC_Sync_Operation::doit_prologue() {
Heap_lock->lock();
return true;
}
void VM_GC_Sync_Operation::doit_epilogue() {
Heap_lock->unlock();
}
void VM_Verify::doit() {
Universe::heap()->prepare_for_verify();
Universe::verify();
}
VM_GC_Operation::~VM_GC_Operation() {
CollectedHeap* ch = Universe::heap();
ch->soft_ref_policy()->set_all_soft_refs_clear(false);
@ -94,8 +108,7 @@ bool VM_GC_Operation::doit_prologue() {
proper_unit_for_byte_size(NewSize)));
}
// If the GC count has changed someone beat us to the collection
Heap_lock->lock();
VM_GC_Sync_Operation::doit_prologue();
// Check invocations
if (skip_operation()) {
@ -116,7 +129,7 @@ void VM_GC_Operation::doit_epilogue() {
if (Universe::has_reference_pending_list()) {
Heap_lock->notify_all();
}
Heap_lock->unlock();
VM_GC_Sync_Operation::doit_epilogue();
}
bool VM_GC_HeapInspection::skip_operation() const {

View File

@ -39,17 +39,27 @@
// a set of operations (VM_Operation) related to GC.
//
// VM_Operation
// VM_GC_Sync_Operation
// VM_GC_Operation
// VM_GC_HeapInspection
// VM_GenCollectFull
// VM_GenCollectFullConcurrent
// VM_ParallelGCSystemGC
// VM_CollectForAllocation
// VM_GenCollectForAllocation
// VM_ParallelGCFailedAllocation
// VM_GC_HeapInspection
// VM_PopulateDynamicDumpSharedSpace
// VM_GenCollectFull
// VM_GenCollectFullConcurrent
// VM_ParallelGCSystemGC
// VM_CollectForAllocation
// VM_GenCollectForAllocation
// VM_ParallelGCFailedAllocation
// VM_Verify
// VM_PopulateDumpSharedSpace
//
// VM_GC_Sync_Operation
// - implements only synchronization with other VM operations of the
// same kind using the Heap_lock, not actually doing a GC.
//
// VM_GC_Operation
// - implements methods common to all classes in the hierarchy:
// prevents multiple gc requests and manages lock on heap;
// - implements methods common to all operations that perform garbage collections,
// checking that the VM is in a state to do GC and preventing multiple GC
// requests.
//
// VM_GC_HeapInspection
// - prints class histogram on SIGBREAK if PrintClassHistogram
@ -68,11 +78,37 @@
// - these operations preform full collection of heaps of
// different kind
//
// VM_Verify
// - verifies the heap
//
// VM_PopulateDynamicDumpSharedSpace
// - populates the CDS archive area with the information from the archive file.
//
// VM_PopulateDumpSharedSpace
// - creates the CDS archive
//
class VM_GC_Operation: public VM_Operation {
class VM_GC_Sync_Operation : public VM_Operation {
public:
VM_GC_Sync_Operation() : VM_Operation() { }
// Acquires the Heap_lock.
virtual bool doit_prologue();
// Releases the Heap_lock.
virtual void doit_epilogue();
};
class VM_Verify : public VM_GC_Sync_Operation {
public:
VMOp_Type type() const { return VMOp_Verify; }
void doit();
};
class VM_GC_Operation: public VM_GC_Sync_Operation {
protected:
uint _gc_count_before; // gc count before acquiring PLL
uint _full_gc_count_before; // full gc count before acquiring PLL
uint _gc_count_before; // gc count before acquiring the Heap_lock
uint _full_gc_count_before; // full gc count before acquiring the Heap_lock
bool _full; // whether a "full" collection
bool _prologue_succeeded; // whether doit_prologue succeeded
GCCause::Cause _gc_cause; // the putative cause for this gc op
@ -84,7 +120,7 @@ class VM_GC_Operation: public VM_Operation {
VM_GC_Operation(uint gc_count_before,
GCCause::Cause _cause,
uint full_gc_count_before = 0,
bool full = false) {
bool full = false) : VM_GC_Sync_Operation() {
_full = full;
_prologue_succeeded = false;
_gc_count_before = gc_count_before;
@ -106,9 +142,10 @@ class VM_GC_Operation: public VM_Operation {
}
~VM_GC_Operation();
// Acquire the reference synchronization lock
// Acquire the Heap_lock and determine if this VM operation should be executed
// (i.e. not skipped). Return this result, and also store it in _prologue_succeeded.
virtual bool doit_prologue();
// Do notifyAll (if needed) and release held lock
// Notify the Heap_lock if needed and release it.
virtual void doit_epilogue();
virtual bool allow_nested_vm_operations() const { return true; }

View File

@ -24,6 +24,7 @@
#include "precompiled.hpp"
#include "gc/shared/gcId.hpp"
#include "gc/shared/gcLocker.hpp"
#include "gc/shared/gcVMOperations.hpp"
#include "gc/shared/isGCActiveMark.hpp"
#include "gc/z/zBreakpoint.hpp"
#include "gc/z/zCollectedHeap.hpp"

View File

@ -27,6 +27,7 @@
#include "classfile/classLoaderData.inline.hpp"
#include "classfile/symbolTable.hpp"
#include "classfile/systemDictionaryShared.hpp"
#include "gc/shared/gcVMOperations.hpp"
#include "logging/log.hpp"
#include "memory/archiveBuilder.hpp"
#include "memory/archiveUtils.inline.hpp"
@ -541,10 +542,10 @@ void DynamicArchiveBuilder::write_archive(char* serialized_data) {
log_info(cds, dynamic)("%d klasses; %d symbols", num_klasses, num_symbols);
}
class VM_PopulateDynamicDumpSharedSpace: public VM_Operation {
class VM_PopulateDynamicDumpSharedSpace: public VM_GC_Sync_Operation {
DynamicArchiveBuilder* _builder;
public:
VM_PopulateDynamicDumpSharedSpace(DynamicArchiveBuilder* builder) : _builder(builder) {}
VM_PopulateDynamicDumpSharedSpace(DynamicArchiveBuilder* builder) : VM_GC_Sync_Operation(), _builder(builder) {}
VMOp_Type type() const { return VMOp_PopulateDumpSharedSpace; }
void doit() {
ResourceMark rm;

View File

@ -33,6 +33,7 @@
#include "classfile/systemDictionaryShared.hpp"
#include "classfile/vmSymbols.hpp"
#include "gc/shared/gcLocker.hpp"
#include "gc/shared/gcVMOperations.hpp"
#include "logging/log.hpp"
#include "logging/logMessage.hpp"
#include "logging/logStream.hpp"
@ -655,8 +656,10 @@ static void verify_the_heap(Klass* k, const char* which) {
ResourceMark rm;
log_info(cds, heap)("Verify heap %s initializing static field(s) in %s",
which, k->external_name());
VM_Verify verify_op;
VMThread::execute(&verify_op);
if (!FLAG_IS_DEFAULT(VerifyArchivedFields)) {
// If VerifyArchivedFields has a non-default value (e.g., specified on the command-line), do
// more expensive checks.

View File

@ -37,6 +37,7 @@
#include "classfile/systemDictionaryShared.hpp"
#include "classfile/vmSymbols.hpp"
#include "code/codeCache.hpp"
#include "gc/shared/gcVMOperations.hpp"
#include "interpreter/abstractInterpreter.hpp"
#include "interpreter/bytecodeStream.hpp"
#include "interpreter/bytecodes.hpp"
@ -577,7 +578,7 @@ void MetaspaceShared::rewrite_nofast_bytecodes_and_calculate_fingerprints(Thread
}
}
class VM_PopulateDumpSharedSpace: public VM_Operation {
class VM_PopulateDumpSharedSpace : public VM_GC_Operation {
private:
GrowableArray<MemRegion> *_closed_archive_heap_regions;
GrowableArray<MemRegion> *_open_archive_heap_regions;
@ -602,6 +603,12 @@ private:
public:
VM_PopulateDumpSharedSpace() : VM_GC_Operation(0, /* total collections, ignored */
GCCause::_archive_time_gc)
{ }
bool skip_operation() const { return false; }
VMOp_Type type() const { return VMOp_PopulateDumpSharedSpace; }
void doit(); // outline because gdb sucks
bool allow_nested_vm_operations() const { return true; }
@ -1085,8 +1092,6 @@ void MetaspaceShared::preload_and_dump(TRAPS) {
#endif
VM_PopulateDumpSharedSpace op;
MutexLocker ml(THREAD, HeapShared::is_heap_object_archiving_allowed() ?
Heap_lock : NULL); // needed by HeapShared::run_gc()
VMThread::execute(&op);
}
}

View File

@ -38,6 +38,7 @@
#include "gc/shared/barrierSet.hpp"
#include "gc/shared/gcId.hpp"
#include "gc/shared/gcLocker.inline.hpp"
#include "gc/shared/gcVMOperations.hpp"
#include "gc/shared/oopStorage.hpp"
#include "gc/shared/oopStorageSet.hpp"
#include "gc/shared/workgroup.hpp"

View File

@ -156,11 +156,6 @@ void VM_ZombieAll::doit() {
#endif // !PRODUCT
void VM_Verify::doit() {
Universe::heap()->prepare_for_verify();
Universe::verify();
}
bool VM_PrintThreads::doit_prologue() {
// Get Heap_lock if concurrent locks will be dumped
if (_print_concurrent_locks) {

View File

@ -277,13 +277,6 @@ class VM_ZombieAll: public VM_Operation {
};
#endif // PRODUCT
class VM_Verify: public VM_Operation {
public:
VMOp_Type type() const { return VMOp_Verify; }
void doit();
};
class VM_PrintThreads: public VM_Operation {
private:
outputStream* _out;