41
41
import java .util .function .Predicate ;
42
42
import java .util .stream .Stream ;
43
43
import jdk .internal .event .ThreadSleepEvent ;
44
- import jdk .internal .javac .PreviewFeature ;
45
- import jdk .internal .misc .PreviewFeatures ;
46
44
import jdk .internal .misc .StructureViolationExceptions ;
47
45
import jdk .internal .misc .TerminatingThreadLocal ;
48
46
import jdk .internal .misc .Unsafe ;
@@ -464,24 +462,32 @@ public static void sleep(long millis) throws InterruptedException {
464
462
if (millis < 0 ) {
465
463
throw new IllegalArgumentException ("timeout value is negative" );
466
464
}
465
+ long nanos = MILLISECONDS .toNanos (millis );
467
466
468
- if (currentThread () instanceof VirtualThread vthread ) {
469
- long nanos = MILLISECONDS .toNanos (millis );
470
- vthread .sleepNanos (nanos );
471
- return ;
472
- }
473
-
467
+ // create jdk.ThreadSleepEvent event if enabled
468
+ ThreadSleepEvent event = null ;
474
469
if (ThreadSleepEvent .isTurnedOn ()) {
475
- ThreadSleepEvent event = new ThreadSleepEvent ();
476
470
try {
477
- event .time = MILLISECONDS .toNanos (millis );
471
+ event = new ThreadSleepEvent ();
472
+ event .time = nanos ;
478
473
event .begin ();
474
+ } catch (OutOfMemoryError e ) {
475
+ event = null ;
476
+ }
477
+ }
478
+
479
+ try {
480
+ if (currentThread () instanceof VirtualThread vthread ) {
481
+ vthread .sleepNanos (nanos );
482
+ } else {
479
483
sleep0 (millis );
480
- } finally {
481
- event .commit ();
482
484
}
483
- } else {
484
- sleep0 (millis );
485
+ } finally {
486
+ try {
487
+ if (event != null ) event .commit ();
488
+ } catch (OutOfMemoryError e ) {
489
+ // ignore
490
+ }
485
491
}
486
492
}
487
493
@@ -518,14 +524,6 @@ public static void sleep(long millis, int nanos) throws InterruptedException {
518
524
throw new IllegalArgumentException ("nanosecond timeout value out of range" );
519
525
}
520
526
521
- if (currentThread () instanceof VirtualThread vthread ) {
522
- // total sleep time, in nanoseconds
523
- long totalNanos = MILLISECONDS .toNanos (millis );
524
- totalNanos += Math .min (Long .MAX_VALUE - totalNanos , nanos );
525
- vthread .sleepNanos (totalNanos );
526
- return ;
527
- }
528
-
529
527
if (nanos > 0 && millis < Long .MAX_VALUE ) {
530
528
millis ++;
531
529
}
@@ -550,20 +548,40 @@ public static void sleep(long millis, int nanos) throws InterruptedException {
550
548
*/
551
549
public static void sleep (Duration duration ) throws InterruptedException {
552
550
long nanos = NANOSECONDS .convert (duration ); // MAX_VALUE if > 292 years
553
- if (nanos < 0 )
551
+ if (nanos < 0 ) {
554
552
return ;
553
+ }
555
554
556
- if (currentThread () instanceof VirtualThread vthread ) {
557
- vthread .sleepNanos (nanos );
558
- return ;
555
+ // create jdk.ThreadSleepEvent event if enabled
556
+ ThreadSleepEvent event = null ;
557
+ if (ThreadSleepEvent .isTurnedOn ()) {
558
+ try {
559
+ event = new ThreadSleepEvent ();
560
+ event .time = nanos ;
561
+ event .begin ();
562
+ } catch (OutOfMemoryError e ) {
563
+ event = null ;
564
+ }
559
565
}
560
566
561
- // convert to milliseconds
562
- long millis = MILLISECONDS .convert (nanos , NANOSECONDS );
563
- if (nanos > NANOSECONDS .convert (millis , MILLISECONDS )) {
564
- millis += 1L ;
567
+ try {
568
+ if (currentThread () instanceof VirtualThread vthread ) {
569
+ vthread .sleepNanos (nanos );
570
+ } else {
571
+ // convert to milliseconds
572
+ long millis = NANOSECONDS .toMillis (nanos );
573
+ if (nanos > MILLISECONDS .toNanos (millis )) {
574
+ millis += 1L ;
575
+ }
576
+ sleep0 (millis );
577
+ }
578
+ } finally {
579
+ try {
580
+ if (event != null ) event .commit ();
581
+ } catch (OutOfMemoryError e ) {
582
+ // ignore
583
+ }
565
584
}
566
- sleep (millis );
567
585
}
568
586
569
587
/**
@@ -611,7 +629,7 @@ public static void onSpinWait() {}
611
629
* the constructing thread.
612
630
* See Thread initialization.
613
631
*/
614
- static final int NO_INHERIT_THREAD_LOCALS = 1 << 1 ;
632
+ static final int NO_INHERIT_THREAD_LOCALS = 1 << 2 ;
615
633
616
634
/**
617
635
* Helper class to generate thread identifiers. The identifiers start at
@@ -800,9 +818,8 @@ private static ClassLoader contextClassLoader(Thread parent) {
800
818
* }
801
819
*
802
820
* @return A builder for creating {@code Thread} or {@code ThreadFactory} objects.
803
- * @since 19
821
+ * @since 21
804
822
*/
805
- @ PreviewFeature (feature = PreviewFeature .Feature .VIRTUAL_THREADS )
806
823
public static Builder .OfPlatform ofPlatform () {
807
824
return new ThreadBuilders .PlatformThreadBuilder ();
808
825
}
@@ -821,12 +838,9 @@ public static Builder.OfPlatform ofPlatform() {
821
838
* }
822
839
*
823
840
* @return A builder for creating {@code Thread} or {@code ThreadFactory} objects.
824
- * @throws UnsupportedOperationException if preview features are not enabled
825
- * @since 19
841
+ * @since 21
826
842
*/
827
- @ PreviewFeature (feature = PreviewFeature .Feature .VIRTUAL_THREADS )
828
843
public static Builder .OfVirtual ofVirtual () {
829
- PreviewFeatures .ensureEnabled ();
830
844
return new ThreadBuilders .VirtualThreadBuilder ();
831
845
}
832
846
@@ -856,14 +870,10 @@ public static Builder.OfVirtual ofVirtual() {
856
870
*
857
871
* @see Thread#ofPlatform()
858
872
* @see Thread#ofVirtual()
859
- * @since 19
873
+ * @since 21
860
874
*/
861
- @ PreviewFeature (feature = PreviewFeature .Feature .VIRTUAL_THREADS )
862
875
public sealed interface Builder
863
- permits Builder .OfPlatform ,
864
- Builder .OfVirtual ,
865
- ThreadBuilders .BaseThreadBuilder {
866
-
876
+ permits Builder .OfPlatform , Builder .OfVirtual {
867
877
868
878
/**
869
879
* Sets the thread name.
@@ -962,9 +972,8 @@ public sealed interface Builder
962
972
* this interface causes a {@code NullPointerException} to be thrown.
963
973
*
964
974
* @see Thread#ofPlatform()
965
- * @since 19
975
+ * @since 21
966
976
*/
967
- @ PreviewFeature (feature = PreviewFeature .Feature .VIRTUAL_THREADS )
968
977
sealed interface OfPlatform extends Builder
969
978
permits ThreadBuilders .PlatformThreadBuilder {
970
979
@@ -1037,9 +1046,8 @@ default OfPlatform daemon() {
1037
1046
* this interface causes a {@code NullPointerException} to be thrown.
1038
1047
*
1039
1048
* @see Thread#ofVirtual()
1040
- * @since 19
1049
+ * @since 21
1041
1050
*/
1042
- @ PreviewFeature (feature = PreviewFeature .Feature .VIRTUAL_THREADS )
1043
1051
sealed interface OfVirtual extends Builder
1044
1052
permits ThreadBuilders .VirtualThreadBuilder {
1045
1053
@@ -1467,14 +1475,11 @@ public Thread(ThreadGroup group, Runnable task, String name,
1467
1475
*
1468
1476
* @param task the object to run when the thread executes
1469
1477
* @return a new, and started, virtual thread
1470
- * @throws UnsupportedOperationException if preview features are not enabled
1471
1478
* @see <a href="#inheritance">Inheritance when creating threads</a>
1472
- * @since 19
1479
+ * @since 21
1473
1480
*/
1474
- @ PreviewFeature (feature = PreviewFeature .Feature .VIRTUAL_THREADS )
1475
1481
public static Thread startVirtualThread (Runnable task ) {
1476
1482
Objects .requireNonNull (task );
1477
- PreviewFeatures .ensureEnabled ();
1478
1483
var thread = ThreadBuilders .newVirtualThread (null , null , 0 , task );
1479
1484
thread .start ();
1480
1485
return thread ;
@@ -1486,9 +1491,8 @@ public static Thread startVirtualThread(Runnable task) {
1486
1491
*
1487
1492
* @return {@code true} if this thread is a virtual thread
1488
1493
*
1489
- * @since 19
1494
+ * @since 21
1490
1495
*/
1491
- @ PreviewFeature (feature = PreviewFeature .Feature .VIRTUAL_THREADS )
1492
1496
public final boolean isVirtual () {
1493
1497
return (this instanceof BaseVirtualThread );
1494
1498
}
@@ -1523,6 +1527,8 @@ void start(ThreadContainer container) {
1523
1527
throw new IllegalThreadStateException ();
1524
1528
1525
1529
// bind thread to container
1530
+ if (this .container != null )
1531
+ throw new IllegalThreadStateException ();
1526
1532
setThreadContainer (container );
1527
1533
1528
1534
// start thread
@@ -1597,15 +1603,17 @@ void clearReferences() {
1597
1603
* a chance to clean up before it actually exits.
1598
1604
*/
1599
1605
private void exit () {
1600
- // pop any remaining scopes from the stack, this may block
1601
- if (headStackableScopes != null ) {
1602
- StackableScope .popAll ();
1603
- }
1604
-
1605
- // notify container that thread is exiting
1606
- ThreadContainer container = threadContainer ();
1607
- if (container != null ) {
1608
- container .onExit (this );
1606
+ try {
1607
+ // pop any remaining scopes from the stack, this may block
1608
+ if (headStackableScopes != null ) {
1609
+ StackableScope .popAll ();
1610
+ }
1611
+ } finally {
1612
+ // notify container that thread is exiting
1613
+ ThreadContainer container = threadContainer ();
1614
+ if (container != null ) {
1615
+ container .onExit (this );
1616
+ }
1609
1617
}
1610
1618
1611
1619
try {
0 commit comments