Book a Demo!
CoCalc Logo Icon
StoreFeaturesDocsShareSupportNewsAboutPoliciesSign UpSign In
PojavLauncherTeam
GitHub Repository: PojavLauncherTeam/openjdk-multiarch-jdk8u
Path: blob/aarch64-shenandoah-jdk8u272-b10/jdk/src/macosx/classes/apple/launcher/JavaAppLauncher.java
38829 views
1
/*
2
* Copyright (c) 2011, 2012, Oracle and/or its affiliates. All rights reserved.
3
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
4
*
5
* This code is free software; you can redistribute it and/or modify it
6
* under the terms of the GNU General Public License version 2 only, as
7
* published by the Free Software Foundation. Oracle designates this
8
* particular file as subject to the "Classpath" exception as provided
9
* by Oracle in the LICENSE file that accompanied this code.
10
*
11
* This code is distributed in the hope that it will be useful, but WITHOUT
12
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
13
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
14
* version 2 for more details (a copy is included in the LICENSE file that
15
* accompanied this code).
16
*
17
* You should have received a copy of the GNU General Public License version
18
* 2 along with this work; if not, write to the Free Software Foundation,
19
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
20
*
21
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
22
* or visit www.oracle.com if you need additional information or have any
23
* questions.
24
*/
25
26
package apple.launcher;
27
28
import java.io.*;
29
import java.lang.reflect.*;
30
import java.text.MessageFormat;
31
import java.util.*;
32
import java.util.jar.*;
33
34
import javax.swing.*;
35
36
class JavaAppLauncher implements Runnable {
37
static {
38
java.security.AccessController.doPrivileged(
39
new java.security.PrivilegedAction<Void>() {
40
public Void run() {
41
System.loadLibrary("osx");
42
return null;
43
}
44
});
45
}
46
47
private static native <T> T nativeConvertAndRelease(final long ptr);
48
private static native void nativeInvokeNonPublic(Class<? extends Method> cls, Method m, String[] args);
49
50
// entry point from native
51
static void launch(final long javaDictionaryPtr, final boolean verbose) {
52
final Map<String, ?> javaDictionary = nativeConvertAndRelease(javaDictionaryPtr);
53
(new JavaAppLauncher(javaDictionary, verbose)).run();
54
}
55
56
// these are the values for the enumeration JavaFailureMode
57
static final String kJavaFailureMainClassNotSpecified = "MainClassNotSpecified";
58
static final String kJavaFailureMainClassNotFound = "CannotLoadMainClass";
59
static final String kJavaFailureMainClassHasNoMain = "NoMainMethod";
60
static final String kJavaFailureMainClassMainNotStatic = "MainNotStatic";
61
static final String kJavaFailureMainThrewException = "MainThrewException";
62
static final String kJavaFailureMainInitializerException = "MainInitializerException";
63
64
final boolean verbose; // Normally set by environment variable JAVA_LAUNCHER_VERBOSE.
65
final Map<String, ?> javaDictionary;
66
67
JavaAppLauncher(final Map<String, ?> javaDictionary, final boolean verbose) {
68
this.verbose = verbose;
69
this.javaDictionary = javaDictionary;
70
}
71
72
@Override
73
public void run() {
74
final Method m = loadMainMethod(getMainMethod());
75
final String methodName = m.getDeclaringClass().getName() + ".main(String[])";
76
try {
77
log("Calling " + methodName + " method");
78
m.invoke(null, new Object[] { getArguments() });
79
log(methodName + " has returned");
80
} catch (final IllegalAccessException x) {
81
try {
82
nativeInvokeNonPublic(m.getClass(), m, getArguments());
83
} catch (final Throwable excpt) {
84
logError(methodName + " threw an exception:");
85
if ((excpt instanceof UnsatisfiedLinkError) && excpt.getMessage().equals("nativeInvokeNonPublic")) {
86
showFailureAlertAndKill(kJavaFailureMainThrewException, "nativeInvokeNonPublic not registered");
87
} else {
88
excpt.printStackTrace();
89
showFailureAlertAndKill(kJavaFailureMainThrewException, excpt.toString());
90
}
91
}
92
} catch (final InvocationTargetException invokeExcpt) {
93
logError(methodName + " threw an exception:");
94
invokeExcpt.getTargetException().printStackTrace();
95
showFailureAlertAndKill(kJavaFailureMainThrewException, invokeExcpt.getTargetException().toString());
96
}
97
}
98
99
Method loadMainMethod(final String mainClassName) {
100
try {
101
final Class<?> mainClass = Class.forName(mainClassName, true, sun.misc.Launcher.getLauncher().getClassLoader());
102
final Method mainMethod = mainClass.getDeclaredMethod("main", new Class[] { String[].class });
103
if ((mainMethod.getModifiers() & Modifier.STATIC) == 0) {
104
logError("The main(String[]) method of class " + mainClassName + " is not static!");
105
showFailureAlertAndKill(kJavaFailureMainClassMainNotStatic, mainClassName);
106
}
107
return mainMethod;
108
} catch (final ExceptionInInitializerError x) {
109
logError("The main class \"" + mainClassName + "\" had a static initializer throw an exception.");
110
x.getException().printStackTrace();
111
showFailureAlertAndKill(kJavaFailureMainInitializerException, x.getException().toString());
112
} catch (final ClassNotFoundException x) {
113
logError("The main class \"" + mainClassName + "\" could not be found.");
114
showFailureAlertAndKill(kJavaFailureMainClassNotFound, mainClassName);
115
} catch (final NoSuchMethodException x) {
116
logError("The main class \"" + mainClassName + "\" has no static main(String[]) method.");
117
showFailureAlertAndKill(kJavaFailureMainClassHasNoMain, mainClassName);
118
} catch (final NullPointerException x) {
119
logError("No main class specified");
120
showFailureAlertAndKill(kJavaFailureMainClassNotSpecified, null);
121
}
122
123
return null;
124
}
125
126
// get main class name from 'Jar' key, or 'MainClass' key
127
String getMainMethod() {
128
final Object javaJar = javaDictionary.get("Jar");
129
if (javaJar != null) {
130
if (!(javaJar instanceof String)) {
131
logError("'Jar' key in 'Java' sub-dictionary of Info.plist requires a string value");
132
return null;
133
}
134
135
final String jarPath = (String)javaJar;
136
if (jarPath.length() == 0) {
137
log("'Jar' key of sub-dictionary 'Java' of Info.plist key is empty");
138
} else {
139
// extract main class from manifest of this jar
140
final String main = getMainFromManifest(jarPath);
141
if (main == null) {
142
logError("jar file '" + jarPath + "' does not have Main-Class: attribute in its manifest");
143
return null;
144
}
145
146
log("Main class " + main + " found in jar manifest");
147
return main;
148
}
149
}
150
151
final Object javaMain = javaDictionary.get("MainClass");
152
if (!(javaMain instanceof String)) {
153
logError("'MainClass' key in 'Java' sub-dictionary of Info.plist requires a string value");
154
return null;
155
}
156
157
final String main = (String)javaMain;
158
if (main.length() == 0) {
159
log("'MainClass' key of sub-dictionary 'Java' of Info.plist key is empty");
160
return null;
161
}
162
163
log("Main class " + (String)javaMain + " found via 'MainClass' key of sub-dictionary 'Java' of Info.plist key");
164
return (String)javaMain;
165
}
166
167
// get arguments for main(String[]) out of Info.plist and command line
168
String[] getArguments() {
169
// check for 'Arguments' key, which contains the main() args if not defined in Info.plist
170
final Object javaArguments = javaDictionary.get("Arguments");
171
if (javaArguments == null) {
172
// no arguments
173
log("No arguments for main(String[]) specified");
174
return new String[0];
175
}
176
177
if (javaArguments instanceof List) {
178
final List<?> args = (List<?>)javaArguments;
179
final int count = args.size();
180
log("Arguments to main(String[" + count + "]):");
181
182
final String[] result = new String[count];
183
for (int i = 0; i < count; ++i) {
184
final Object element = args.get(i);
185
if (element instanceof String) {
186
result[i] = (String)element;
187
} else {
188
logError("Found non-string in array");
189
}
190
log(" arg[" + i + "]=" + result[i]);
191
}
192
return result;
193
}
194
195
logError("'Arguments' key in 'Java' sub-dictionary of Info.plist requires a string value or an array of strings");
196
return new String[0];
197
}
198
199
// returns name of main class, or null
200
String getMainFromManifest(final String jarpath) {
201
JarFile jar = null;
202
try {
203
jar = new JarFile(jarpath);
204
final Manifest man = jar.getManifest();
205
final Attributes attr = man.getMainAttributes();
206
return attr.getValue("Main-Class");
207
} catch (final IOException x) {
208
// shrug
209
} finally {
210
if (jar != null) {
211
try {
212
jar.close();
213
} catch (final IOException x) { }
214
}
215
}
216
return null;
217
}
218
219
void log(final String s) {
220
if (!verbose) return;
221
System.out.println("[LaunchRunner] " + s);
222
}
223
224
static void logError(final String s) {
225
System.err.println("[LaunchRunner Error] " + s);
226
}
227
228
// This kills the app and does not return!
229
static void showFailureAlertAndKill(final String msg, String arg) {
230
if (arg == null) arg = "<<null>>";
231
JOptionPane.showMessageDialog(null, getMessage(msg, arg), "", JOptionPane.ERROR_MESSAGE);
232
System.exit(-1);
233
}
234
235
static String getMessage(final String msgKey, final Object ... args) {
236
final String msg = ResourceBundle.getBundle("appLauncherErrors").getString(msgKey);
237
return MessageFormat.format(msg, args);
238
}
239
}
240
241