27
27
28
28
import java .io .InputStream ;
29
29
import java .io .PrintStream ;
30
+ import java .lang .reflect .Field ;
30
31
32
+ import com .oracle .svm .core .util .VMError ;
33
+ import com .oracle .svm .util .ReflectionUtil ;
34
+ import jdk .internal .access .SharedSecrets ;
31
35
import org .graalvm .nativeimage .ImageSingletons ;
32
36
import org .graalvm .nativeimage .hosted .Feature ;
33
37
@@ -51,19 +55,36 @@ public class SystemInOutErrFeature implements InternalFeature, FeatureSingleton
51
55
private final InputStream hostedIn ;
52
56
private final PrintStream hostedOut ;
53
57
private final PrintStream hostedErr ;
58
+ private final InputStream hostedInitialIn ;
59
+ private final PrintStream hostedInitialErr ;
54
60
55
61
public SystemInOutErrFeature () {
56
62
hostedIn = System .in ;
57
63
NativeImageSystemIOWrappers wrappers = NativeImageSystemIOWrappers .singleton ();
58
64
hostedOut = wrappers .outWrapper ;
59
65
hostedErr = wrappers .errWrapper ;
66
+ hostedInitialIn = SharedSecrets .getJavaLangAccess ().initialSystemIn ();
67
+ /*
68
+ * GR-55515: Migrate to JavaLangAccess#initialSystemErr(). The method
69
+ * JavaLangAccess#initialSystemErr() and the System#initialErr field were both introduced in
70
+ * JDK 23. Once JDK 21 compatibility is no longer required, consider switching to
71
+ * SharedSecrets.getJavaLangAccess().initialSystemErr().
72
+ */
73
+ Field initialErrField = ReflectionUtil .lookupField (true , System .class , "initialErr" );
74
+ try {
75
+ hostedInitialErr = initialErrField != null ? (PrintStream ) initialErrField .get (null ) : null ;
76
+ } catch (IllegalAccessException illegalAccess ) {
77
+ throw VMError .shouldNotReachHere (illegalAccess );
78
+ }
60
79
}
61
80
62
81
private SystemInOutErrSupport runtime ;
63
82
64
83
private static final String SYSTEM_IN_KEY_NAME = "System#in" ;
65
84
private static final String SYSTEM_ERR_KEY_NAME = "System#err" ;
66
85
private static final String SYSTEM_OUT_KEY_NAME = "System#out" ;
86
+ private static final String SYSTEM_INITIAL_IN_KEY_NAME = "System#initialIn" ;
87
+ private static final String SYSTEM_INITIAL_ERR_KEY_NAME = "System#initialErr" ;
67
88
68
89
@ Override
69
90
public void afterRegistration (AfterRegistrationAccess access ) {
@@ -82,6 +103,8 @@ public void duringSetup(DuringSetupAccess access) {
82
103
registry .registerHeapConstant (SYSTEM_IN_KEY_NAME , runtime .in ());
83
104
registry .registerHeapConstant (SYSTEM_OUT_KEY_NAME , runtime .out ());
84
105
registry .registerHeapConstant (SYSTEM_ERR_KEY_NAME , runtime .err ());
106
+ registry .registerHeapConstant (SYSTEM_INITIAL_IN_KEY_NAME , runtime .initialIn ());
107
+ registry .registerHeapConstant (SYSTEM_INITIAL_ERR_KEY_NAME , runtime .initialErr ());
85
108
}
86
109
access .registerObjectReplacer (this ::replaceStreamsWithRuntimeObject );
87
110
} else {
@@ -102,6 +125,10 @@ Object replaceStreamsWithRuntimeObject(Object object) {
102
125
return runtime .out ();
103
126
} else if (object == hostedErr ) {
104
127
return runtime .err ();
128
+ } else if (object == hostedInitialErr ) {
129
+ return runtime .initialErr ();
130
+ } else if (object == hostedInitialIn ) {
131
+ return runtime .initialIn ();
105
132
} else {
106
133
return object ;
107
134
}
@@ -114,6 +141,10 @@ ImageHeapConstant replaceStreamsWithLayerConstant(CrossLayerConstantRegistry reg
114
141
return registry .getConstant (SYSTEM_OUT_KEY_NAME );
115
142
} else if (object == hostedErr ) {
116
143
return registry .getConstant (SYSTEM_ERR_KEY_NAME );
144
+ } else if (object == hostedInitialErr ) {
145
+ return registry .getConstant (SYSTEM_INITIAL_ERR_KEY_NAME );
146
+ } else if (object == hostedInitialIn ) {
147
+ return registry .getConstant (SYSTEM_INITIAL_IN_KEY_NAME );
117
148
} else {
118
149
return null ;
119
150
}
0 commit comments