Summary
STATUS: I'm in the process of updating the page to switch from using 'contentions' → 'ref_count' for the async deflation protocol.
This page describes adding support for Async Monitor Deflation to OpenJDK. The primary goal of this project is to reduce the time spent in safepoint cleanup operations.
RFE: 8153224 Monitor deflation prolong safepoints
         https://bugs.openjdk.java.net/browse/JDK-8153224
Full Webrev: http://cr.openjdk.java.net/~dcubed/8153224-webrev/56-for-jdk13.full/ (NOT YET UPLOADED)
Inc Webrev: http://cr.openjdk.java.net/~dcubed/8153224-webrev/56-for-jdk13.inc/ (NOT YET UPLOADED)
Background
This patch for Async Monitor Deflation is based on Carsten Varming's
...
- Setting a NULL owner field to DEFLATER_MARKER with cmpxchg() forces any contending thread through the slow path. A racing thread would be trying to set the owner field.
- Making a zero contentions ref_count field a large negative value with cmpxchg() forces racing threads to retry. A racing thread would have set the owner field (after we stored DEFLATER_MARKER) and would be trying to increment the contentions ref_count field.
- If the owner field is still equal to DEFLATER_MARKER, then we have won all the races and can deflate the monitor.
...
ObjectMonitor::install_displaced_markword_in_object() is the new piece of code that handles all the racy situations with restoring an object's header asynchronously. The function is called from a few of two places (deflation , object monitor entry, and saving an ObjectMonitor* in an ObjectMonitorHandle). The restoration protocol for the object's header uses the mark bit along with the hash() value staying at zero to indicate that the object's header is being restored. Only one of the possible racing scenarios can win and the losing scenarios all adapt to the winning scenario's object header value.
3) Using "owner" or "
...
ref_count" With Interference Detection
Various code paths have been updated to recognize an owner field equal to DEFLATER_MARKER or a negative contentions ref_count field and those code paths will retry their operation. This is the shortest "Key Part" description, but don't be fooled. See "Gory Details" below.
An Example of ObjectMonitor Interference Detection
For example, when ObjectMonitor::enter() detects genuine contention via the owner field, it atomically increments the contentions field to indicate that the ObjectMonitor is busy. The thread calling entersave_om_ptr() is used to safely save an ObjectMonitor* in an ObjectMonitorHandle. ObjectSynchronizer::deflate_monitor_using_JT() is used to asynchronously deflate an idle monitor. save_om_ptr() and deflate_monitor_using_JT() can interfere with each other. The thread calling save_om_ptr() (T-entersave) is potentially racing with an Async Monitor Deflation by another JavaThread (T-deflate) so both threads have to check the result results of the raceraces.
Start of the Race
                     T-save                  ObjectMonitor              T-deflate
    T-enter           +----------------------  +-------+  ----------------+  ------------------------
    ----------------
    save_om_ptr() {         | owner=NULL            |  deflate_monitor_using_JT() {
       1> atomic inc              ref_count  | contentionsref_count=0        0   |        | 1> cmpxchg(DEFLATER_MARKER, &owner, NULL)
                                     +-----------------------+
- The data fields are at their starting values.
- The "1>" markers are showing where each thread is at for the ObjectMonitor box:
 - T-deflate is about to execute cmpxchg().
- T-
 enter hasn't done anything yet- save is about to increment the ref_count.
 
 
Racing Threads
                         T-save                  ObjectMonitor              T-deflate
    T-enter                 +-----------------------  +  -----------------------+  ---------------------
    ----------------------
    save_om_ptr() {         | owner=DEFLATER_MARKER |  deflate_monitor_using_JT() {
      1> <owneratomic isinc contended>ref_count  | ref_count=0     | contentions=0             |    cmpxchg(DEFLATER_MARKER, &owner, NULL)
    atomic inc contentions                          +-----------------------+    :
                                                        1> prev = cmpxchg(-max_jint, &contentionsref_count, 0)
- T-deflate has executed cmpxchg() and set owner to DEFLATE_MARKER.
- T-save still hasn't done anything yet
- The "1>" markers are showing where each thread is at for the ObjectMonitor box:enter has observed the contended owner field.
 enter- save and T-deflate are racing to update the 
 contentions
 
T-deflate Wins
                              T-save                             ObjectMonitor              T-deflate
    T-enter                          +-----------------------+  --------    +-----------------------+  -------------
    -------------------------------  | 
    save_om_ptr() {                    | owner=DEFLATER_MARKER |  deflate_monitor_using_JT() {
    <owner is contended>        atomic inc ref_count             | contentionsref_count=-max_jint+1 |    cmpxchg(DEFLATER_MARKER, &owner, NULL)
      atomic1> incif contentions    (owner == DEFLATER_MARKER  &&   +-----------------------+    :
    if (contentions      ref_count <= 0) &&{                       ||                                         prev = cmpxchg(-max_jint, &contentionsref_count, 0)
        ownerrestore == DEFLATER_MARKER) {    obj header                            if (prev     == 0 &&
      restore obj header                                                    \/             1> if (prev == 0 &&
        atomic dec ref_count           +-----------------------+        owner == DEFLATER_MARKER) {
          2> retryreturn enter                         false to force retry    | owner=DEFLATER_MARKER |                     restore obj header
    }                                          }                                | ref_count=-max_jint                   |   2> finish the deflation
                                                          +-----------------------+        }
- This diagram starts after "Racing Threads".
- The "1>" markers are showing where each thread is at for that ObjectMonitor box:
 enter- save and T-deflate both observe owner == DEFLATER_MARKER and a negative 
 contentions
- T-enter save has lost the race, : it restores the obj header and it retries(not shown) and decrements the ref_count.
- T-deflate restores the obj header (not shown).
- The "2>" markers are showing where each thread is at for that ObjectMonitor box.
- T-save returns false to cause the caller to retry.
- T-deflate finishes the deflation and it finishes deflation of the ObjectMonitor.
 
T-
...
save Wins
                        T-save                           ObjectMonitor               ObjectMonitor              T-deflate
    T-enter                          +-----------------------+  -------    +-----------------------+  --------------
    -------------------------------
    save_om_ptr() {                    | owner=DEFLATER_MARKER |  deflate_monitor_using_JT() {
    <owner is contended> atomic inc ref_count             | contentions=1        ref_count=1           |    cmpxchg(DEFLATER_MARKER, &owner, NULL)
      1> atomicif inc(owner contentions ==        DEFLATER_MARKER &&  +-----------------------+    :
    if (contentions <= 0 &&           ref_count <= 0) {                                           ||                prev = cmpxchg(-max_jint, &contentionsref_count, 0)
         } else {         owner == DEFLATER_MARKER) {                                 if (prev == 0 &&
    }\/         else {     1>                   if (prev == 0 &&
      save om_ptr               in the               +-----------------------+           owner == DEFLATER_MARKER) {
      do contended ObjectMonitorHandle            | owner=NULL            |    } else {
   2> return true              } else {
      | ref_count=1    enter      work |      cmpxchg(NULL, &owner, DEFLATER_MARKER)
                                         cmpxchg(NULL, &owner, DEFLATER_MARKER)+-----------------------+   2> return
- This diagram starts after "Racing Threads".
- The "1>" markers are showing where each thread is at for the ObjectMonitor box:
 enter- save and T-deflate both observe a 
 contentions
- T-enter save has won the race and it proceeds with the normal contended enter worksaves the ObjectMonitor* in the ObjectMonitorHandle (not shown).
- T-deflate detects that it has lost the race (prev != 0) and bails out on deflating the ObjectMonitor:
 - Before bailing out T-deflate tries to restore the owner field to NULL if it is still DEFLATER_MARKER.
 
- The "2>" markers are showing where each thread is at for that ObjectMonitor box.
 
T-enter T-enter Wins By A-B-A
                         T-enter                                       ObjectMonitor                T-deflate
    T-enter                                     +---------------------------+  -----------------  +-------------------------
   +  ------------------------------------------
    ObjectMonitor::enter() {                      | owner=DEFLATER_MARKER   |  deflate_monitor_using_JT() {
      <owner is contended>                        | contentions=1          ref_count=1             |    cmpxchg(DEFLATER_MARKER, &owner, NULL)
      1> atomicEnterI() inc{ contentions                                  +-------------------------+ 1> :
 1>        if (contentionsowner <== 0DEFLATER_MARKER &&                                               ||              2> : <thread_stalls>
           owner == DEFLATER_MARKER) { cmpxchg(Self, &owner,                          \/                       \/          :
    } else {    :
                    DEFLATER_MARKER)               +-------------------------+    :
      EnterI()              ==      DEFLATER_MARKER) {              | owner=Self/T-enter      |    :
        cmpxchg(Self, &owner, DEFLATER_MARKER)  | contentions=0 // EnterI is done                       | ref_count=0             |    : <thread_resumes>
          return            atomic dec contentions                    +-------------------------+    prev = cmpxchg(-max_jint, &contentionsref_count, 0)
   2>   }  }                                                     ||                 if (prev == 0 &&
    } // finished with enter enter() is done                                      \/                  3>   owner == DEFLATER_MARKER) {
 3> : <does appatomic work>dec ref_count                          +-------------------------+    } else {
 2> : <does exit()app monitorwork>                              | owner=Self/T-enter|NULL |      cmpxchg(NULL, &owner, DEFLATER_MARKER)
 3> :     owner   = NULL                              | contentions=0     | ref_count=-max_jint     |      atomic add max_jint to contentionsref_count
                exit() monitor                                +-------------------------+   3>4> bailout on deflation
   4> owner = NULL                                            ||                 }
                                                              }\/
                                                  +-------------------------+
                                                  | owner=Self/T-enter|NULL |
                                                  | ref_count=0             |
                                                  +-------------------------+
- T-deflate has executed cmpxchg() and set owner to DEFLATE_MARKER.
- T-enter has called ObjectMonitor::enter() with "ref_count == 1", noticed that the owner is contended and is about to call ObjectMonitor::EnterI().
- The first 
- This diagram starts after "Racing Threads".
- T-enter incremented contentions to 1.
- The first ObjectMonitor box is showing the fields at this point and the "1>" markers are showing where each thread is at for that ObjectMonitor box.
- T-deflate stalls after setting the owner field to DEFLATER_MARKER.
- T-enter has won the race and calls EnterI() to do the contended enter work.- EnterI() observes owner == DEFLATER_MARKER and uses cmpxchg() to set the owner field to Self/T-enter.
- T-enter decrements the contentions field because it is no longer contending for the monitor; it owns the monitor.
 
- The second ObjectMonitor box is showing the fields at this point and the "2>" markers are showing where each thread is at for that ObjectMonitor box.
- T-deflate resumes, sets the contentions field to -max_jint (not shown), and passes the first part of the bailout expression because "prev == 0".
- T-deflate observes that "owner != DEFLATE_MARKER" and bails out on deflation:
 - Depending on when T-deflate resumes after the stall, it will see "owner == T-enter" or "owner == NULL".
- Both of those values will cause deflation to bailout so we have to conditionally undo work:
 - restore the owner field to NULL if it is still DEFLATER_MARKER (it's not DEFLATER_MARKER)
- undo setting contentions to -max_jint by atomically adding max_jint to contentions which will restore contentions to its proper value.
 
 
- The third ObjectMonitor box is showing the fields at this point and the "3>1>" markers are showing where each thread is at for that ObjectMonitor box.
- T-deflate stalls after setting the owner field to DEFLATER_MARKER.
- If the T-enter thread has managed to enter but not exit the monitor during the T-deflate stall, then our owner field A-B-A transition is:
 NULL → DEFLATE_MARKER → calls EnterI() to do the contended enter work.- EnterI() observes owner == DEFLATER_MARKER and uses cmpxchg() to set the owner field to Self/T-enter.
 so we really have A1-B-A2, but the A-B-A principal still holds.
- If the T-enter thread has managed to enter and exit the monitor during the T-deflate stall, then our owner field A-B-A transition is:
 NULL → DEFLATE_MARKER → Self/T-enter  → NULL
 so we really have A-B1-B2-A, but the A-B-A principal still holds.
 
T-enter and T-deflate Both Lose
This subsection is pure theory right now. I don't have a failing test case that illustrates this race result.
After working out the bug described in the "T-deflate and T-hash Both Lose" subsection below, it is time to take a closer look at the T-enter versus T-deflate race. For analysis of this race to make sense, the ref_count field has to be introduced in this subsection instead of in the "Hashcodes and Object Header Interference" section below.
- T-enter owns the monitor, returns from EnterI(), and returns from enter().
- The ObjectMonitorHandle destructor decrements the ref_count.
 
- T-enter is now ready to do work that requires the monitor to be owned.
- The second ObjectMonitor box is showing the fields at this point and the "2>" markers are showing where each thread is at for that ObjectMonitor box.
- T-enter is doing app work (but it also could have finished and exited the monitor).
- T-deflate resumes, sets the ref_count field to -max_jint, and passes the first part of the bailout expression because "prev == 0".
- The third ObjectMonitor box is showing the fields at this point and the "3>" markers are showing where each thread is at for that ObjectMonitor box.
- T-deflate performs the A-B-A check which observes that "owner != DEFLATE_MARKER" and bails out on deflation:
 - Depending on when T-deflate resumes after the stall, it will see "owner == T-enter" or "owner == NULL".
- Both of those values will cause deflation to bailout so we have to conditionally undo work:
 - restore the owner field to NULL if it is still DEFLATER_MARKER (it's not DEFLATER_MARKER)
- undo setting ref_count to -max_jint by atomically adding max_jint to ref_count which will restore ref_count to its proper value.
 
- If the T-enter thread has managed to enter but not exit the monitor during the T-deflate stall, then our owner field A-B-A transition is:
- so we really have A1-B-A2, but the A-B-A principal still holds. 
- If the T-enter thread has managed to enter and exit the monitor during the T-deflate stall, then our owner field A-B-A transition is: 
- so we really have A-B1-B2-A, but the A-B-A principal still holds. 
 
- T-enter finished doing app work and is about to exit the monitor (or it has already exited the monitor). 
- The fourth ObjectMonitor box is showing the fields at this point and the "4>" markers are showing where each thread is at for that ObjectMonitor box. 
 
After T-deflate has won the race for deflating an ObjectMonitor it has to restore the header in the associated object. Of course another thread can be trying to do something to the object's header at the same time. Isn't asynchronous work exciting?!?!
ObjectMonitor::install_displaced_markword_in_object() is called from two places so we can have a race between a T-save thread and a T-deflate thread:
Start of the Race
    T-save                                       object           T-deflate
    --------                                                ObjectMonitor                T-deflate
    T-enter                                     +-------------------------+  -----------------------------------------------
    ------------------------------------------  | owner=DEFLATER_MARKER   |  deflate_monitor_using_JT() {
    ref_count inc by ObjectMonitorHandle        | contentions=0           |    cmpxchg(DEFLATER_MARKER, &owner, NULL)
    <owner is contended>                        | ref_count=1             |    if (waiters != 0 || ref_count != 0) {
 1> atomic inc contentions                      +-------------------------+    }
    if (contentions <= 0 &&----------  +-------------+  --------------------------------------------
    install_displaced_markword_in_object() {     | mark=om_ptr |  install_displaced_markword_in_object() {
                          dmw ||= header()                             +-------------+   1> prevdmw = cmpxchg(-max_jint, &contentions, 0)
    header()
      if (!dmw->is_marked() &&                                      if (!dmw->is_marked() &&
          ownerdmw->hash() == DEFLATER_MARKER0) {                                           dmw->hash() == 0) {
        create marked_dmw                                \/               2> if (prev == 0 &&
  2>  restore obj header                        +-------------------------+        owner == DEFLATER_MARKER &&create marked_dmw
        dmw = cmpxchg(marked_dmw, &header, dmw)                        dmw = cmpxchg(marked_dmw, &header, dmw)
      retry} enter                               | owner=DEFLATER_MARKER   |        cmpxchg(-max_jint, &ref_count, 0) == 0) {
    }         }
- The data field (mark) is at its starting value.
- 'dmw' and 'marked_dmw' are local copies in each thread.
- T-save and T-deflate are both calling install_displaced_markword_in_object() at the same time.
- Both threads are poised to call cmpxchg() at the same time.
 
T-deflate Wins First Race
    T-save                                       object                                        | contentions=-max_jint   |      restore obj header
    T-deflate
    -------------------------------------------  +-------------+   -------------------------------------------
    install_displaced_markword_in_object() {     | mark=om_ptr |   install_displaced_markword_in_object() {
      dmw = header()                             +-------------+     dmw = header()
      if (!dmw->is_marked() &&                   | ref_count=1             |      finish the deflation
                                                +-------------------------+    } else {  
                                                                                 cmpxchg(NULL, &owner, DEFLATER_MARKER)
                                       if (!dmw->is_marked() &&
          dmw->hash() == 0) {                                            dmw->hash() == 0) {
        create marked_dmw                                              create marked_dmw
        dmw = cmpxchg(marked_dmw, &header, dmw)                        dmw = cmpxchg(marked_dmw, &header, dmw)
      }                                                              }
      // dmw == marked_dmw here                                      // dmw == original dmw here
      if (dmw->is_marked())                                          if (dmw->is_marked())
      unmark dmw                                                     unmark dmw
      obj = object()                                                 obj = object()
      obj->cas_set_mark(dmw, this)                                   obj->cas_set_mark(dmw, this)
- The return value from cmpxchg() in each thread will be different.
- Since T-deflate won the race, its 'dmw' variable contains the header/dmw from the ObjectMonitor.
- Since T-save lost the race, its 'dmw' variable contains the 'marked_dmw' set by T-deflate.- T-save will unmark its 'dmw' variable.
 
- Both threads are poised to call cas_set_mark() at the same time.
 
T-save Wins First Race
    T-save                                       object            T-deflate
    -------------------------------------------  +-------------+   -------------------------------------------
    install_displaced_markword_in_object() {     | mark=om_ptr |   install_displaced_markword_in_object() {
      dmw = header()                             +-------------+     dmw = header()
      if   (!dmw->is_marked() &&                atomic add max_jint to contentions
                    if (!dmw->is_marked() &&
          dmw->hash() == 0) {                                            dmw->hash() == 0) {
        create marked_dmw                                              create marked_dmw
        dmw = cmpxchg(marked_dmw, &header, dmw)                        dmw = cmpxchg(marked_dmw, &header, dmw)
      }                                                              }
      // dmw == original dmw here                                    // dmw ==      bailout on deflation
marked_dmw here
      if (dmw->is_marked())                                          if (dmw->is_marked())
        unmark dmw                                                     unmark dmw
      obj = object()                                                 obj = object()
      obj->cas_set_mark(dmw, this)                                                                               }
- T-deflate has set owner to DEFLATER_MARKER and made it past the first ref_count check.
- T-deflate is about change contentions to -max_jint.
- T-enter calls ObjectMonitor::enter() with an ObjectMonitorHandle already in place so "ref_count == 1".
- T-enter detects that owner is contended and is about to increment contentions.
- The first ObjectMonitor box is showing the fields at this point and the "1>" markers are showing where each thread is at for that ObjectMonitor box.
- T-deflate set the contentions field to -max_jint and is about to make the last of the protocol checks.
- T-enter sees "contentions <= 0 && owner == DEFLATER_MARKER" and realizes that it has lost the race.
- The second ObjectMonitor box is showing the fields at this point and the "2>" markers are showing where each thread is at for that ObjectMonitor box.
- T-enter restores the obj header (not shown) and the object is now disconnected from the ObjectMonitor.
- T-deflate sees "prev == 0 && owner == DEFLATER_MARKER" and tries to set the ref_count field to -max_jint.
- T-enter returns false to its caller to cause a retry (which will also decrement the ref_count in the ObjectMonitorHandle destructor).
- If T-enter manages to decrement the ref_count before T-deflate calls cmpxchg():- T-deflate will change the ref_count field to -max_jint
- T-deflate will restore the obj header (not shown and already done) and finish the deflation
- T-enter will retry and there is no problem here (luckily).
 
- If T-deflate calls cmpxchg() while "ref_count == 1" that will fail:
 - T-deflate bails out on deflation but it has to restore some data if possible:
 - The return value of cmpxchg() is not checked here.
- If T-deflate cannot restore the owner field to NULL, then another thread has managed to enter the monitor (or enter and exit the monitor) and we don't want to overwrite that information.
- Add back max_jint to restore the contentions field to its proper value (which may not be the same as when we started).
 
- T-enter will retry, but the ObjectMonitor is still on the in-use list and still thinks it is associated with the object. Ouch!
 
 
I have to look at this new theory with fresh eyes, but if it holds together, then T-enter's "contentions <= 0 && owner == DEFLATER_MARKER" check will need to be changed to "contentions <= 0 && owner == DEFLATER_MARKER && ref_count <= 0" as was done for save_om_ptr().
After T-deflate has won the race for deflating an ObjectMonitor it has to restore the header in the associated object. Of course another thread can be trying to do something to the object's header at the same time. Isn't asynchronous work exciting?!?!
ObjectMonitor::install_displaced_markword_in_object() is called from two places so we can have a race between a T-enter thread and a T-deflate thread:
Start of the Race
   obj->cas_set_mark(dmw, this)
- This diagram is the same as "T-deflate Wins First Race" except we've swapped the post cmpxchg() comments.
- Since T-save won the race, its 'dmw' variable contains the header/dmw from the ObjectMonitor.
- Since T-deflate lost the race, its 'dmw' variable contains the 'marked_dmw' set by T-save.- T-deflate will unmark its 'dmw' variable.
 
- Both threads are poised to call cas_set_mark() at the same time.
 
Either Wins the Second Race
    T-save                                       object            T-deflate
    -----------------------------    T-enter                                      object           T-deflate
    --------------  +-------------+   ----------------  +-------------+  --------------------------------------------
    install_
    install_displaced_markword_in_object() {       | mark=om_ptrdmw    |   install_displaced_markword_in_object() {
         dmw = header()                                               +-------------+     dmw = header()
         if (!dmw->is_marked() &&                                                                            if (!dmw->is_marked() &&
                 dmw->hash() == 0) {                                                                                    dmw->hash() == 0) {
             create marked_dmw                 dmw                                              create marked_dmw
        dmw                      create marked_dmw
        dmw = cmpxchg(marked= cmpxchg(marked_dmw, &header, dmw)                                             dmw = cmpxchg(marked_dmw, &header, dmw)
      }                                                              }
      // dmw == ...                                                  // dmw == ...
      if (dmw->is_marked())                                          if (dmw->is_marked())
        unmark dmw                                                     unmark dmw
      obj = object()                                                 obj = object()
      obj->cas_set_mark(dmw, this)                                      }
- The data field (mark) is at its starting value.
- 'dmw' and 'marked_dmw' are local copies in each thread.
- T-enter and T-deflate are both calling install_displaced_markword_in_object() at the same time.
- Both threads are poised to call cmpxchg() at the same time.
 
...
obj->cas_set_mark(dmw, this)
- It does not matter whether T-save or T-deflate won the cmpxchg() call so the comment does not say who won.
- It does not matter whether T-save or T-deflate won the cas_set_mark() call; in this scenario both were trying to restore the same value.
- The object's mark field has changed from 'om_ptr' → 'dmw'.
 
Please notice that install_displaced_markword_in_object() does not do any retries on any code path:
- Instead the code adapts to being the loser in a cmpxchg() by unmarking its copy of the dmw.
- In the second race, if a thread loses the cas_set_mark() race, there is also no need to retry because the object's header has been restored by the other thread.
 
Hashcodes and Object Header Interference
If we have a race between a T-deflate thread and a thread trying to get/set a hashcode (T-hash), then the race is between the ObjectMonitorHandle.save_om_ptr(obj, mark) call in T-hash and deflation protocol in T-deflate.
Start of the Race
    T-enter                                   hash                   ObjectMonitor              object            T-deflate
    ----------------------------  +---------------  +-------------+    -------------------------------------------
    installsave_displaced_markword_in_objectom_ptr() {             | markowner=om_ptrNULL            |   installdeflate_displacedmonitor_markwordusing_in_objectJT() {
        :                       dmw = header()                             +| ref_count=0           | 1> cmpxchg(DEFLATER_MARKER, &owner, NULL)
   1> atomic inc ref_count  +-----------------------+
- The data fields are at their starting values.
- T-deflate is about to execute cmpxchg().
- T-hash is about to increment ref_count.
- The "1>" markers are showing where each thread is at for the ObjectMonitor box.
 
Racing Threads
    T-hash                  ObjectMonitor              T-deflate
    ----------------------  +-----------------------+  ------------------------------------------
    save_om_ptr() {         | owner=DEFLATER_MARKER |  deflate_monitor_using_JT() {
      :  +     dmw = header()
      if (!dmw->is_marked() &&                             | ref_count=0           |   if cmpxchg(!dmw->is_marked() &&
          dmw->hash() == 0) {                                            dmw->hash() ==DEFLATER_MARKER, &owner, NULL)
   1> atomic inc ref_count  +-----------------------+    if (contentions != 0 || waiters != 0) {
                                   create marked_dmw                                              create marked_dmw
        dmw = cmpxchg(marked_dmw, &header, dmw)                        dmw = cmpxchg(marked_dmw, &header, dmw)
      }                                                              }
      // dmw == marked_dmw here                                      // dmw == original dmw here
      if (dmw->is_marked())                                          if (dmw->is_marked())
      unmark dmw                                                     unmark dmw
      obj = object()                                                 obj = object()
      obj->cas_set_mark(dmw, this)                                   obj->cas_set_mark(dmw, this)
- The return value from cmpxchg() in each thread will be different.
- Since T-deflate won the race, its 'dmw' variable contains the header/dmw from the ObjectMonitor.
- Since T-enter lost the race, its 'dmw' variable contains the 'marked_dmw' set by T-deflate.- T-enter will unmark its 'dmw' variable.
 
- Both threads are poised to call cas_set_mark() at the same time.
 
T-enter Wins First Race
                  }
                                                      1> prev = cmpxchg(-max_jint, &ref_count, 0)
- T-deflate has set the owner field to DEFLATER_MARKER.
- The "1>" markers are showing where each thread is at for the ObjectMonitor box:- T-deflate is about to execute cmpxchg().
- T-save is about to increment the ref_count.
 
 
T-deflate Wins
If T-deflate wins the race, then T-hash will have to retry at most once.
    T-hash                      ObjectMonitor                 T-enter                                      object            T-deflate
    -----------------------------   +--------------  +-------------+    -------------------------------------------
    installsave_displaced_markword_in_objectom_ptr() {               | markowner=omDEFLATER_ptrMARKER |   installdeflate_displacedmonitor_markwordusing_in_objectJT() {
      1> atomic dmwinc = header()                 ref_count      | ref_count=-max_jint   |    cmpxchg(DEFLATER_MARKER, &owner, NULL)
      if (owner ==              +-----------------------+    if dmw(contentions != header()
      if (!dmw->is_marked() &&0 || waiters != 0) {
          DEFLATER_MARKER &&               ||                }
          ref_count <= 0) {                \/                prev if= cmpxchg(!dmw->is_marked() &&
          dmw->hash() == 0) {                                            dmw->hash()max_jint, &ref_count, 0)
        restore obj header      +-----------------------+ 1> if (prev == 0) {
   &&
          atomic createdec markedref_dmw                                        count       create| marked_dmw
    owner=DEFLATER_MARKER |        dmwowner == cmpxchg(marked_dmw, &header, dmw)                        dmw = cmpxchg(marked_dmw, &header, dmw)
      }                                                              }
      // dmw == original dmw here                                    // dmw == marked_dmw here
      if (dmw->is_marked())                                          if (dmw->is_marked())
        unmark dmw                                                     unmark dmw
      obj = object()                                                 obj = object()
      obj->cas_set_mark(dmw, this)                                   obj->cas_set_mark(dmw, this)
- This diagram is the same as "T-deflate Wins First Race" except we've swapped the post cmpxchg() comments.
- Since T-enter won the race, its 'dmw' variable contains the header/dmw from the ObjectMonitor.
- Since T-deflate lost the race, its 'dmw' variable contains the 'marked_dmw' set by T-enter.- T-deflate will unmark its 'dmw' variable.
 
- Both threads are poised to call cas_set_mark() at the same time.
 
Either Wins the Second Race
DEFLATER_MARKER) {
     2> return false to         | ref_count=-max_jint   |      restore obj header
          cause a retry         +-----------------------+   2> finish the deflation
      }                                                      }- T-deflate made it past the cmpxchg() before T-hash incremented it.
- T-deflate set the ref_count field to -max_jint and is about to make the last of the protocol checks.
- The first ObjectMonitor box is showing the fields at this point and the "1>" markers are showing where each thread is at for that ObjectMonitor box.
- T-deflate sees "prev == 0 && owner == DEFLATER_MARKER" so it knows that it has won the race.
- T-deflate restores obj header (not shown).
- T-hash increments the ref_count.
- T-hash observes "owner == DEFLATER_MARKER && ref_count <= 0" so it restores obj header (not shown) and decrements ref_count.
- The second ObjectMonitor box is showing the fields at this point and the "2>" markers are showing where each thread is at for that ObjectMonitor box.
- T-deflate finishes the deflation work.
- T-hash returns false to cause a retry and when T-hash retries:
 - it observes the restored object header (done by T-hash or T-deflate):- if the object's header does not have a hash, then generate a hash and merge it with the object's header.
- Otherwise, extract the hash from the object's header and return it.
 
 
 
T-hash Wins
If T-hash wins the race, then the ref_count will cause T-deflate to bail out on deflating the monitor.
Note: header is not mentioned in any of the previous sections for simplicity.
    T-hash                      ObjectMonitor              T-deflate
        T-enter                                      object            T-deflate
    ---------------------------------   +----------  +-------------+    -------------------------------------------
    installsave_displaced_markword_in_objectom_ptr() {               | mark=dmw   header=dmw_no_hash    |   installdeflate_displacedmonitor_markwordusing_in_objectJT() {
        atomic   dmw = header()                  inc ref_count      |        +-------------+     dmw = header()
      if (!dmw->is_marked() &&    owner=DEFLATER_MARKER |    cmpxchg(DEFLATER_MARKER, &owner, NULL)
   1> if (owner ==              | ref_count=1           |    if (contentions != 0 || waiters != 0) {
          DEFLATER_MARKER &&    +-----------------------+    }
          ref_count <= 0) {    if (!dmw->is_marked() &&
          dmw->hash() == 0) {                                            dmw->hash() == 0) {
        create marked_dmw                                              create marked_dmw
        dmw ||             1> prev = cmpxchg(marked-max_dmwjint, &headerref_count, dmw)                   0)
      } else {      dmw = cmpxchg(marked_dmw, &header, dmw)
      }                                                              }
                  \// dmw                if (prev == ... 0 &&
     2> save om_ptr in the      +-----------------------+                                        // dmwowner == ...
    DEFLATER_MARKER) {
          if (dmw->is_marked())                                          if (dmw->is_marked())
        unmark dmw                                                     unmark dmw
      obj = object()                                                 obj = object()
      obj->cas_set_mark(dmw, this)                                   obj->cas_set_mark(dmw, this)
- It does not matter whether T-enter or T-deflate won the cmpxchg() call so the comment does not say who won.
- It does not matter whether T-enter or T-deflate won the cas_set_mark() call; in this scenario both were trying to restore the same value.
- The object's mark field has changed from 'om_ptr' → 'dmw'.
 
Please notice that install_displaced_markword_in_object() does not do any retries on any code path:
- Instead the code adapts to being the loser in a cmpxchg() by unmarking its copy of the dmw.
- In the second race, if a thread loses the cas_set_mark() race, there is also no need to retry because the object's header has been restored by the other thread.
 
Hashcodes and Object Header Interference
If we have a race between a T-deflate thread and a thread trying to get/set a hashcode (T-hash), then the race is between the ObjectMonitorHandle.save_om_ptr(obj, mark) call in T-hash and deflation protocol in T-deflate.
Note: ref_count is not mentioned in any of the previous sections for simplicity.
Start of the Race
    T-hash                  ObjectMonitor              T-deflate
    ----------------------  +-----------------------+  ----------------------------------------
    save_om_ptr() {         | owner=NULL            |  deflate_monitor_using_JT() {
      :                     | contentions=0         | 1> cmpxchg(DEFLATER_MARKER, &owner, NULL)
   1> atomic inc ref_count  | ref_count=0           |
                            +-----------------------+
- The data fields are at their starting values.
- T-deflate is about to execute cmpxchg().
- T-hash is about to increment ref_count.
- The "1>" markers are showing where each thread is at for the ObjectMonitor box.
 
Racing Threads
    T-hash                  ObjectMonitor              T-deflate
    ----------------------  +-----------------------+  --------------------------------------------
    save_om_ptr() {         | owner=DEFLATER_MARKER |  deflate_monitor_using_JT() {
      :                     | contentions=0         |    cmpxchg(DEFLATER_MARKER, &owner, NULL)
   1> atomic inc ref_count  | ref_count=0           | 1> if (waiters != 0 || ref_count != 0) {
                            +-----------------------+    }
                                                         prev = cmpxchg(-max_jint, &contentions, 0)
- T-deflate has set the owner field to DEFLATER_MARKER.
- T-deflate is about to check the waiters and ref_count fields.
- T-hash is about to inc the ref_count field (T-hash has made no progress).
- The "1>" markers are showing where each thread is at for the ObjectMonitor box.
 
T-deflate Wins
If T-deflate wins the race, then T-hash will have to retry at most once.
    T-hash                      ObjectMonitor              T-deflate
    -------------------------   +-----------------------+  -----------------------------------------------
    save_om_ptr() {             | owner=DEFLATER_MARKER |  deflate_monitor_using_JT() {
   1> atomic inc ref_count      | contentions=-max_jint |    cmpxchg(DEFLATER_MARKER, &owner, NULL)
      if (owner ==              | ref_count=0           |    if (waiters != 0 || ref_count != 0) {
          DEFLATER_MARKER &&    +-----------------------+    }
          contentions <= 0 &&              ||                prev = cmpxchg(-max_jint, &contentions, 0)
          ref_count <= 0) {                \/             1> if (prev == 0 &&
        restore obj header      +-----------------------+        owner == DEFLATER_MARKER &&
        atomic dec ref_count    | owner=DEFLATER_MARKER |        cmpxchg(-max_jint, &ref_count, 0) == 0) {
     2> return false to         | contentions=-max_jint |      restore obj header
          cause a retry         | ref_count=-max_jint   |   2> finish the deflation
      }                         +-----------------------+    }
- T-deflate made it past the first ref_count check before T-hash incremented it.
- T-deflate set the contentions field to -max_jint and is about to make the last of the protocol checks.
- The first ObjectMonitor box is showing the fields at this point and the "1>" markers are showing where each thread is at for that ObjectMonitor box.
- T-deflate sees "prev == 0 && owner == DEFLATER_MARKER" and sets the ref_count field to -max_jint so it has won the race.
- T-deflate restores obj header (not shown).
- T-hash increments the ref_count.
- T-hash observes "owner == DEFLATER_MARKER && contentions <= 0 && ref_count <= 0" so it restores obj header (not shown) and decrements ref_count.
- The second ObjectMonitor box is showing the fields at this point and the "2>" markers are showing where each thread is at for that ObjectMonitor box.
- T-deflate finishes the deflation work.
- T-hash returns false to cause a retry and when T-hash retries:
 - it observes the restored object header (done by T-hash or T-deflate):- if the object's header does not have a hash, then generate a hash and merge it with the object's header.
- Otherwise, extract the hash from the object's header and return it.
 
 
 
T-hash Wins Scenario 1
If T-hash wins the race, then the ref_count will cause T-deflate to bail out on deflating the monitor.
Note: header is not mentioned in any of the previous sections for simplicity.
    T-hash                      ObjectMonitor              T-deflate
    -------------------------   +-----------------------+  --------------------------------------------
    save_om_ptr() {             | header=dmw_no_hash    |  deflate_monitor_using_JT() {
      atomic inc ref_count      | owner=DEFLATER_MARKER |    cmpxchg(DEFLATER_MARKER, &owner, NULL)
   1> if (owner ==              | contentions=0         | 1> if (waiters != 0 || ref_count != 0) {
          DEFLATER_MARKER &&    | ref_count=1           |      cmpxchg(NULL, &owner, DEFLATER_MARKER)
          contentions <= 0 &&   +-----------------------+   2> bailout on deflation
          ref_count <= 0) {                ||                }
      }                                    \/                prev = cmpxchg(-max_jint, &contentions, 0)
      if (object no longer      +-----------------------+
          has a monitor or      | header=dmw_no_hash    |
          is a different        | owner=NULL            |
          monitor) {            | contentions=0         |
        atomic dec ref_count    | ref_count=1           |
        return false to         +-----------------------+
          cause a retry                    ||
      }                                    \/
   2> save om_ptr in the        +-----------------------+
        ObjectMonitorHandle     | header=dmw_hash       |
    }                           | owner=NULL            |
    if save_om_ptr() {          | contentions=0         |
      if no hash                | ref_count=1           |
        gen hash & merge        +-----------------------+
      hash = hash(header)
    }
 3> atomic dec ref_count
    return hash
- T-hash has incremented ref_count before T-deflate made it past that check.
- The first ObjectMonitor box is showing the fields at this point and the "1>" markers are showing where each thread is at for that ObjectMonitor box.
- T-deflate bails out on deflation, but first it tries to restore the owner field:
 - The return value of cmpxchg() is not checked here.
- If T-deflate cannot restore the owner field to NULL, then another thread has managed to enter the monitor (or enter and exit the monitor) and we don't want to overwrite that information.
 
- T-hash observes:- "owner == DEFLATER_MARKER && contentions == 0 && ref_count > 0" or
- "owner == NULL && contentions == 0 && ref_count > 0" so it does not cause a retry.
 
- T-hash verifies that the object still has a monitor and that monitor still refers to our current ObjectMonitor.
- The second ObjectMonitor box is showing the fields at this point and the "2>" markers are showing where each thread is at for that ObjectMonitor box.
- T-hash saves the ObjectMonitor* in the ObjectMonitorHandle (not shown) and returns to the caller.
- save_om_ptr() returns true since the ObjectMonitor is safe:- if ObjectMonitor's 'header/dmw' field does not have a hash, then generate a hash and merge it with the 'header/dmw' field.
- Otherwise, extract the hash from the ObjectMonitor's 'header/dmw' field.
 
- The third ObjectMonitor box is showing the fields at this point and the "3>" marker is showing where T-hash is at for that ObjectMonitor box.
- T-hash decrements the ref_count field.
- T-hash returns the hash value.
 
T-hash Wins Scenario 2
In this T-hash wins scenario, the need for setting ref_count to a large negative value in the third part of the protocol is illustrated.
    T-hash                      ObjectMonitor              T-deflate
    -------------------------   +-----------------------+  -----------------------------------------------
    save_om_ptr() {             | header=dmw_no_hash    |  deflate_monitor_using_JT() {
      atomic inc ref_count      | owner=DEFLATER_MARKER |    cmpxchg(DEFLATER_MARKER, &owner, NULL)
      if (owner ==              | contentions=0         |    if (waiters != 0 || ref_count != 0) {
          DEFLATER_MARKER &&    | ref_count=1           |    }
          contentions <= 0 &&   +-----------------------+ 1> prev = cmpxchg(-max_jint, &contentions, 0)
          ref_count <= 0) {                ||             2> if (prev == 0 &&
      }                                    \/                    owner == DEFLATER_MARKER &&
   1> if (object no longer      +-----------------------+        cmpxchg(-max_jint, &ref_count, 0) == 0) {
          has a monitor or      | header=dmw_no_hash    |    } else {
          is a different        | owner=DEFLATER_MARKER |      cmpxchg(NULL, &owner, DEFLATER_MARKER)
          monitor) {            | contentions=-max_jint |      atomic add max_jint to contentions
        atomic dec ref_count    | ref_count=1           |   3> bailout on deflation
        return false to         +-----------------------+    }
          cause a retry                    ||
      }                                    \/
   2> save om_ptr in the        +-----------------------+
        ObjectMonitorHandle     | header=dmw_hash       |
    }                           | owner=NULL            |
    if save_om_ptr() {          | contentions=0         |
      if no hash                | ref_count=1           |
        gen hash & merge        +-----------------------+
      hash = hash(header)
    }
 3> atomic dec ref_count
    return hash
- T-deflate made it past the first ref_count check before T-hash incremented it.
- T-hash made it past the "owner == DEFLATER_MARKER && contentions <= 0 && ref_count <= 0" check before T-deflate updated contentions.
- The first ObjectMonitor box is showing the fields at this point and the "1>" markers are showing where each thread is at for that ObjectMonitor box.
- T-deflate sets the contentions field to -max_jint and is about the make the last of the protocol checks.
- T-hash verifies that the object still has a monitor and that monitor still refers to our current ObjectMonitor.
- The second ObjectMonitor box is showing the fields at this point and the "2>" markers are showing where each thread is at for that ObjectMonitor box.
- T-deflate sees "prev == 0 && owner == DEFLATER_MARKER" and fails to set the ref_count field to -max_jint so it has lost the race.
- T-deflate bails out on deflation but it has to restore some data if possible:
 - The return value of cmpxchg() is not checked here.
- If T-deflate cannot restore the owner field to NULL, then another thread has managed to enter the monitor (or enter and exit the monitor) and we don't want to overwrite that information.
- Add back max_jint to restore the contentions field to its proper value (which may not be the same as when we started).
 
- T-hash saves the ObjectMonitor* in the ObjectMonitorHandle (not shown) and returns to the caller.
- save_om_ptr() returns true since the ObjectMonitor is safe:- if ObjectMonitor's 'header/dmw' field does not have a hash, then generate a hash and merge it with the 'header/dmw' field.
- Otherwise, extract the hash from the ObjectMonitor's 'header/dmw' field.
 
- The third ObjectMonitor box is showing the fields at this point and the "3>" markers are showing where each thread is at for that ObjectMonitor box.
- T-hash decrements the ref_count field.
- T-hash returns the hash value.
 
T-deflate and T-hash Both Lose
This subsection title is NOT a typo. It was previously possible for both T-deflate and T-hash to lose the race. In the "CR0/v2.00/3-for-jdk13" version of the code, the double loss was not an issue. The addition of "restore obj header" to save_om_ptr() created a bug where the ObjectMonitor can be disconnected from the object without the ObjectMonitor being deflated. This led to a situation where the ObjectMonitor is still on the in-use list and thinks it is owned by the object, but the object does not agree. This rare bug existed in the "CR1/v2.01/4-for-jdk13" version of the code. save_om_ptr() and deflate_monitor_using_JT() have been changed to recognize a large negative ref_count value as a marker that async deflation has won the race. With that change in place, it is no longer possible for both T-deflate and T-hash to lose the same race.
    T-hash                      ObjectMonitor              T-deflate
    -------------------------   +-----------------------+  -----------------------------------------------
    save_om_ptr() {             | owner=DEFLATER_MARKER |  deflate_monitor_using_JT() {
   1> atomic inc ref_count      | contentions=-max_jint |    cmpxchg(DEFLATER_MARKER, &owner, NULL)
      if (owner ==              | ref_count=0           |    if (waiters != 0 || ref_count != 0) {
          DEFLATER_MARKER &&    +-----------------------+    }
          contentions <= 0 &&                                prev = cmpxchg(-max_jint, &contentions, 0)
          ref_count <= 0) {                                  if (prev == 0 &&
        restore obj header                                       owner == DEFLATER_MARKER &&
        atomic dec ref_count                                1>   cmpxchg(-max_jint, &ref_count, 0) == 0) {
        return false to                                      } else {
          cause a retry                                        cmpxchg(NULL, &owner, DEFLATER_MARKER)
      }                                                        atomic add max_jint to contentions
                                                               bailout on deflation
                                                             }
ObjectMonitorHandle   | header=dmw_no_hash    |    } else {
        return true             | owner=NULL            |      cmpxchg(NULL, &owner, DEFLATER_MARKER)
      }                         | ref_count=1           |   2> bailout on deflation
    }                           +-----------------------+    }
    if save_om_ptr() {                     ||
      if no hash                           \/
        gen hash & merge        +-----------------------+
      hash = hash(header)       | header=dmw_hash       |
    }                           | owner=NULL            |
 3> atomic dec ref_count        | ref_count=1           |
    return hash                 +-----------------------+- T-deflate has set the owner field to DEFLATER_MARKER.
- T-hash has incremented ref_count before T-deflate made it to cmpxchg().
- The first ObjectMonitor box is showing the fields at this point and the "1>" markers are showing where each thread is at for that ObjectMonitor box.
- T-deflate bails out on deflation, but first it tries to restore the owner field:
 - The return value of cmpxchg() is not checked here.
- If T-deflate cannot restore the owner field to NULL, then another thread has managed to enter the monitor (or enter and exit the monitor) and we don't want to overwrite that information.
 
- T-hash observes:- "owner == DEFLATER_MARKER && ref_count > 0" or
- "owner == NULL && ref_count > 0" so it gets ready to save the ObjectMonitor*.
 
- The second ObjectMonitor box is showing the fields at this point and the "2>" markers are showing where each thread is at for that ObjectMonitor box.
- T-hash saves the ObjectMonitor* in the ObjectMonitorHandle (not shown) and returns to the caller.
- save_om_ptr() returns true since the ObjectMonitor is safe:- if ObjectMonitor's 'header/dmw' field does not have a hash, then generate a hash and merge it with the 'header/dmw' field.
- Otherwise, extract the hash from the ObjectMonitor's 'header/dmw' field.
 
- The third 
- T-deflate made it all the way to the cmpxchg() that makes ref_count a large negative value.
- T-hash is ready to increment ref_count.
- The first ObjectMonitor box is showing the fields at this point and the "1>3>" markers are marker is showing where each thread T-hash is at for that ObjectMonitor box.
- If T-deflate makes hash decrements the ref_count a large negative value, then this is the "T-deflate Wins" scenario abovefield.If 
- T-hash makes "ref_count == 1", then this is the "T-hash Wins Scenario 2" abovereturns the hash value.
 
Please note that in Carsten's original prototype, there was another race in ObjectSynchronizer::FastHashCode() when the object's monitor had to be inflated. The setting of the hashcode in the ObjectMonitor's header/dmw could race with T-deflate. That race is resolved in this version by the use of an ObjectMonitorHandle in the call to ObjectSynchronizer::inflate(). The ObjectMonitor* returned by ObjectMonitorHandle.om_ptr() has a non-zero ref_count so no additional races with T-deflate are possible.
...
- Counterpart function mapping for those that know the existing code:- ObjectSynchronizer class:- deflate_idle_monitors() has deflate_global_idle_monitors_using_JT(), deflate_per_thread_idle_monitors_using_JT(), and deflate_common_idle_monitors_using_JT().
- deflate_monitor_list() has deflate_monitor_list_using_JT()
- deflate_monitor() has deflate_monitor_using_JT()
 
- ObjectMonitor class:- is_busy() has is_busy_async()
- clear() has clear_using_JT()
 
 
- These functions recognize the Async Monitor Deflation protocol and adapt their operations:ObjectMonitor::enter()
 - ObjectMonitor::EnterI()
- ObjectMonitor::ReenterI()
- most callers to enter() had to indirectly adapt to the protocol and retry their operations.
 Also these functions had to adapt and retry their operations:
 - ObjectSynchronizer::quick_enter()
- ObjectSynchronizer::slow_enter::EnterI()
- ObjectSynchronizerObjectMonitor::reenterReenterI()
 
- Also these functions had to adapt and retry their operations:
 - ObjectSynchronizer::jniquick_enter()
- ObjectSynchronizer::FastHashCode()
- ObjectSynchronizer::current_thread_holds_lock()
- ObjectSynchronizer::query_lock_ownership()
- ObjectSynchronizer::get_lock_owner()
- ObjectSynchronizer::monitors_iterate()
- ObjectSynchronizer::inflate_helper()
- ObjectSynchronizer::inflate() 
 
- Various assertions had to be modified to pass without their real check when AsyncDeflateIdleMonitors is true; this is due to the change in semantics for the ObjectMonitor owner and contentions fieldsfield.
- ObjectMonitor has a new allocation_state field that supports three states: 'Free', 'New', 'Old'. Async Monitor Deflation is only applied to ObjectMonitors that have reached the 'Old' state.- Note: Prior to CR1/v2.01/4-for-jdk13, the allocation state was transitioned from 'New' to 'Old' in deflate_monitor_via_JT(). This meant that deflate_monitor_via_JT() had to see an ObjectMonitor twice before deflating it. This policy was intended to prevent oscillation from 'New'
 , . Async Monitor Deflation is only applied to ObjectMonitors that have reached the 'Old' state. When the Async Monitor Deflation code sees an ObjectMonitor in the 'New' state, it is changed to the 'Old' state, but is not deflated. This prevents a newly allocated ObjectMonitor from being immediately deflated which could cause an inflation<->deflation oscillation- and back again.
- In CR1/v2.01/4-for-jdk13, the allocation state is transitioned from 'New' -> "Old" in inflate(). This makes ObjectMonitors available for deflation earlier. So far there has been no signs of oscillation from 'New' → 'Old' and back again.
 
- ObjectMonitor has a new ref_count field that is used as part of the async deflation protocol and to indicate that an ObjectMonitor* is in use so the ObjectMonitor should not be deflated; this is needed for operations on non-busy monitors so that ObjectMonitor values don't change while they are being queried. There is a new ObjectMonitorHandle helper to manage the ref_count.
- The ObjectMonitor::owner() accessor detects DEFLATER_MARKER and returns NULL in that case to minimize the places that need to understand the new DEFLATER_MARKER value.
- System.gc()/JVM_GC() causes a special monitor list cleanup request which uses the safepoint based monitor list mechanism. So even if AsyncDeflateIdleMonitors is enabled, the safepoint based mechanism is still used by this special case.- This is necessary for those tests that do something to cause an object's monitor to be inflated, clear the only reference to the object and then expect that enough System.gc() calls will eventually cause the object to be GC'ed even when the thread never inflates another object's monitor. Yes, we have several tests like that. :-)