RFR (S): 8182703: Correct G1 barrier queue lock orderings
Erik Österlund
erik.osterlund at oracle.com
Mon Jun 26 13:34:22 UTC 2017
Hi,
Webrev: http://cr.openjdk.java.net/~eosterlund/8182703/webrev.02/
Bug: https://bugs.openjdk.java.net/browse/JDK-8182703
The G1 barrier queues have very awkward lock orderings for the following
reasons:
1) These queues may queue up things when performing a reference write or
resolving a jweak (intentionally or just happened to be jweak, even
though it looks like a jobject), which can happen in a lot of places in
the code. We resolve JNIHandles while holding special locks in many
places. We perform reference writes also in many places. Now the
unsuspecting hotspot developer might think that it is okay to resolve a
JNIHandle or perform a reference write while possibly holding a special
lock. But no. In some cases, object writes have been moved out of locks
and replaced with lock-free CAS, only to dodge the G1 write barrier
locks. I don't think the G1 lock ordering issues should shape the shared
code rather than the other way around.
2) There is an issue that the shared queue locks have a "special" rank,
which is below the lock ranks used by the cbl monitor and free list
monitor. This leads to an issue when these locks have to be taken while
holding the shared queue locks. The current solution is to drop the
shared queue locks temporarily, introducing nasty data races. These
races are guarded, but the whole race seems very unnecessary.
I argue that if the G1 write barrier queue locks were simply set
appropriately in the first place by analyzing what ranks they should
have, none of the above issues would exist. Therefore I propose this new
ordering.
Specifically, I recognize that locks required for performing memory
accesses and resolving JNIHandles are more special than the "special"
rank. Therefore, this change introduces a new lock ordering category
called "access", which is to be used by barriers required to perform
memory accesses. In other words, by recognizing the rank is more special
than "special", we can remove "special" code to walk around making its
rank more "special". That seems desirable to me. The access locks need
to comply to the same constraints as the special locks: they may not
perform safepoint checks.
The old lock ranks were:
SATB_Q_FL_lock: special
SATB_Q_CBL_mon: leaf - 1
Shared_SATB_Q_lock: leaf - 1
DirtyCardQ_FL_lock: special
DirtyCardQ_CBL_mon: leaf - 1
Shared_DirtyCardQ_lock: leaf - 1
The new lock ranks are:
SATB_Q_FL_lock: access (special - 2)
SATB_Q_CBL_mon: access (special - 2)
Shared_SATB_Q_lock: access + 1 (special - 1)
DirtyCardQ_FL_lock: access (special - 2)
DirtyCardQ_CBL_mon: access (special - 2)
Shared_DirtyCardQ_lock: access + 1 (special - 1)
Analysis:
Each PtrQueue and PtrQueueSet group, SATB or DirtyCardQ have the same
group of locks. The free list lock, the completed buffer list monitor
and the shared queue lock.
Observations:
1) The free list lock and completed buffer list monitors (members of
PtrQueueSet) are disjoint. We never hold both of them at the same time.
Rationale: The free list lock is only used from
PtrQueueSet::allocate_buffer, PtrQueueSet::deallocate_buffer and
PtrQueueSet::reduce_free_list, and no callsite from there can be
expanded where the cbl monitor is acquired. So therefore it is
impossible to acquire the cbl monitor while holding the free list lock.
The opposite case of acquiring the free list lock while holding the cbl
monitor is also not possible; only the following places acquire the cbl
monitor: PtrQueueSet::enqueue_complete_buffer,
PtrQueueSet::merge_bufferlists,
PtrQueueSet::assert_completed_buffer_list_len_correct,
PtrQueueSet::notify_if_necessary, FreeIdSet::claim_par_id,
FreeIdSet::release_par_id, DirtyCardQueueSet::get_completed_buffer,
DirtyCardQueueSet::clear,
SATBMarkQueueSet::apply_closure_to_completed_buffer and
SATBMarkQueueSet::abandon_partial_marking. Again, neither of these paths
where the cbl monitor is held can expand callsites to a place where the
free list locks are held. Therefore it holds that the cbl monitor can
not be held while the free list lock is held, and the free list lock can
not be held while the cbl monitor is held. Therefore they are held
disjointly.
2) We might hold the shared queue locks before acquiring the completed
buffer list monitor. (today we drop the shared queue lock then and
reacquire it later as a hack as already described)
3) We do not acquire a shared queue lock while holding the free list
lock or completed buffer list monitor, as there is no reference from a
PtrQueueSet to its shared queue, so those code paths do not know how to
reach the shared PtrQueue to acquire its lock. The derived classes are
exceptions but they never use the shared queue lock while holding the
completed buffer list monitor or free list lock. DirtyCardQueueSet uses
the shared queue for concatenating logs (in a safepoint without holding
those locks). The SATBMarkQueueSet uses the shared queue for filtering
the buffers, fiddling with activeness, printing and resetting, all
without grabbing any locks.
4) We do not acquire any other lock (above event) while holding the free
list lock or completed buffer list monitors. This was discovered by
manually expanding the call graphs from where these two locks are held.
Derived constraints:
a) Because of observation 1, the free list lock and completed buffer
list monitors can have the same rank.
b) Because of observations 1 and 2, the shared queue lock ought to have
a rank higher than the ranks of the free list lock and the completed
buffer list monitors (not the case today).
c) Because of of observation 3 and 2, the free list lock and completed
buffer list monitors ought to have a rank lower than the rank of the
shared queue lock.
d) Because of observation 4 (and constraints a-c), all the barrier locks
should be below the "special" rank without violating any existing ranks.
The proposed new lock ranks conform to the constraints derived from my
observations. It is worth noting that the potential relationship that
could break (and why they do not) are:
1) If a lock is acquired from within the barriers that does not involve
the shared queue lock, the free list lock or the completed buffer list
monitor, we have now inverted their relationship as that other lock
would probably have a rank higher than or equal to "special". But due to
observation 4, there are no such cases.
2) The relationship between the shared queue lock and the completed
buffer list monitor has been changed so both can be held at the same
time if the shared queue lock is acquired first (which it is). This is
arguably the way it should have been from the first place, and the old
solution had ugly hacks where we would drop the shared queue lock to not
run into the lock order assert (and only not to run into the lock order
assert, i.e. not to avoid potential deadlock) to ensure the locks are
not held at the same time. That code has now been removed, so that the
shared queue lock is still held when enqueueing completed buffers (no
dodgy dropping and reclaiming), and the code for handling the races due
to multiple concurrent enqueuers has also been removed and replaced with
an assertion that there simply should not be multiple concurrent
enqueuers. Since the shared queue lock is now held throughout the whole
operation, there will be no concurrent enqueuers.
3) The completed buffer list monitor used to have a higher rank than the
free list lock. Now they have the same. Therefore, they could previously
allow them to be held at the same time if the cbl monitor was acquired
first. However, as discussed, there is no such case, and they ought to
have the same rank not to confuse their true disjointness. If anyone
insists we do not break this relationship despite the true disjointness,
I could consent to adding another access lock rank, like this:
http://cr.openjdk.java.net/~eosterlund/8182703/webrev.01/ but I think it
seems better to have the same rank since they are actually truly
disjoint and should remain disjoint.
I do recognize that long term, we *might* want a lock-free solution or
something (not saying we do or do not). But until then, the ranks ought
to be corrected so that they do not cause these problems causing
everyone to bash their head against the awkward G1 lock ranks throughout
the code and make hacks around it.
Testing: JPRT with hotspot all and lots of local testing.
Thanks,
/Erik
More information about the hotspot-gc-dev
mailing list