Removing intrinsic of Thread.isInterrupted()

Karen Kinnear karen.kinnear at oracle.com
Tue Feb 25 11:56:32 PST 2014


Vladimir,

I updated the bug to reflect the code review email thread between Yumin, myself
and David Holmes.

To the best of my understanding there is a potential timing hole here, with the clearInterrupted
case (study the webrev for the fix), i.e. in the isInterrupted(false) case.

What I proposed was that we could keep the intrinsic quick check for isInterrupted bit, but not
the logic for the isInterrupted(false) - unless you want to change it to add the Windows
WaitForSingleObject call - which I assume removes the benefit of the intrinsic.

Does that make sense to you?
thanks,
Karen

On Feb 25, 2014, at 2:14 PM, Vladimir Kozlov wrote:

> Yumin,
> 
> On 2/24/14 5:46 PM, Yumin Qi wrote:
>> Hi, Compiler team
>> 
>>   I worked on this bug: 6498581:ThreadInterruptTest3 produces wrong
>> output on Windows.  This is a problem thread sleep wakes up spuriously
>> due to a race condition between os::interrupt and os::is_interruped.
>> Detail please see bug comments.
>> 
>>  https://bugs.openjdk.java.net/browse/JDK-6498581
>> 
>>   The fix is (without removing intrinsic, but intend to remove it) :
>>   http://cr.openjdk.java.net/~minqi/6498581/webrev00/
>> 
>>   One problem is that Thread.isInterrupted() is intrinsic and there is
>> chance that code like
>> 
>>   boolean a = Thread.currentThread().isInterrupted();
>>   boolean b = Thread.interrupted();
>> 
>>   Will get different value. (fast/slow path)
> 
> How you come to this conclusion? You my be mistaken. We intrinsify native boolean isInterrupted(boolean ClearInterrupted) and not java method: isInterrupted().
> 
> Also you are comparing different code which is nothing to do with intrinsic:
> 
> isInterrupted() passes ClearInterrupted=false:
> 
>    public boolean isInterrupted() {
>        return isInterrupted(false);
>    }
> 
> when interrupted() passes 'true':
> 
>    public static boolean interrupted() {
>        return currentThread().isInterrupted(true);
>    }
> 
> Both method calls native isInterrupted(bool) which is intrinsified. So both calls intrinsic. There should be no difference.
> 
> From performance point of view, as Aleksey pointed, there is huge difference. We can't remove intrinsic.
> 
> Thanks,
> Vladimir
> 
>>   I tried to remove the intrinsic code and done a test using following
>> code. The result showed there is no difference by removing the intrinsic
>> of Thread.isInterrupted().
>> 
>> // test performance of removing Thread.isInterrupted() inlining
>> public class TestThreadInterrupted {
>>     public static void main(String... args) {
>>         Thread t = new Thread () {
>>             public void run() {
>>                 boolean isInt = false;
>>                 while (!isInt) {
>>                     try {
>>                         Thread.sleep(30);
>>                     } catch (InterruptedException ie) {
>>                         isInt = true;
>>                     }
>>                 }
>>             }
>>         };
>> 
>>         t.start();
>>         // run
>>         long start, finish, isum = 0L, osum = 0L;
>>         int  NUM = 20000;
>>         for (int j = 0; j < 100; j++) {
>>         isum = 0L;
>>         for (int i = 0; i < NUM; i++) {
>>             start = System.currentTimeMillis();
>>             t.isInterrupted();
>>             finish = System.currentTimeMillis();
>>             isum += (finish - start);
>>         }
>> 
>>         System.out.println("Total cost of " + NUM + " calls is " + isum
>> + " ms");
>>         osum += isum;
>>         }
>>         System.out.println("Average " + osum/100 + " ms");
>>         t.interrupt();
>>         try {
>>             t.join();
>>         } catch (InterruptedException e) {}
>>     }
>> }
>> 
>> And found there is no difference on Solaris-x64/sparcv9, Windows(32/64),
>> linux(32/64) before and after the removing of intrinsic
>> Thread.isInterrupted().
>> 
>> Should I remove the intrinsic?
>> 
>> Data (no major difference  for both with/without intrinsic):
>> 
>> 1)windows :
>> ....
>> Total cost of 20000 calls is 2 ms
>> Total cost of 20000 calls is 1 ms
>> Total cost of 20000 calls is 1 ms
>> Total cost of 20000 calls is 1 ms
>> Total cost of 20000 calls is 2 ms
>> Total cost of 20000 calls is 1 ms
>> Total cost of 20000 calls is 2 ms
>> Total cost of 20000 calls is 2 ms
>> Total cost of 20000 calls is 2 ms
>> Total cost of 20000 calls is 0 ms
>> Total cost of 20000 calls is 2 ms
>> Total cost of 20000 calls is 2 ms
>> Average 1 ms
>> 
>> 2) Solaris-x64
>> ....
>> Total cost of 20000 calls is 3 ms
>> Total cost of 20000 calls is 1 ms
>> Total cost of 20000 calls is 4 ms
>> Total cost of 20000 calls is 6 ms
>> Total cost of 20000 calls is 6 ms
>> Total cost of 20000 calls is 5 ms
>> Total cost of 20000 calls is 7 ms
>> Total cost of 20000 calls is 5 ms
>> Total cost of 20000 calls is 5 ms
>> Total cost of 20000 calls is 1 ms
>> Total cost of 20000 calls is 3 ms
>> Total cost of 20000 calls is 2 ms
>> Total cost of 20000 calls is 3 ms
>> Total cost of 20000 calls is 3 ms
>> Total cost of 20000 calls is 5 ms
>> Total cost of 20000 calls is 4 ms
>> Total cost of 20000 calls is 4 ms
>> Total cost of 20000 calls is 7 ms
>> Average 4 ms
>> 
>> 3) Linux:
>> 
>> ....
>> Total cost of 20000 calls is 30 ms
>> Total cost of 20000 calls is 29 ms
>> Total cost of 20000 calls is 26 ms
>> Total cost of 20000 calls is 26 ms
>> Total cost of 20000 calls is 26 ms
>> Total cost of 20000 calls is 24 ms
>> Total cost of 20000 calls is 29 ms
>> Total cost of 20000 calls is 25 ms
>> Total cost of 20000 calls is 20 ms
>> Average 24 ms
>> 
>> 
>> Thanks
>> Yumin



More information about the hotspot-compiler-dev mailing list