1 /*
2 * Copyright (C) 2008 The Android Open Source Project
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16 /*
17 * Command-line invocation of the Dalvik VM.
18 */
19 #include "jni.h"
20
21 #include <stdlib.h>
22 #include <stdio.h>
23 #include <string.h>
24 #include <signal.h>
25 #include <assert.h>
26
27
28 /*
29 * We want failed write() calls to just return with an error.
30 */
blockSigpipe()31 static void blockSigpipe()
32 {
33 sigset_t mask;
34
35 sigemptyset(&mask);
36 sigaddset(&mask, SIGPIPE);
37 if (sigprocmask(SIG_BLOCK, &mask, NULL) != 0)
38 fprintf(stderr, "WARNING: SIGPIPE not blocked\n");
39 }
40
41 /*
42 * Create a String[] and populate it with the contents of argv.
43 */
createStringArray(JNIEnv * env,char * const argv[],int argc)44 static jobjectArray createStringArray(JNIEnv* env, char* const argv[], int argc)
45 {
46 jclass stringClass = NULL;
47 jobjectArray strArray = NULL;
48 jobjectArray result = NULL;
49 int i;
50
51 stringClass = env->FindClass("java/lang/String");
52 if (env->ExceptionCheck()) {
53 fprintf(stderr, "Got exception while finding class String\n");
54 goto bail;
55 }
56 assert(stringClass != NULL);
57 strArray = env->NewObjectArray(argc, stringClass, NULL);
58 if (env->ExceptionCheck()) {
59 fprintf(stderr, "Got exception while creating String array\n");
60 goto bail;
61 }
62 assert(strArray != NULL);
63
64 for (i = 0; i < argc; i++) {
65 jstring argStr;
66
67 argStr = env->NewStringUTF(argv[i]);
68 if (env->ExceptionCheck()) {
69 fprintf(stderr, "Got exception while allocating Strings\n");
70 goto bail;
71 }
72 assert(argStr != NULL);
73 env->SetObjectArrayElement(strArray, i, argStr);
74 env->DeleteLocalRef(argStr);
75 }
76
77 /* return the array, and ensure we don't delete the local ref to it */
78 result = strArray;
79 strArray = NULL;
80
81 bail:
82 env->DeleteLocalRef(stringClass);
83 env->DeleteLocalRef(strArray);
84 return result;
85 }
86
87 /*
88 * Determine whether or not the specified method is public.
89 *
90 * Returns JNI_TRUE on success, JNI_FALSE on failure.
91 */
methodIsPublic(JNIEnv * env,jclass clazz,jmethodID methodId)92 static int methodIsPublic(JNIEnv* env, jclass clazz, jmethodID methodId)
93 {
94 static const int PUBLIC = 0x0001; // java.lang.reflect.Modifiers.PUBLIC
95 jobject refMethod = NULL;
96 jclass methodClass = NULL;
97 jmethodID getModifiersId;
98 int modifiers;
99 int result = JNI_FALSE;
100
101 refMethod = env->ToReflectedMethod(clazz, methodId, JNI_FALSE);
102 if (refMethod == NULL) {
103 fprintf(stderr, "Dalvik VM unable to get reflected method\n");
104 goto bail;
105 }
106
107 /*
108 * We now have a Method instance. We need to call
109 * its getModifiers() method.
110 */
111 methodClass = env->FindClass("java/lang/reflect/Method");
112 if (methodClass == NULL) {
113 fprintf(stderr, "Dalvik VM unable to find class Method\n");
114 goto bail;
115 }
116 getModifiersId = env->GetMethodID(methodClass,
117 "getModifiers", "()I");
118 if (getModifiersId == NULL) {
119 fprintf(stderr, "Dalvik VM unable to find reflect.Method.getModifiers\n");
120 goto bail;
121 }
122
123 modifiers = env->CallIntMethod(refMethod, getModifiersId);
124 if ((modifiers & PUBLIC) == 0) {
125 fprintf(stderr, "Dalvik VM: main() is not public\n");
126 goto bail;
127 }
128
129 result = JNI_TRUE;
130
131 bail:
132 env->DeleteLocalRef(refMethod);
133 env->DeleteLocalRef(methodClass);
134 return result;
135 }
136
137 /*
138 * Parse arguments. Most of it just gets passed through to the VM. The
139 * JNI spec defines a handful of standard arguments.
140 */
main(int argc,char * const argv[])141 int main(int argc, char* const argv[])
142 {
143 JavaVM* vm = NULL;
144 JNIEnv* env = NULL;
145 JavaVMInitArgs initArgs;
146 JavaVMOption* options = NULL;
147 char* slashClass = NULL;
148 int optionCount, curOpt, i, argIdx;
149 int needExtra = JNI_FALSE;
150 int result = 1;
151
152 setvbuf(stdout, NULL, _IONBF, 0);
153
154 /* ignore argv[0] */
155 argv++;
156 argc--;
157
158 /*
159 * If we're adding any additional stuff, e.g. function hook specifiers,
160 * add them to the count here.
161 *
162 * We're over-allocating, because this includes the options to the VM
163 * plus the options to the program.
164 */
165 optionCount = argc;
166
167 options = (JavaVMOption*) malloc(sizeof(JavaVMOption) * optionCount);
168 memset(options, 0, sizeof(JavaVMOption) * optionCount);
169
170 /*
171 * Copy options over. Everything up to the name of the class starts
172 * with a '-' (the function hook stuff is strictly internal).
173 *
174 * [Do we need to catch & handle "-jar" here?]
175 */
176 for (curOpt = argIdx = 0; argIdx < argc; argIdx++) {
177 if (argv[argIdx][0] != '-' && !needExtra)
178 break;
179 options[curOpt++].optionString = strdup(argv[argIdx]);
180
181 /* some options require an additional arg */
182 needExtra = JNI_FALSE;
183 if (strcmp(argv[argIdx], "-classpath") == 0 ||
184 strcmp(argv[argIdx], "-cp") == 0)
185 /* others? */
186 {
187 needExtra = JNI_TRUE;
188 }
189 }
190
191 if (needExtra) {
192 fprintf(stderr, "Dalvik VM requires value after last option flag\n");
193 goto bail;
194 }
195
196 /* insert additional internal options here */
197
198 assert(curOpt <= optionCount);
199
200 initArgs.version = JNI_VERSION_1_4;
201 initArgs.options = options;
202 initArgs.nOptions = curOpt;
203 initArgs.ignoreUnrecognized = JNI_FALSE;
204
205 //printf("nOptions = %d\n", initArgs.nOptions);
206
207 blockSigpipe();
208
209 /*
210 * Start VM. The current thread becomes the main thread of the VM.
211 */
212 if (JNI_CreateJavaVM(&vm, &env, &initArgs) < 0) {
213 fprintf(stderr, "Dalvik VM init failed (check log file)\n");
214 goto bail;
215 }
216
217 /*
218 * Make sure they provided a class name. We do this after VM init
219 * so that things like "-Xrunjdwp:help" have the opportunity to emit
220 * a usage statement.
221 */
222 if (argIdx == argc) {
223 fprintf(stderr, "Dalvik VM requires a class name\n");
224 goto bail;
225 }
226
227 /*
228 * We want to call main() with a String array with our arguments in it.
229 * Create an array and populate it. Note argv[0] is not included.
230 */
231 jobjectArray strArray;
232 strArray = createStringArray(env, &argv[argIdx+1], argc-argIdx-1);
233 if (strArray == NULL)
234 goto bail;
235
236 /*
237 * Find [class].main(String[]).
238 */
239 jclass startClass;
240 jmethodID startMeth;
241 char* cp;
242
243 /* convert "com.android.Blah" to "com/android/Blah" */
244 slashClass = strdup(argv[argIdx]);
245 for (cp = slashClass; *cp != '\0'; cp++)
246 if (*cp == '.')
247 *cp = '/';
248
249 startClass = env->FindClass(slashClass);
250 if (startClass == NULL) {
251 fprintf(stderr, "Dalvik VM unable to locate class '%s'\n", slashClass);
252 goto bail;
253 }
254
255 startMeth = env->GetStaticMethodID(startClass,
256 "main", "([Ljava/lang/String;)V");
257 if (startMeth == NULL) {
258 fprintf(stderr, "Dalvik VM unable to find static main(String[]) in '%s'\n",
259 slashClass);
260 goto bail;
261 }
262
263 /*
264 * Make sure the method is public. JNI doesn't prevent us from calling
265 * a private method, so we have to check it explicitly.
266 */
267 if (!methodIsPublic(env, startClass, startMeth))
268 goto bail;
269
270 /*
271 * Invoke main().
272 */
273 env->CallStaticVoidMethod(startClass, startMeth, strArray);
274
275 if (!env->ExceptionCheck())
276 result = 0;
277
278 bail:
279 /*printf("Shutting down Dalvik VM\n");*/
280 if (vm != NULL) {
281 /*
282 * This allows join() and isAlive() on the main thread to work
283 * correctly, and also provides uncaught exception handling.
284 */
285 if (vm->DetachCurrentThread() != JNI_OK) {
286 fprintf(stderr, "Warning: unable to detach main thread\n");
287 result = 1;
288 }
289
290 if (vm->DestroyJavaVM() != 0)
291 fprintf(stderr, "Warning: Dalvik VM did not shut down cleanly\n");
292 /*printf("\nDalvik VM has exited\n");*/
293 }
294
295 for (i = 0; i < optionCount; i++)
296 free((char*) options[i].optionString);
297 free(options);
298 free(slashClass);
299 /*printf("--- VM is down, process exiting\n");*/
300 return result;
301 }
302