RFR: 8257831: Suspend with handshakes [v2]
Patricio Chilano Mateo
pchilanomate at openjdk.java.net
Tue Apr 6 17:31:36 UTC 2021
On Tue, 6 Apr 2021 15:46:17 GMT, Richard Reingruber <rrich at openjdk.org> wrote:
>> (technically this will be ordered by the poll is since polls are only disarmed by threads them selfs)
>> (meaning if you really promise to call should_process() after you have seen "set_suspend(true);" you will see the async handshake since then we do take the HandshakeState lock.)
>
>> Also when the suspend request happened while A was blocked then after
>> current->set_thread_state_fence(_thread_blocked_trans); the check of
>> is_suspended() will return true even if the handshake state lock is not
>> acquired for the check
>
> That statement of mine is wrong. (Hopefully) correct is that after the complete
> state change from _thread_blocked to _thread_in_vm which includes being blocked
> for a safepoint/handshake the current thread would be able to check
> is_suspended() without holding the handshake state lock. It does not make a lot
> of sense then because in an unsafe state JavaThread::current()->is_suspended()
> will always return false.
>
>> Thread A wait on OM X in blocked.
>> Thread Z suspends Thread A, thread Z have no relation to OM X.
>> Thread B unlocks OM X, thread B have no relation to the suspend request.
>> Thread A locks OM X while blocked.
>> Thread A was not allowed to lock OM X due to it is actually suspended.
>> Thread A unlocks OM X and waits until resumed.
>
> I understand that example now too: OM and suspend operations are unrelated. So I
> thought it would be ok for A to enter OM X, but it is not. A thread must not
> leave a safe state if it was suspended in that state (with a handshake). If it
> did, e.g., its stack could become not walkable. And it must not enter the
> monitor either.
>
> Sorry for the confusion. The check is good.
But I still think we can just call is_suspended() here. If is_suspended() returns false, then Z hasn't finished suspending A yet (either async operation was still not added to the queue in case this is the first suspend for A, or do_handshake() hasn't yet finished for that SuspendThreadHandshake operation in case there was already an async operation in the queue). Since we have already acquired OM, then Z will not see that after being suspended A switched from not owning OM to owning it. It will always see that A owned OM. And A will still block for the suspension in the 'else' clause since poll is armed due to SuspendThreadHandshake operation(even if async operation was still not added to the queue at the time A called is_suspended()), but we wouldn't need to release OM.
If is_suspended() returns true then at most we will have a false positive, since it could be that after the check somebody already resumed A, but that can also happen even with suspend_request_pending() right after releasing HandshakeState::_lock.
Is that correct or am I missing something?
-------------
PR: https://git.openjdk.java.net/jdk/pull/3191
More information about the serviceability-dev
mailing list