Unsafe: removing the monitorEnter/monitorExit/tryMonitorEnter methods

David Holmes david.holmes at oracle.com
Thu Feb 27 09:58:24 UTC 2014


On 27/02/2014 6:38 PM, Peter Levart wrote:
> On 02/27/2014 08:41 AM, David Holmes wrote:
>> On 27/02/2014 5:30 PM, Peter Levart wrote:
>>> On 02/27/2014 08:29 AM, Peter Levart wrote:
>>>> On 02/26/2014 09:54 PM, Martin Buchholz wrote:
>>>>> I don't recall having added this particular wart
>>>>> to test/java/lang/ProcessBuilder/Basic.java, and history suggests that
>>>>> others did that.
>>>>>
>>>>> It does seem that being able to tell whether a java object monitor is
>>>>> currently locked is useful for debugging and monitoring - there
>>>>> should be a
>>>>> way to do that.
>>>>
>>>> Thread.holdsLock(Object) ?
>>>
>>> Ah, you meant to query from some other thread, right?
>>
>> Right - that is the usage in Basic.java but I'm still scratching my
>> head trying to understand what finding the BufferedInputStream locked
>> is supposed to signify.
>
> Here's the relevant code block:
>
>                      while (unsafe.tryMonitorEnter(s)) {
>                          unsafe.monitorExit(s);
>                          Thread.sleep(1);
>                      }
>
> This code block waits until it finds that some thread has acquired the
> lock on 's' for at least 1 ms (or else it can miss it and will wait
> further) ...
>
> Since this block is executed for s instanceof BufferedInputStream, what
> this block does, is wait until one of synchronized methods on
> BufferedInputStream is called. The thread that does that, calls read()
> or read(byte[]) and blocks while holding a lock, since the child process
> that is spawned and whose stdout the thread is trying to read in this
> situation does a 10 minute sleep.

Ah I see - thanks for the analysis. So this is a classic 'barrier' 
waiting for the other thread to have reached the synchronized method.

> I think the following could be used instead for the same purpose:
>
>              while (!isLocked(s, 100L)) {
>                  Thread.sleep(1);
>              }
>
> // using the following utility method:
>
>      static boolean isLocked(final Object monitor, long millis) throws
> InterruptedException {
>          Thread t = new Thread() {
>              @Override
>              public void run() {
>                  synchronized (monitor) { }
>              }
>          };
>          t.start();
>          t.join(millis);
>          return t.isAlive();
>      }

Functionally yes this will only return true if the object is locked for 
 >100ms. But it is potentially going to generate a lot of temporary 
threads. And in theory at least the JIT can optimize the run method into 
nothing.

Still it is a cute trick :)

David

>
> Regards, Peter
>
>>
>> David
>> -----
>>
>>> Peter
>>>
>>>>
>>>> Regards, Peter
>>>>
>>>>>
>>>>>
>>>>> On Wed, Feb 26, 2014 at 7:12 AM, Paul Sandoz <paul.sandoz at oracle.com>
>>>>> wrote:
>>>>>
>>>>>> Hi,
>>>>>>
>>>>>> Out of all the methods on Unsafe i think the
>>>>>> monitorEnter/monitorExit/tryMonitorEnter are the least used and are
>>>>>> very
>>>>>> strong candidates for removal.
>>>>>>
>>>>>> 99% of use-cases are supported by classes in the
>>>>>> java.util.concurrent.locks package.
>>>>>>
>>>>>>
>>>>>> Within the JDK itself it is only used in one JDK test file
>>>>>> test/java/lang/ProcessBuilder/Basic.java:
>>>>>>
>>>>>>                      while (unsafe.tryMonitorEnter(s)) {
>>>>>>                          unsafe.monitorExit(s);
>>>>>>                          Thread.sleep(1);
>>>>>>                      }
>>>>>>
>>>>>> for a test verifying an EOF is received on pending reads and it is
>>>>>> polling
>>>>>> to check when the process builder acquires the lock before
>>>>>> destroying the
>>>>>> process, presumably to avoid some sort of race condition that
>>>>>> occasionally
>>>>>> causes the test to fail.
>>>>>>
>>>>>> I believe this test as been through a number of rounds, i stared at
>>>>>> things
>>>>>> for a bit, but cannot quickly work out a replacement; i lack the
>>>>>> knowledge
>>>>>> on this area.
>>>>>>
>>>>>>
>>>>>> Outside of the JDK i can only find one usage of monitorExit/Enter
>>>>>> (according to grep code) in JBoss modules, and i believe this is
>>>>>> only used
>>>>>> on Java 1.6 to work around some limitations in class loading that
>>>>>> were
>>>>>> fixed in 1.7.
>>>>>>
>>>>>>
>>>>>> Given such very limited use i propose to remove these methods after
>>>>>> having
>>>>>> worked out a fix for ProcessBuilder/Basic.java test.
>>>>>>
>>>>>> Paul.
>>>>>>
>>>>
>>>
>



More information about the core-libs-dev mailing list