RFR (S) 8074355: make MutexLocker smarter about non-JavaThreads

David Holmes david.holmes at oracle.com
Mon Apr 29 00:42:48 UTC 2019


Hi Coleen,

On 27/04/2019 2:10 am, coleen.phillimore at oracle.com wrote:
> Summary: Use safepoint_check_always/safepoint_check_never instead of 
> safepoint_check_sometimes for locks that are taken by JavaThreads and 
> non-JavaThreads

To clarify: the safepoint_check_[always|never|sometimes] pertains only 
to the behaviour of JavaThreads that use the lock, independently of 
whether a lock may be used by both JavaThreads and non-JavaThreads.

> This patch moves all but 3 of the locks to not be 
> safepoint_check_sometimes.  We have plans to fix these three.  Also, 

So have you established that the reasons these were 'sometimes' locks no 
longer apply and so it is correct to change them? Or are you relying on 
testing to expose any mistaken assumptions?

> this patch allows for being explicit about safepoint checking or not 
> when the thread is a non-java thread, which is something that Kim 
> objected to in my first patch.

I don't understand what you mean by this. NJTs can currently call either 
lock() or lock_without_safepoint_check().

> Tested with mach5 tier1-3.
> 
> open webrev at http://cr.openjdk.java.net/~coleenp/2019/8074355.03/webrev
> bug link https://bugs.openjdk.java.net/browse/JDK-8074355

src/hotspot/share/gc/shared/oopStorage.cpp

How can these mutexes go from currently always needing safepoint checks 
to now never needing them? Are they in fact never used by JavaThreads?

---

src/hotspot/share/runtime/mutex.hpp

  95   void check_safepoint_state   (Thread* self, bool safepoint_check) 
   NOT_DEBUG_RETURN;

Please use the same parameter name as in the implementation: 
do_safepoint_check.

109   // Java and NonJavaThreads. When the lock is initialized with 
_safepoint_check_always,
  110   // that means that whenever the lock is acquired by a 
JavaThread, it will verify that each
  111   // acquition from a JavaThread is done with a safepoint check.

That can simplify to just:

109   // Java and NonJavaThreads. When the lock is initialized with 
_safepoint_check_always,
110   // that means that whenever the lock is acquired by a JavaThread, 
it will verify that
111   // it is done with a safepoint check.

Should we then continue:

111   // it is done with a safepoint check. In corollary when the lock
112   // is initialized with _safepoint_check_never, that means that
113   // whenever the lock is acquired by a JavaThread it will verify
114   // that it is done without a safepoint check.

?

---

38   SafepointCheckRequired not_allowed = do_safepoint_check ? 
_safepoint_check_never : _safepoint_check_always;
39   assert(!self->is_Java_thread() || _safepoint_check_required != 
not_allowed,

I found this code very difficult to understand due to some previous 
choices. All of the names that start with underscore give the illusion 
(to me) of being variables (or at least being the same kind of thing) 
but two are enum values and one is a field. Using 
this->_safepoint_check_required would at least make it clearer which is 
the field.

  43   // Allow NonJavaThreads to lock and wait with a safepoint check 
for locks that may be shared with JavaThreads.
  44   assert(self->is_Java_thread() || !do_safepoint_check || 
_safepoint_check_required != _safepoint_check_never,
  45          "NonJavaThreads can only check for safepoints if shared 
with JavaThreads");

This is very confusing: NJTs don't do safepoint checks. I think what you 
mean here is that you will allow a NJT to call lock() rather than 
lock_without_safepoint_check() but only if the mutex is "shared with 
JavaThreads". But always/sometimes/never has nothing to with whether the 
lock is shared between JTs and NJTs. I understand that a NJT-only mutex 
should, by convention, be created with _safepoint_check_never - but it 
really makes no practical difference. Further, a mutex used only by 
JavaThreads could in theory also be _safepoint_check_never.

47   // Only Threads_lock, Heap_lock and SR_lock may be 
safepoint_check_sometimes.
48   assert(_safepoint_check_required != _safepoint_check_sometimes || 
this == Threads_lock || this == Heap_lock ||
49          this->rank() == suspend_resume,
50          "Lock has _safepoint_check_sometimes %s", this->name());

This assert belongs in the constructor, not in every lock operation, as 
it depends only on the monitor instance not on the thread doing the lock.

---

Thanks,
David


> Thanks,
> Coleen


More information about the hotspot-dev mailing list