Skip to content

Commit 06e6b1f

Browse files
committedApr 9, 2021
8259242: Remove ProtectionDomainSet_lock
Reviewed-by: dholmes, pchilanomate
1 parent 9bb1863 commit 06e6b1f

File tree

10 files changed

+129
-82
lines changed

10 files changed

+129
-82
lines changed
 

‎src/hotspot/share/classfile/dictionary.cpp

+45-38
Original file line numberDiff line numberDiff line change
@@ -44,6 +44,7 @@
4444
#include "runtime/javaCalls.hpp"
4545
#include "runtime/mutexLocker.hpp"
4646
#include "runtime/safepointVerifiers.hpp"
47+
#include "utilities/growableArray.hpp"
4748
#include "utilities/hashtable.inline.hpp"
4849

4950
// Optimization: if any dictionary needs resizing, we set this flag,
@@ -78,20 +79,19 @@ Dictionary::~Dictionary() {
7879

7980
DictionaryEntry* Dictionary::new_entry(unsigned int hash, InstanceKlass* klass) {
8081
DictionaryEntry* entry = (DictionaryEntry*)Hashtable<InstanceKlass*, mtClass>::new_entry(hash, klass);
81-
entry->set_pd_set(NULL);
82+
entry->release_set_pd_set(NULL);
8283
assert(klass->is_instance_klass(), "Must be");
8384
return entry;
8485
}
8586

86-
8787
void Dictionary::free_entry(DictionaryEntry* entry) {
8888
// avoid recursion when deleting linked list
8989
// pd_set is accessed during a safepoint.
9090
// This doesn't require a lock because nothing is reading this
9191
// entry anymore. The ClassLoader is dead.
92-
while (entry->pd_set() != NULL) {
93-
ProtectionDomainEntry* to_delete = entry->pd_set();
94-
entry->set_pd_set(to_delete->next());
92+
while (entry->pd_set_acquire() != NULL) {
93+
ProtectionDomainEntry* to_delete = entry->pd_set_acquire();
94+
entry->release_set_pd_set(to_delete->next_acquire());
9595
delete to_delete;
9696
}
9797
BasicHashtable<mtClass>::free_entry(entry);
@@ -141,15 +141,26 @@ bool DictionaryEntry::is_valid_protection_domain(Handle protection_domain) {
141141
: contains_protection_domain(protection_domain());
142142
}
143143

144+
// Reading the pd_set on each DictionaryEntry is lock free and cannot safepoint.
145+
// Adding and deleting entries is under the SystemDictionary_lock
146+
// Deleting unloaded entries on ClassLoaderData for dictionaries that are not unloaded
147+
// is a three step process:
148+
// moving the entries to a separate list, handshake to wait for
149+
// readers to complete (see NSV here), and then actually deleting the entries.
150+
// Deleting entries is done by the ServiceThread when triggered by class unloading.
151+
144152
bool DictionaryEntry::contains_protection_domain(oop protection_domain) const {
153+
assert(Thread::current()->is_Java_thread() || SafepointSynchronize::is_at_safepoint(),
154+
"can only be called by a JavaThread or at safepoint");
155+
// This cannot safepoint while reading the protection domain set.
156+
NoSafepointVerifier nsv;
145157
#ifdef ASSERT
146158
if (protection_domain == instance_klass()->protection_domain()) {
147-
MutexLocker ml(ProtectionDomainSet_lock, Mutex::_no_safepoint_check_flag);
148159
// Ensure this doesn't show up in the pd_set (invariant)
149160
bool in_pd_set = false;
150-
for (ProtectionDomainEntry* current = pd_set();
161+
for (ProtectionDomainEntry* current = pd_set_acquire();
151162
current != NULL;
152-
current = current->next()) {
163+
current = current->next_acquire()) {
153164
if (current->object_no_keepalive() == protection_domain) {
154165
in_pd_set = true;
155166
break;
@@ -167,30 +178,23 @@ bool DictionaryEntry::contains_protection_domain(oop protection_domain) const {
167178
return true;
168179
}
169180

170-
// Lock the pd_set list. This lock cannot safepoint since the caller holds
171-
// a Dictionary entry, which can be moved if the Dictionary is resized.
172-
MutexLocker ml(ProtectionDomainSet_lock, Mutex::_no_safepoint_check_flag);
173-
for (ProtectionDomainEntry* current = pd_set();
181+
for (ProtectionDomainEntry* current = pd_set_acquire();
174182
current != NULL;
175-
current = current->next()) {
183+
current = current->next_acquire()) {
176184
if (current->object_no_keepalive() == protection_domain) {
177185
return true;
178186
}
179187
}
180188
return false;
181189
}
182190

183-
184191
void DictionaryEntry::add_protection_domain(Dictionary* dict, Handle protection_domain) {
185192
assert_locked_or_safepoint(SystemDictionary_lock);
186193
if (!contains_protection_domain(protection_domain())) {
187194
ProtectionDomainCacheEntry* entry = SystemDictionary::cache_get(protection_domain);
188-
// The pd_set in the dictionary entry is protected by a low level lock.
189-
// With concurrent PD table cleanup, these links could be broken.
190-
MutexLocker ml(ProtectionDomainSet_lock, Mutex::_no_safepoint_check_flag);
191-
ProtectionDomainEntry* new_head =
192-
new ProtectionDomainEntry(entry, pd_set());
193-
set_pd_set(new_head);
195+
// Additions and deletions hold the SystemDictionary_lock, readers are lock-free
196+
ProtectionDomainEntry* new_head = new ProtectionDomainEntry(entry, _pd_set);
197+
release_set_pd_set(new_head);
194198
}
195199
LogTarget(Trace, protectiondomain) lt;
196200
if (lt.is_enabled()) {
@@ -420,8 +424,9 @@ void Dictionary::validate_protection_domain(unsigned int name_hash,
420424

421425
// During class loading we may have cached a protection domain that has
422426
// since been unreferenced, so this entry should be cleared.
423-
void Dictionary::clean_cached_protection_domains() {
424-
assert_locked_or_safepoint(SystemDictionary_lock);
427+
void Dictionary::clean_cached_protection_domains(GrowableArray<ProtectionDomainEntry*>* delete_list) {
428+
assert(Thread::current()->is_Java_thread(), "only called by JavaThread");
429+
assert_lock_strong(SystemDictionary_lock);
425430
assert(!loader_data()->has_class_mirror_holder(), "cld should have a ClassLoader holder not a Class holder");
426431

427432
if (loader_data()->is_the_null_class_loader_data()) {
@@ -435,8 +440,7 @@ void Dictionary::clean_cached_protection_domains() {
435440
probe = probe->next()) {
436441
Klass* e = probe->instance_klass();
437442

438-
MutexLocker ml(ProtectionDomainSet_lock, Mutex::_no_safepoint_check_flag);
439-
ProtectionDomainEntry* current = probe->pd_set();
443+
ProtectionDomainEntry* current = probe->pd_set_acquire();
440444
ProtectionDomainEntry* prev = NULL;
441445
while (current != NULL) {
442446
if (current->object_no_keepalive() == NULL) {
@@ -450,18 +454,19 @@ void Dictionary::clean_cached_protection_domains() {
450454
ls.print(" loading: "); probe->instance_klass()->print_value_on(&ls);
451455
ls.cr();
452456
}
453-
if (probe->pd_set() == current) {
454-
probe->set_pd_set(current->next());
457+
if (probe->pd_set_acquire() == current) {
458+
probe->release_set_pd_set(current->next_acquire());
455459
} else {
456460
assert(prev != NULL, "should be set by alive entry");
457-
prev->set_next(current->next());
461+
prev->release_set_next(current->next_acquire());
458462
}
459-
ProtectionDomainEntry* to_delete = current;
460-
current = current->next();
461-
delete to_delete;
463+
// Mark current for deletion but in the meantime it can still be
464+
// traversed.
465+
delete_list->push(current);
466+
current = current->next_acquire();
462467
} else {
463468
prev = current;
464-
current = current->next();
469+
current = current->next_acquire();
465470
}
466471
}
467472
}
@@ -552,20 +557,20 @@ void SymbolPropertyTable::free_entry(SymbolPropertyEntry* entry) {
552557
}
553558

554559
void DictionaryEntry::verify_protection_domain_set() {
555-
MutexLocker ml(ProtectionDomainSet_lock, Mutex::_no_safepoint_check_flag);
556-
for (ProtectionDomainEntry* current = pd_set(); // accessed at a safepoint
560+
assert(SafepointSynchronize::is_at_safepoint(), "must only be called as safepoint");
561+
for (ProtectionDomainEntry* current = pd_set_acquire(); // accessed at a safepoint
557562
current != NULL;
558-
current = current->_next) {
559-
guarantee(oopDesc::is_oop_or_null(current->_pd_cache->object_no_keepalive()), "Invalid oop");
563+
current = current->next_acquire()) {
564+
guarantee(oopDesc::is_oop_or_null(current->object_no_keepalive()), "Invalid oop");
560565
}
561566
}
562567

563568
void DictionaryEntry::print_count(outputStream *st) {
564-
MutexLocker ml(ProtectionDomainSet_lock, Mutex::_no_safepoint_check_flag);
569+
assert_locked_or_safepoint(SystemDictionary_lock);
565570
int count = 0;
566-
for (ProtectionDomainEntry* current = pd_set(); // accessed inside SD lock
571+
for (ProtectionDomainEntry* current = pd_set_acquire();
567572
current != NULL;
568-
current = current->_next) {
573+
current = current->next_acquire()) {
569574
count++;
570575
}
571576
st->print_cr("pd set count = #%d", count);
@@ -596,6 +601,8 @@ void Dictionary::print_on(outputStream* st) const {
596601
// redundant and obvious.
597602
st->print(", ");
598603
cld->print_value_on(st);
604+
st->print(", ");
605+
probe->print_count(st);
599606
}
600607
st->cr();
601608
}

‎src/hotspot/share/classfile/dictionary.hpp

+6-12
Original file line numberDiff line numberDiff line change
@@ -33,6 +33,7 @@
3333

3434
class DictionaryEntry;
3535
class ProtectionDomainEntry;
36+
template <typename T> class GrowableArray;
3637

3738
//~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
3839
// The data structure for the class loader data dictionaries.
@@ -67,7 +68,7 @@ class Dictionary : public Hashtable<InstanceKlass*, mtClass> {
6768
void all_entries_do(KlassClosure* closure);
6869
void classes_do(MetaspaceClosure* it);
6970

70-
void clean_cached_protection_domains();
71+
void clean_cached_protection_domains(GrowableArray<ProtectionDomainEntry*>* delete_list);
7172

7273
// Protection domains
7374
InstanceKlass* find(unsigned int hash, Symbol* name, Handle protection_domain);
@@ -111,18 +112,11 @@ class DictionaryEntry : public HashtableEntry<InstanceKlass*, mtClass> {
111112
// Contains the set of approved protection domains that can access
112113
// this dictionary entry.
113114
//
114-
// This protection domain set is a set of tuples:
115-
//
116-
// (InstanceKlass C, initiating class loader ICL, Protection Domain PD)
117-
//
118115
// [Note that C.protection_domain(), which is stored in the java.lang.Class
119116
// mirror of C, is NOT the same as PD]
120117
//
121-
// If such an entry (C, ICL, PD) exists in the table, it means that
122-
// it is okay for a class Foo to reference C, where
123-
//
124-
// Foo.protection_domain() == PD, and
125-
// Foo's defining class loader == ICL
118+
// If an entry for PD exists in the list, it means that
119+
// it is okay for a caller class to reference the class in this dictionary entry.
126120
//
127121
// The usage of the PD set can be seen in SystemDictionary::validate_protection_domain()
128122
// It is essentially a cache to avoid repeated Java up-calls to
@@ -147,8 +141,8 @@ class DictionaryEntry : public HashtableEntry<InstanceKlass*, mtClass> {
147141
return (DictionaryEntry**)HashtableEntry<InstanceKlass*, mtClass>::next_addr();
148142
}
149143

150-
ProtectionDomainEntry* pd_set() const { return _pd_set; }
151-
void set_pd_set(ProtectionDomainEntry* new_head) { _pd_set = new_head; }
144+
ProtectionDomainEntry* pd_set_acquire() const { return Atomic::load_acquire(&_pd_set); }
145+
void release_set_pd_set(ProtectionDomainEntry* entry) { Atomic::release_store(&_pd_set, entry); }
152146

153147
// Tells whether the initiating class' protection domain can access the klass in this entry
154148
inline bool is_valid_protection_domain(Handle protection_domain);

‎src/hotspot/share/classfile/protectionDomainCache.cpp

+50-6
Original file line numberDiff line numberDiff line change
@@ -34,6 +34,8 @@
3434
#include "memory/universe.hpp"
3535
#include "oops/oop.inline.hpp"
3636
#include "oops/weakHandle.inline.hpp"
37+
#include "runtime/atomic.hpp"
38+
#include "utilities/growableArray.hpp"
3739
#include "utilities/hashtable.inline.hpp"
3840

3941
unsigned int ProtectionDomainCacheTable::compute_hash(Handle protection_domain) {
@@ -58,29 +60,75 @@ void ProtectionDomainCacheTable::trigger_cleanup() {
5860
}
5961

6062
class CleanProtectionDomainEntries : public CLDClosure {
63+
GrowableArray<ProtectionDomainEntry*>* _delete_list;
64+
public:
65+
CleanProtectionDomainEntries(GrowableArray<ProtectionDomainEntry*>* delete_list) :
66+
_delete_list(delete_list) {}
67+
6168
void do_cld(ClassLoaderData* data) {
6269
Dictionary* dictionary = data->dictionary();
6370
if (dictionary != NULL) {
64-
dictionary->clean_cached_protection_domains();
71+
dictionary->clean_cached_protection_domains(_delete_list);
6572
}
6673
}
6774
};
6875

76+
static GrowableArray<ProtectionDomainEntry*>* _delete_list = NULL;
77+
78+
class HandshakeForPD : public HandshakeClosure {
79+
public:
80+
HandshakeForPD() : HandshakeClosure("HandshakeForPD") {}
81+
82+
void do_thread(Thread* thread) {
83+
log_trace(protectiondomain)("HandshakeForPD::do_thread: thread="
84+
INTPTR_FORMAT, p2i(thread));
85+
}
86+
};
87+
88+
static void purge_deleted_entries() {
89+
// If there are any deleted entries, Handshake-all then they'll be
90+
// safe to remove since traversing the pd_set list does not stop for
91+
// safepoints and only JavaThreads will read the pd_set.
92+
// This is actually quite rare because the protection domain is generally associated
93+
// with the caller class and class loader, which if still alive will keep this
94+
// protection domain entry alive.
95+
if (_delete_list->length() >= 10) {
96+
HandshakeForPD hs_pd;
97+
Handshake::execute(&hs_pd);
98+
99+
for (int i = _delete_list->length() - 1; i >= 0; i--) {
100+
ProtectionDomainEntry* entry = _delete_list->at(i);
101+
_delete_list->remove_at(i);
102+
delete entry;
103+
}
104+
assert(_delete_list->length() == 0, "should be cleared");
105+
}
106+
}
107+
69108
void ProtectionDomainCacheTable::unlink() {
70109
// The dictionary entries _pd_set field should be null also, so nothing to do.
71110
assert(java_lang_System::allow_security_manager(), "should not be called otherwise");
72111

112+
// Create a list for holding deleted entries
113+
if (_delete_list == NULL) {
114+
_delete_list = new (ResourceObj::C_HEAP, mtClass)
115+
GrowableArray<ProtectionDomainEntry*>(20, mtClass);
116+
}
117+
73118
{
74119
// First clean cached pd lists in loaded CLDs
75120
// It's unlikely, but some loaded classes in a dictionary might
76121
// point to a protection_domain that has been unloaded.
77122
// The dictionary pd_set points at entries in the ProtectionDomainCacheTable.
78123
MutexLocker ml(ClassLoaderDataGraph_lock);
79124
MutexLocker mldict(SystemDictionary_lock); // need both.
80-
CleanProtectionDomainEntries clean;
125+
CleanProtectionDomainEntries clean(_delete_list);
81126
ClassLoaderDataGraph::loaded_cld_do(&clean);
82127
}
83128

129+
// Purge any deleted entries outside of the SystemDictionary_lock.
130+
purge_deleted_entries();
131+
84132
MutexLocker ml(SystemDictionary_lock);
85133
int oops_removed = 0;
86134
for (int i = 0; i < table_size(); ++i) {
@@ -129,10 +177,6 @@ oop ProtectionDomainCacheEntry::object() {
129177
return literal().resolve();
130178
}
131179

132-
oop ProtectionDomainEntry::object() {
133-
return _pd_cache->object();
134-
}
135-
136180
// The object_no_keepalive() call peeks at the phantomly reachable oop without
137181
// keeping it alive. This is okay to do in the VM thread state if it is not
138182
// leaked out to become strongly reachable.

‎src/hotspot/share/classfile/protectionDomainCache.hpp

+9-14
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2017, 2019, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2017, 2021, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -27,7 +27,7 @@
2727

2828
#include "oops/oop.hpp"
2929
#include "oops/weakHandle.hpp"
30-
#include "memory/iterator.hpp"
30+
#include "runtime/atomic.hpp"
3131
#include "utilities/hashtable.hpp"
3232

3333
// This class caches the approved protection domains that can access loaded classes.
@@ -62,8 +62,6 @@ class ProtectionDomainCacheEntry : public HashtableEntry<WeakHandle, mtClass> {
6262
// The amount of different protection domains used is typically magnitudes smaller
6363
// than the number of system dictionary entries (loaded classes).
6464
class ProtectionDomainCacheTable : public Hashtable<WeakHandle, mtClass> {
65-
friend class VMStructs;
66-
private:
6765
ProtectionDomainCacheEntry* bucket(int i) const {
6866
return (ProtectionDomainCacheEntry*) Hashtable<WeakHandle, mtClass>::bucket(i);
6967
}
@@ -104,20 +102,17 @@ class ProtectionDomainCacheTable : public Hashtable<WeakHandle, mtClass> {
104102
};
105103

106104

105+
// This describes the linked list protection domain for each DictionaryEntry in pd_set.
107106
class ProtectionDomainEntry :public CHeapObj<mtClass> {
108-
friend class VMStructs;
109-
public:
110-
ProtectionDomainEntry* _next;
111107
ProtectionDomainCacheEntry* _pd_cache;
108+
ProtectionDomainEntry* volatile _next;
109+
public:
112110

113-
ProtectionDomainEntry(ProtectionDomainCacheEntry* pd_cache, ProtectionDomainEntry* next) {
114-
_pd_cache = pd_cache;
115-
_next = next;
116-
}
111+
ProtectionDomainEntry(ProtectionDomainCacheEntry* pd_cache,
112+
ProtectionDomainEntry* head) : _pd_cache(pd_cache), _next(head) {}
117113

118-
ProtectionDomainEntry* next() { return _next; }
119-
void set_next(ProtectionDomainEntry* entry) { _next = entry; }
120-
oop object();
114+
ProtectionDomainEntry* next_acquire() { return Atomic::load_acquire(&_next); }
115+
void release_set_next(ProtectionDomainEntry* entry) { Atomic::release_store(&_next, entry); }
121116
oop object_no_keepalive();
122117
};
123118
#endif // SHARE_CLASSFILE_PROTECTIONDOMAINCACHE_HPP

‎src/hotspot/share/runtime/mutexLocker.cpp

-2
Original file line numberDiff line numberDiff line change
@@ -42,7 +42,6 @@
4242
Mutex* Patching_lock = NULL;
4343
Mutex* CompiledMethod_lock = NULL;
4444
Monitor* SystemDictionary_lock = NULL;
45-
Mutex* ProtectionDomainSet_lock = NULL;
4645
Mutex* SharedDictionary_lock = NULL;
4746
Mutex* Module_lock = NULL;
4847
Mutex* CompiledIC_lock = NULL;
@@ -261,7 +260,6 @@ void mutex_init() {
261260
def(JmethodIdCreation_lock , PaddedMutex , special-2, true, _safepoint_check_never); // used for creating jmethodIDs.
262261

263262
def(SystemDictionary_lock , PaddedMonitor, leaf, true, _safepoint_check_always);
264-
def(ProtectionDomainSet_lock , PaddedMutex , leaf-1, true, _safepoint_check_never);
265263
def(SharedDictionary_lock , PaddedMutex , leaf, true, _safepoint_check_always);
266264
def(Module_lock , PaddedMutex , leaf+2, false, _safepoint_check_always);
267265
def(InlineCacheBuffer_lock , PaddedMutex , leaf, true, _safepoint_check_never);

‎src/hotspot/share/runtime/mutexLocker.hpp

-1
Original file line numberDiff line numberDiff line change
@@ -34,7 +34,6 @@
3434
extern Mutex* Patching_lock; // a lock used to guard code patching of compiled code
3535
extern Mutex* CompiledMethod_lock; // a lock used to guard a compiled method and OSR queues
3636
extern Monitor* SystemDictionary_lock; // a lock on the system dictionary
37-
extern Mutex* ProtectionDomainSet_lock; // a lock on the pd_set list in the system dictionary
3837
extern Mutex* SharedDictionary_lock; // a lock on the CDS shared dictionary
3938
extern Mutex* Module_lock; // a lock on module and package related data structures
4039
extern Mutex* CompiledIC_lock; // a lock used to guard compiled IC patching and access

‎test/hotspot/jtreg/runtime/Dictionary/ClassForName.java

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2018, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2018, 2021, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -37,7 +37,7 @@ public class ClassForName {
3737

3838
public ClassForName() {
3939
try {
40-
// class_loader = URLClassLoader, protection_domain = ClassForName.getProtectionDomain()
40+
// class_loader = App$ClassLoader, protection_domain = ClassForName.getProtectionDomain()
4141
Class.forName(java.util.List.class.getName(), false,
4242
ClassLoader.getSystemClassLoader());
4343
} catch (Throwable e) {

‎test/hotspot/jtreg/runtime/Dictionary/ProtectionDomainCacheTest.java

+6-3
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2018, 2020, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2018, 2021, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -84,7 +84,9 @@ public static void main(final String[] args) throws Exception {
8484
CLASSFILENAME);
8585
Files.delete(classFile);
8686

87-
loadAndRun(jarFilePath);
87+
for (int i = 0; i < 20; i++) {
88+
loadAndRun(jarFilePath);
89+
}
8890

8991
// Give the GC a chance to unload protection domains
9092
for (int i = 0; i < 100; i++) {
@@ -101,11 +103,12 @@ public static void main(String args[]) throws Exception {
101103
"-XX:+UnlockDiagnosticVMOptions",
102104
"-XX:VerifySubSet=dictionary",
103105
"-XX:+VerifyAfterGC",
104-
"-Xlog:gc+verify,protectiondomain=debug",
106+
"-Xlog:gc+verify,protectiondomain=trace",
105107
"-Djava.security.manager",
106108
Test.class.getName());
107109
OutputAnalyzer output = new OutputAnalyzer(pb.start());
108110
output.shouldContain("PD in set is not alive");
111+
output.shouldContain("HandshakeForPD::do_thread");
109112
output.shouldHaveExitValue(0);
110113
}
111114
}

‎test/hotspot/jtreg/vmTestbase/vm/mlvm/mixed/stress/java/findDeadlock/TestDescription.java

+9-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2018, 2020, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2018, 2021, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -43,7 +43,14 @@
4343
* @comment build test class and indify classes
4444
* @build vm.mlvm.mixed.stress.java.findDeadlock.INDIFY_Test
4545
* @run driver vm.mlvm.share.IndifiedClassesBuilder
46-
*
4746
* @run main/othervm -Xlog:gc,safepoint vm.mlvm.mixed.stress.java.findDeadlock.INDIFY_Test
47+
*
48+
* To see code that takes more time to safepoint run with:
49+
* main/othervm -XX:+SafepointTimeout -XX:+UnlockDiagnosticVMOptions
50+
* -XX:+AbortVMOnSafepointTimeout
51+
* -XX:SafepointTimeoutDelay=500
52+
* -XX:+PrintSystemDictionaryAtExit
53+
* -Xlog:gc,safepoint
54+
* vm.mlvm.mixed.stress.java.findDeadlock.INDIFY_Test
4855
*/
4956

‎test/hotspot/jtreg/vmTestbase/vm/mlvm/tools/Indify.java

+2-2
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2010, 2018, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2010, 2021, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -109,7 +109,7 @@ public static void main(String... av) throws IOException {
109109
public boolean keepgoing = false;
110110
public boolean expandProperties = false;
111111
public boolean overwrite = false;
112-
public boolean quiet = false;
112+
public boolean quiet = true;
113113
public boolean verbose = false;
114114
public boolean transitionalJSR292 = true; // default to false later
115115
public boolean all = false;

0 commit comments

Comments
 (0)
Please sign in to comment.