Skip to content

Commit f259eea

Browse files
author
Daniel D. Daugherty
committedApr 3, 2021
8264393: JDK-8258284 introduced dangling TLH race
Reviewed-by: dholmes, rehn, eosterlund
1 parent 9b2232b commit f259eea

File tree

4 files changed

+44
-43
lines changed

4 files changed

+44
-43
lines changed
 

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

+19-23
Original file line numberDiff line numberDiff line change
@@ -417,7 +417,7 @@ void SafeThreadsListPtr::acquire_stable_list() {
417417
_previous = _thread->_threads_list_ptr;
418418
_thread->_threads_list_ptr = this;
419419

420-
if (_thread->get_threads_hazard_ptr() == NULL) {
420+
if (_thread->get_threads_hazard_ptr() == NULL && _previous == NULL) {
421421
// The typical case is first.
422422
acquire_stable_list_fast_path();
423423
return;
@@ -484,8 +484,6 @@ void SafeThreadsListPtr::acquire_stable_list_fast_path() {
484484
//
485485
void SafeThreadsListPtr::acquire_stable_list_nested_path() {
486486
assert(_thread != NULL, "sanity check");
487-
assert(_thread->get_threads_hazard_ptr() != NULL,
488-
"cannot have a NULL regular hazard ptr when acquiring a nested hazard ptr");
489487

490488
// The thread already has a hazard ptr (ThreadsList ref) so we need
491489
// to create a nested ThreadsListHandle with the current ThreadsList
@@ -506,7 +504,7 @@ void SafeThreadsListPtr::acquire_stable_list_nested_path() {
506504
}
507505
// Clear the hazard ptr so we can go through the fast path below and
508506
// acquire a nested stable ThreadsList.
509-
Atomic::store(&_thread->_threads_hazard_ptr, (ThreadsList*)NULL);
507+
_thread->set_threads_hazard_ptr(NULL);
510508

511509
if (EnableThreadSMRStatistics && _thread->nested_threads_hazard_ptr_cnt() > ThreadsSMRSupport::_nested_thread_list_max) {
512510
ThreadsSMRSupport::_nested_thread_list_max = _thread->nested_threads_hazard_ptr_cnt();
@@ -523,11 +521,26 @@ void SafeThreadsListPtr::acquire_stable_list_nested_path() {
523521
//
524522
void SafeThreadsListPtr::release_stable_list() {
525523
assert(_thread != NULL, "sanity check");
526-
assert(_thread->get_threads_hazard_ptr() != NULL, "sanity check");
527-
assert(_thread->get_threads_hazard_ptr() == _list, "sanity check");
528524
assert(_thread->_threads_list_ptr == this, "sanity check");
529525
_thread->_threads_list_ptr = _previous;
530526

527+
// We're releasing either a leaf or nested ThreadsListHandle. In either
528+
// case, we set this thread's hazard ptr back to NULL and we do it before
529+
// _nested_handle_cnt is decremented below.
530+
_thread->set_threads_hazard_ptr(NULL);
531+
if (_previous != NULL) {
532+
// The ThreadsListHandle being released is a nested ThreadsListHandle.
533+
if (EnableThreadSMRStatistics) {
534+
_thread->dec_nested_threads_hazard_ptr_cnt();
535+
}
536+
// The previous ThreadsList is stable because the _nested_handle_cnt is
537+
// > 0, but we cannot safely make it this thread's hazard ptr again.
538+
// The protocol for installing and verifying a ThreadsList as a
539+
// thread's hazard ptr is handled by acquire_stable_list_fast_path().
540+
// And that protocol cannot be properly done with a ThreadsList that
541+
// might not be the current system ThreadsList.
542+
assert(_previous->_list->_nested_handle_cnt > 0, "must be > zero");
543+
}
531544
if (_has_ref_count) {
532545
// This thread created a nested ThreadsListHandle after the current
533546
// ThreadsListHandle so we had to protect this ThreadsList with a
@@ -536,23 +549,6 @@ void SafeThreadsListPtr::release_stable_list() {
536549

537550
log_debug(thread, smr)("tid=" UINTX_FORMAT ": SafeThreadsListPtr::release_stable_list: delete nested list pointer to ThreadsList=" INTPTR_FORMAT, os::current_thread_id(), p2i(_list));
538551
}
539-
if (_previous == NULL) {
540-
// The ThreadsListHandle being released is a leaf ThreadsListHandle.
541-
// This is the "normal" case and this is where we set this thread's
542-
// hazard ptr back to NULL.
543-
_thread->set_threads_hazard_ptr(NULL);
544-
} else {
545-
// The ThreadsListHandle being released is a nested ThreadsListHandle.
546-
if (EnableThreadSMRStatistics) {
547-
_thread->dec_nested_threads_hazard_ptr_cnt();
548-
}
549-
// The previous ThreadsList becomes this thread's hazard ptr again.
550-
// It is a stable ThreadsList since the non-zero _nested_handle_cnt
551-
// keeps it from being freed so we can just set the thread's hazard
552-
// ptr without going through the stabilization/tagging protocol.
553-
assert(_previous->_list->_nested_handle_cnt > 0, "must be > than zero");
554-
_thread->set_threads_hazard_ptr(_previous->_list);
555-
}
556552

557553
// After releasing the hazard ptr, other threads may go ahead and
558554
// free up some memory temporarily used by a ThreadsList snapshot.

‎test/hotspot/gtest/runtime/test_ThreadsListHandle.cpp

+21-16
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2020, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2020, 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
@@ -213,9 +213,10 @@ TEST_VM(ThreadsListHandle, sanity) {
213213

214214
// Test case: after first nested ThreadsListHandle (tlh2) has been destroyed
215215

216-
// Verify the current thread refers to tlh1:
217-
EXPECT_EQ(ThreadsListHandleTest::get_Thread_threads_hazard_ptr(thr), tlh1.list())
218-
<< "thr->_threads_hazard_ptr must match tlh1.list()";
216+
// Verify the current thread's hazard ptr is NULL:
217+
EXPECT_EQ(ThreadsListHandleTest::get_Thread_threads_hazard_ptr(thr), (ThreadsList*)NULL)
218+
<< "thr->_threads_hazard_ptr must be NULL";
219+
// Verify the current thread's threads list ptr refers to tlh1:
219220
EXPECT_EQ(ThreadsListHandleTest::get_Thread_threads_list_ptr(thr), list_ptr1)
220221
<< "thr->_threads_list_ptr must match list_ptr1";
221222
EXPECT_EQ(ThreadsListHandleTest::get_Thread_nested_threads_hazard_ptr_cnt(thr), (uint)0)
@@ -396,9 +397,10 @@ TEST_VM(ThreadsListHandle, sanity) {
396397

397398
// Test case: after double nested ThreadsListHandle (tlh3) has been destroyed
398399

399-
// Verify the current thread refers to tlh2:
400-
EXPECT_EQ(ThreadsListHandleTest::get_Thread_threads_hazard_ptr(thr), tlh2.list())
401-
<< "thr->_threads_hazard_ptr must match tlh2.list()";
400+
// Verify the current thread's hazard ptr is NULL:
401+
EXPECT_EQ(ThreadsListHandleTest::get_Thread_threads_hazard_ptr(thr), (ThreadsList*)NULL)
402+
<< "thr->_threads_hazard_ptr must be NULL";
403+
// Verify the current thread's threads list ptr refers to tlh2:
402404
EXPECT_EQ(tlh1.list(), tlh2.list())
403405
<< "tlh1.list() must match tlh2.list()";
404406
EXPECT_EQ(ThreadsListHandleTest::get_Thread_threads_list_ptr(thr), list_ptr2)
@@ -443,9 +445,10 @@ TEST_VM(ThreadsListHandle, sanity) {
443445

444446
// Test case: after first nested ThreadsListHandle (tlh2) has been destroyed
445447

446-
// Verify the current thread refers to tlh1:
447-
EXPECT_EQ(ThreadsListHandleTest::get_Thread_threads_hazard_ptr(thr), tlh1.list())
448-
<< "thr->_threads_hazard_ptr must match tlh1.list()";
448+
// Verify the current thread's hazard ptr is NULL:
449+
EXPECT_EQ(ThreadsListHandleTest::get_Thread_threads_hazard_ptr(thr), (ThreadsList*)NULL)
450+
<< "thr->_threads_hazard_ptr must be NULL";
451+
// Verify the current thread's threads list ptr refers to tlh1:
449452
EXPECT_EQ(ThreadsListHandleTest::get_Thread_threads_list_ptr(thr), list_ptr1)
450453
<< "thr->_threads_list_ptr must match list_ptr1";
451454
EXPECT_EQ(ThreadsListHandleTest::get_Thread_nested_threads_hazard_ptr_cnt(thr), (uint)0)
@@ -562,9 +565,10 @@ TEST_VM(ThreadsListHandle, sanity) {
562565

563566
// Test case: after first back-to-back nested ThreadsListHandle (tlh2a) has been destroyed
564567

565-
// Verify the current thread refers to tlh1:
566-
EXPECT_EQ(ThreadsListHandleTest::get_Thread_threads_hazard_ptr(thr), tlh1.list())
567-
<< "thr->_threads_hazard_ptr must match tlh1.list()";
568+
// Verify the current thread's hazard ptr is NULL:
569+
EXPECT_EQ(ThreadsListHandleTest::get_Thread_threads_hazard_ptr(thr), (ThreadsList*)NULL)
570+
<< "thr->_threads_hazard_ptr must be NULL";
571+
// Verify the current thread's threads list ptr refers to tlh1:
568572
EXPECT_EQ(ThreadsListHandleTest::get_Thread_threads_list_ptr(thr), list_ptr1)
569573
<< "thr->_threads_list_ptr must match list_ptr1";
570574
EXPECT_EQ(ThreadsListHandleTest::get_Thread_nested_threads_hazard_ptr_cnt(thr), (uint)0)
@@ -639,9 +643,10 @@ TEST_VM(ThreadsListHandle, sanity) {
639643

640644
// Test case: after second back-to-back nested ThreadsListHandle (tlh2b) has been destroyed
641645

642-
// Verify the current thread refers to tlh1:
643-
EXPECT_EQ(ThreadsListHandleTest::get_Thread_threads_hazard_ptr(thr), tlh1.list())
644-
<< "thr->_threads_hazard_ptr must match tlh1.list()";
646+
// Verify the current thread's hazard ptr is NULL:
647+
EXPECT_EQ(ThreadsListHandleTest::get_Thread_threads_hazard_ptr(thr), (ThreadsList*)NULL)
648+
<< "thr->_threads_hazard_ptr must be NULL";
649+
// Verify the current thread's threads list ptr refers to tlh1:
645650
EXPECT_EQ(ThreadsListHandleTest::get_Thread_threads_list_ptr(thr), list_ptr1)
646651
<< "thr->_threads_list_ptr must match list_ptr1";
647652
EXPECT_EQ(ThreadsListHandleTest::get_Thread_nested_threads_hazard_ptr_cnt(thr), (uint)0)

‎test/hotspot/jtreg/runtime/ErrorHandling/NestedThreadsListHandleInErrorHandlingTest.java

+2-2
Original file line numberDiff line numberDiff line change
@@ -98,8 +98,8 @@ public static void main(String[] args) throws Exception {
9898
// a ThreadsListHandle in addition to what the test creates:
9999
Pattern.compile(".*, _nested_thread_list_max=2"),
100100
// The current thread (marked with '=>') in the threads list
101-
// should show a hazard ptr and a nested hazard ptr:
102-
Pattern.compile("=>.* JavaThread \"main\" .* _threads_hazard_ptr=0x[0-9A-Fa-f][0-9A-Fa-f]*, _nested_threads_hazard_ptr_cnt=1, _nested_threads_hazard_ptr=0x[0-9A-Fa-f][0-9A-Fa-f]*.*"),
101+
// should show a nested hazard ptr:
102+
Pattern.compile("=>.* JavaThread \"main\" .*, _nested_threads_hazard_ptr_cnt=1, _nested_threads_hazard_ptr=0x[0-9A-Fa-f][0-9A-Fa-f]*.*"),
103103
};
104104
int currentPattern = 0;
105105

‎test/hotspot/jtreg/runtime/ErrorHandling/ThreadsListHandleInErrorHandlingTest.java

+2-2
Original file line numberDiff line numberDiff line change
@@ -98,8 +98,8 @@ public static void main(String[] args) throws Exception {
9898
// a ThreadsListHandle:
9999
Pattern.compile(".*, _nested_thread_list_max=1"),
100100
// The current thread (marked with '=>') in the threads list
101-
// should show a hazard ptr and no nested hazard ptrs:
102-
Pattern.compile("=>.* JavaThread \"main\" .* _threads_hazard_ptr=0x[0-9A-Fa-f][0-9A-Fa-f]*, _nested_threads_hazard_ptr_cnt=0.*"),
101+
// should show no nested hazard ptrs:
102+
Pattern.compile("=>.* JavaThread \"main\" .*, _nested_threads_hazard_ptr_cnt=0.*"),
103103
};
104104
int currentPattern = 0;
105105

0 commit comments

Comments
 (0)
Please sign in to comment.