33
33
import com .oracle .svm .core .annotate .Alias ;
34
34
import com .oracle .svm .core .annotate .InjectAccessors ;
35
35
import com .oracle .svm .core .annotate .TargetClass ;
36
+ import com .oracle .svm .core .annotate .TargetElement ;
36
37
import com .oracle .svm .core .feature .AutomaticallyRegisteredFeature ;
37
38
import com .oracle .svm .core .feature .InternalFeature ;
39
+ import com .oracle .svm .core .util .VMError ;
40
+
41
+ import jdk .graal .compiler .serviceprovider .JavaVersionUtil ;
38
42
39
43
@ TargetClass (java .util .concurrent .CompletableFuture .class )
40
44
final class Target_java_util_concurrent_CompletableFuture {
41
45
// Checkstyle: stop
42
46
@ Alias @ InjectAccessors (CompletableFutureUseCommonPoolAccessor .class ) //
47
+ @ TargetElement (onlyWith = JDK21OrEarlier .class ) //
43
48
private static boolean USE_COMMON_POOL ;
44
49
45
50
@ Alias @ InjectAccessors (CompletableFutureAsyncPoolAccessor .class ) //
51
+ @ TargetElement (onlyWith = JDK21OrEarlier .class ) //
46
52
private static Executor ASYNC_POOL ;
47
53
// Checkstyle: resume
48
54
}
49
55
56
+ @ TargetClass (className = "java.util.concurrent.DelayScheduler" , onlyWith = JDKLatest .class )
57
+ final class Target_java_util_concurrent_DelayScheduler {
58
+ @ Alias @ InjectAccessors (DelaySchedulerNanoTimeOffsetAccessor .class ) //
59
+ private static long nanoTimeOffset ;
60
+ }
61
+
62
+ class DelaySchedulerNanoTimeOffsetAccessor {
63
+ static long get () {
64
+ return DelaySchedulerNanoTimeOffsetHolder .NANO_TIME_OFFSET ;
65
+ }
66
+ }
67
+
68
+ /**
69
+ * Holder for {@link DelaySchedulerNanoTimeOffsetHolder#NANO_TIME_OFFSET}. Initialized at runtime
70
+ * via {@link CompletableFutureFeature}.
71
+ */
72
+ class DelaySchedulerNanoTimeOffsetHolder {
73
+
74
+ public static final long NANO_TIME_OFFSET ;
75
+
76
+ static {
77
+ if (SubstrateUtil .HOSTED ) {
78
+ throw VMError .shouldNotReachHere (DelaySchedulerNanoTimeOffsetHolder .class .getName () + " must only be initialized at runtime" );
79
+ }
80
+ NANO_TIME_OFFSET = Math .min (System .nanoTime (), 0L ) + Long .MIN_VALUE ;
81
+ }
82
+ }
83
+
50
84
class CompletableFutureUseCommonPoolAccessor {
51
85
static boolean get () {
52
86
return CompletableFutureFieldHolder .USE_COMMON_POOL ;
@@ -65,11 +99,22 @@ class CompletableFutureFieldHolder {
65
99
66
100
static final boolean USE_COMMON_POOL = ForkJoinPool .getCommonPoolParallelism () > 1 ;
67
101
68
- static final Executor ASYNC_POOL = USE_COMMON_POOL ? ForkJoinPool .commonPool ()
69
- : SubstrateUtil .cast (new Target_java_util_concurrent_CompletableFuture_ThreadPerTaskExecutor (), Executor .class );
102
+ static final Executor ASYNC_POOL ;
103
+
104
+ static {
105
+ if (JavaVersionUtil .JAVA_SPEC <= 21 ) {
106
+ if (USE_COMMON_POOL ) {
107
+ ASYNC_POOL = ForkJoinPool .commonPool ();
108
+ } else {
109
+ ASYNC_POOL = SubstrateUtil .cast (new Target_java_util_concurrent_CompletableFuture_ThreadPerTaskExecutor (), Executor .class );
110
+ }
111
+ } else {
112
+ ASYNC_POOL = null ;
113
+ }
114
+ }
70
115
}
71
116
72
- @ TargetClass (value = java .util .concurrent .CompletableFuture .class , innerClass = "ThreadPerTaskExecutor" )
117
+ @ TargetClass (value = java .util .concurrent .CompletableFuture .class , innerClass = "ThreadPerTaskExecutor" , onlyWith = JDK21OrEarlier . class )
73
118
final class Target_java_util_concurrent_CompletableFuture_ThreadPerTaskExecutor {
74
119
}
75
120
@@ -78,5 +123,6 @@ class CompletableFutureFeature implements InternalFeature {
78
123
@ Override
79
124
public void duringSetup (DuringSetupAccess access ) {
80
125
RuntimeClassInitialization .initializeAtRunTime (CompletableFutureFieldHolder .class );
126
+ RuntimeClassInitialization .initializeAtRunTime (DelaySchedulerNanoTimeOffsetHolder .class );
81
127
}
82
128
}
0 commit comments