44
44
#include " runtime/javaCalls.hpp"
45
45
#include " runtime/mutexLocker.hpp"
46
46
#include " runtime/safepointVerifiers.hpp"
47
+ #include " utilities/growableArray.hpp"
47
48
#include " utilities/hashtable.inline.hpp"
48
49
49
50
// Optimization: if any dictionary needs resizing, we set this flag,
@@ -78,20 +79,19 @@ Dictionary::~Dictionary() {
78
79
79
80
DictionaryEntry* Dictionary::new_entry (unsigned int hash, InstanceKlass* klass) {
80
81
DictionaryEntry* entry = (DictionaryEntry*)Hashtable<InstanceKlass*, mtClass>::new_entry (hash, klass);
81
- entry->set_pd_set (NULL );
82
+ entry->release_set_pd_set (NULL );
82
83
assert (klass->is_instance_klass (), " Must be" );
83
84
return entry;
84
85
}
85
86
86
-
87
87
void Dictionary::free_entry (DictionaryEntry* entry) {
88
88
// avoid recursion when deleting linked list
89
89
// pd_set is accessed during a safepoint.
90
90
// This doesn't require a lock because nothing is reading this
91
91
// 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 ());
95
95
delete to_delete;
96
96
}
97
97
BasicHashtable<mtClass>::free_entry (entry);
@@ -141,15 +141,26 @@ bool DictionaryEntry::is_valid_protection_domain(Handle protection_domain) {
141
141
: contains_protection_domain (protection_domain ());
142
142
}
143
143
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
+
144
152
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;
145
157
#ifdef ASSERT
146
158
if (protection_domain == instance_klass ()->protection_domain ()) {
147
- MutexLocker ml (ProtectionDomainSet_lock, Mutex::_no_safepoint_check_flag);
148
159
// Ensure this doesn't show up in the pd_set (invariant)
149
160
bool in_pd_set = false ;
150
- for (ProtectionDomainEntry* current = pd_set ();
161
+ for (ProtectionDomainEntry* current = pd_set_acquire ();
151
162
current != NULL ;
152
- current = current->next ()) {
163
+ current = current->next_acquire ()) {
153
164
if (current->object_no_keepalive () == protection_domain) {
154
165
in_pd_set = true ;
155
166
break ;
@@ -167,30 +178,23 @@ bool DictionaryEntry::contains_protection_domain(oop protection_domain) const {
167
178
return true ;
168
179
}
169
180
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 ();
174
182
current != NULL ;
175
- current = current->next ()) {
183
+ current = current->next_acquire ()) {
176
184
if (current->object_no_keepalive () == protection_domain) {
177
185
return true ;
178
186
}
179
187
}
180
188
return false ;
181
189
}
182
190
183
-
184
191
void DictionaryEntry::add_protection_domain (Dictionary* dict, Handle protection_domain) {
185
192
assert_locked_or_safepoint (SystemDictionary_lock);
186
193
if (!contains_protection_domain (protection_domain ())) {
187
194
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);
194
198
}
195
199
LogTarget (Trace, protectiondomain) lt;
196
200
if (lt.is_enabled ()) {
@@ -420,8 +424,9 @@ void Dictionary::validate_protection_domain(unsigned int name_hash,
420
424
421
425
// During class loading we may have cached a protection domain that has
422
426
// 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);
425
430
assert (!loader_data ()->has_class_mirror_holder (), " cld should have a ClassLoader holder not a Class holder" );
426
431
427
432
if (loader_data ()->is_the_null_class_loader_data ()) {
@@ -435,8 +440,7 @@ void Dictionary::clean_cached_protection_domains() {
435
440
probe = probe->next ()) {
436
441
Klass* e = probe->instance_klass ();
437
442
438
- MutexLocker ml (ProtectionDomainSet_lock, Mutex::_no_safepoint_check_flag);
439
- ProtectionDomainEntry* current = probe->pd_set ();
443
+ ProtectionDomainEntry* current = probe->pd_set_acquire ();
440
444
ProtectionDomainEntry* prev = NULL ;
441
445
while (current != NULL ) {
442
446
if (current->object_no_keepalive () == NULL ) {
@@ -450,18 +454,19 @@ void Dictionary::clean_cached_protection_domains() {
450
454
ls.print (" loading: " ); probe->instance_klass ()->print_value_on (&ls);
451
455
ls.cr ();
452
456
}
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 ());
455
459
} else {
456
460
assert (prev != NULL , " should be set by alive entry" );
457
- prev->set_next (current->next ());
461
+ prev->release_set_next (current->next_acquire ());
458
462
}
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 ();
462
467
} else {
463
468
prev = current;
464
- current = current->next ();
469
+ current = current->next_acquire ();
465
470
}
466
471
}
467
472
}
@@ -552,20 +557,20 @@ void SymbolPropertyTable::free_entry(SymbolPropertyEntry* entry) {
552
557
}
553
558
554
559
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
557
562
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" );
560
565
}
561
566
}
562
567
563
568
void DictionaryEntry::print_count (outputStream *st) {
564
- MutexLocker ml (ProtectionDomainSet_lock, Mutex::_no_safepoint_check_flag );
569
+ assert_locked_or_safepoint (SystemDictionary_lock );
565
570
int count = 0 ;
566
- for (ProtectionDomainEntry* current = pd_set (); // accessed inside SD lock
571
+ for (ProtectionDomainEntry* current = pd_set_acquire ();
567
572
current != NULL ;
568
- current = current->_next ) {
573
+ current = current->next_acquire () ) {
569
574
count++;
570
575
}
571
576
st->print_cr (" pd set count = #%d" , count);
@@ -596,6 +601,8 @@ void Dictionary::print_on(outputStream* st) const {
596
601
// redundant and obvious.
597
602
st->print (" , " );
598
603
cld->print_value_on (st);
604
+ st->print (" , " );
605
+ probe->print_count (st);
599
606
}
600
607
st->cr ();
601
608
}
0 commit comments