1 /*
2  * Copyright (c) 1997, 2010, 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 #include <stdlib.h>
27 #include <string.h>
28 #include <stdio.h>
29 
30 #include "jvm.h"
31 #include "jni.h"
32 #include "jni_util.h"
33 
34 /* Due to a bug in the win32 C runtime library strings
35  * such as "z:" need to be appended with a "." so we
36  * must allocate at least 4 bytes to allow room for
37  * this expansion. See 4235353 for details.
38  */
39 #define MALLOC_MIN4(len) ((char *)malloc((len) + 1 < 4 ? 4 : (len) + 1))
40 
41 /**
42  * Throw a Java exception by name. Similar to SignalError.
43  */
44 JNIEXPORT void JNICALL
JNU_ThrowByName(JNIEnv * env,const char * name,const char * msg)45 JNU_ThrowByName(JNIEnv *env, const char *name, const char *msg)
46 {
47     jclass cls = (*env)->FindClass(env, name);
48 
49     if (cls != 0) /* Otherwise an exception has already been thrown */
50         (*env)->ThrowNew(env, cls, msg);
51 }
52 
53 /* JNU_Throw common exceptions */
54 
55 JNIEXPORT void JNICALL
JNU_ThrowNullPointerException(JNIEnv * env,const char * msg)56 JNU_ThrowNullPointerException(JNIEnv *env, const char *msg)
57 {
58     JNU_ThrowByName(env, "java/lang/NullPointerException", msg);
59 }
60 
61 JNIEXPORT void JNICALL
JNU_ThrowArrayIndexOutOfBoundsException(JNIEnv * env,const char * msg)62 JNU_ThrowArrayIndexOutOfBoundsException(JNIEnv *env, const char *msg)
63 {
64     JNU_ThrowByName(env, "java/lang/ArrayIndexOutOfBoundsException", msg);
65 }
66 
67 JNIEXPORT void JNICALL
JNU_ThrowOutOfMemoryError(JNIEnv * env,const char * msg)68 JNU_ThrowOutOfMemoryError(JNIEnv *env, const char *msg)
69 {
70     JNU_ThrowByName(env, "java/lang/OutOfMemoryError", msg);
71 }
72 
73 JNIEXPORT void JNICALL
JNU_ThrowIllegalArgumentException(JNIEnv * env,const char * msg)74 JNU_ThrowIllegalArgumentException(JNIEnv *env, const char *msg)
75 {
76     JNU_ThrowByName(env, "java/lang/IllegalArgumentException", msg);
77 }
78 
79 JNIEXPORT void JNICALL
JNU_ThrowIllegalAccessError(JNIEnv * env,const char * msg)80 JNU_ThrowIllegalAccessError(JNIEnv *env, const char *msg)
81 {
82     JNU_ThrowByName(env, "java/lang/IllegalAccessError", msg);
83 }
84 
85 JNIEXPORT void JNICALL
JNU_ThrowIllegalAccessException(JNIEnv * env,const char * msg)86 JNU_ThrowIllegalAccessException(JNIEnv *env, const char *msg)
87 {
88     JNU_ThrowByName(env, "java/lang/IllegalAccessException", msg);
89 }
90 
91 JNIEXPORT void JNICALL
JNU_ThrowInternalError(JNIEnv * env,const char * msg)92 JNU_ThrowInternalError(JNIEnv *env, const char *msg)
93 {
94     JNU_ThrowByName(env, "java/lang/InternalError", msg);
95 }
96 
97 JNIEXPORT void JNICALL
JNU_ThrowNoSuchFieldException(JNIEnv * env,const char * msg)98 JNU_ThrowNoSuchFieldException(JNIEnv *env, const char *msg)
99 {
100     JNU_ThrowByName(env, "java/lang/NoSuchFieldException", msg);
101 }
102 
103 JNIEXPORT void JNICALL
JNU_ThrowNoSuchMethodException(JNIEnv * env,const char * msg)104 JNU_ThrowNoSuchMethodException(JNIEnv *env, const char *msg)
105 {
106     JNU_ThrowByName(env, "java/lang/NoSuchMethodException", msg);
107 }
108 
109 JNIEXPORT void JNICALL
JNU_ThrowClassNotFoundException(JNIEnv * env,const char * msg)110 JNU_ThrowClassNotFoundException(JNIEnv *env, const char *msg)
111 {
112     JNU_ThrowByName(env, "java/lang/ClassNotFoundException", msg);
113 }
114 
115 JNIEXPORT void JNICALL
JNU_ThrowNumberFormatException(JNIEnv * env,const char * msg)116 JNU_ThrowNumberFormatException(JNIEnv *env, const char *msg)
117 {
118     JNU_ThrowByName(env, "java/lang/NumberFormatException", msg);
119 }
120 
121 JNIEXPORT void JNICALL
JNU_ThrowIOException(JNIEnv * env,const char * msg)122 JNU_ThrowIOException(JNIEnv *env, const char *msg)
123 {
124     JNU_ThrowByName(env, "java/io/IOException", msg);
125 }
126 
127 JNIEXPORT void JNICALL
JNU_ThrowNoSuchFieldError(JNIEnv * env,const char * msg)128 JNU_ThrowNoSuchFieldError(JNIEnv *env, const char *msg)
129 {
130     JNU_ThrowByName(env, "java/lang/NoSuchFieldError", msg);
131 }
132 
133 JNIEXPORT void JNICALL
JNU_ThrowNoSuchMethodError(JNIEnv * env,const char * msg)134 JNU_ThrowNoSuchMethodError(JNIEnv *env, const char *msg)
135 {
136     JNU_ThrowByName(env, "java/lang/NoSuchMethodError", msg);
137 }
138 
139 JNIEXPORT void JNICALL
JNU_ThrowStringIndexOutOfBoundsException(JNIEnv * env,const char * msg)140 JNU_ThrowStringIndexOutOfBoundsException(JNIEnv *env, const char *msg)
141 {
142     JNU_ThrowByName(env, "java/lang/StringIndexOutOfBoundsException", msg);
143 }
144 
145 JNIEXPORT void JNICALL
JNU_ThrowInstantiationException(JNIEnv * env,const char * msg)146 JNU_ThrowInstantiationException(JNIEnv *env, const char *msg)
147 {
148     JNU_ThrowByName(env, "java/lang/InstantiationException", msg);
149 }
150 
151 
152 /* Throw an exception by name, using the string returned by
153  * JVM_LastErrorString for the detail string.  If the last-error
154  * string is NULL, use the given default detail string.
155  */
156 JNIEXPORT void JNICALL
JNU_ThrowByNameWithLastError(JNIEnv * env,const char * name,const char * defaultDetail)157 JNU_ThrowByNameWithLastError(JNIEnv *env, const char *name,
158                              const char *defaultDetail)
159 {
160     char buf[256];
161     int n = JVM_GetLastErrorString(buf, sizeof(buf));
162 
163     if (n > 0) {
164         jstring s = JNU_NewStringPlatform(env, buf);
165         if (s != NULL) {
166             jobject x = JNU_NewObjectByName(env, name,
167                                             "(Ljava/lang/String;)V", s);
168             if (x != NULL) {
169                 (*env)->Throw(env, x);
170             }
171         }
172     }
173     if (!(*env)->ExceptionOccurred(env)) {
174         JNU_ThrowByName(env, name, defaultDetail);
175     }
176 }
177 
178 /* Throw an IOException, using the last-error string for the detail
179  * string.  If the last-error string is NULL, use the given default
180  * detail string.
181  */
182 JNIEXPORT void JNICALL
JNU_ThrowIOExceptionWithLastError(JNIEnv * env,const char * defaultDetail)183 JNU_ThrowIOExceptionWithLastError(JNIEnv *env, const char *defaultDetail)
184 {
185     JNU_ThrowByNameWithLastError(env, "java/io/IOException", defaultDetail);
186 }
187 
188 
189 JNIEXPORT jvalue JNICALL
JNU_CallStaticMethodByName(JNIEnv * env,jboolean * hasException,const char * class_name,const char * name,const char * signature,...)190 JNU_CallStaticMethodByName(JNIEnv *env,
191                            jboolean *hasException,
192                            const char *class_name,
193                            const char *name,
194                            const char *signature,
195                            ...)
196 {
197     jclass clazz;
198     jmethodID mid;
199     va_list args;
200     jvalue result;
201     const char *p = signature;
202 
203     /* find out the return type */
204     while (*p && *p != ')')
205         p++;
206     p++;
207 
208     result.i = 0;
209 
210     if ((*env)->EnsureLocalCapacity(env, 3) < 0)
211         goto done2;
212 
213     clazz = (*env)->FindClass(env, class_name);
214     if (clazz == 0)
215         goto done2;
216     mid = (*env)->GetStaticMethodID(env, clazz, name, signature);
217     if (mid == 0)
218         goto done1;
219     va_start(args, signature);
220     switch (*p) {
221     case 'V':
222         (*env)->CallStaticVoidMethodV(env, clazz, mid, args);
223         break;
224     case '[':
225     case 'L':
226         result.l = (*env)->CallStaticObjectMethodV(env, clazz, mid, args);
227         break;
228     case 'Z':
229         result.z = (*env)->CallStaticBooleanMethodV(env, clazz, mid, args);
230         break;
231     case 'B':
232         result.b = (*env)->CallStaticByteMethodV(env, clazz, mid, args);
233         break;
234     case 'C':
235         result.c = (*env)->CallStaticCharMethodV(env, clazz, mid, args);
236         break;
237     case 'S':
238         result.s = (*env)->CallStaticShortMethodV(env, clazz, mid, args);
239         break;
240     case 'I':
241         result.i = (*env)->CallStaticIntMethodV(env, clazz, mid, args);
242         break;
243     case 'J':
244         result.j = (*env)->CallStaticLongMethodV(env, clazz, mid, args);
245         break;
246     case 'F':
247         result.f = (*env)->CallStaticFloatMethodV(env, clazz, mid, args);
248         break;
249     case 'D':
250         result.d = (*env)->CallStaticDoubleMethodV(env, clazz, mid, args);
251         break;
252     default:
253         (*env)->FatalError(env, "JNU_CallStaticMethodByName: illegal signature");
254     }
255     va_end(args);
256 
257  done1:
258     (*env)->DeleteLocalRef(env, clazz);
259  done2:
260     if (hasException) {
261         *hasException = (*env)->ExceptionCheck(env);
262     }
263     return result;
264 }
265 
266 JNIEXPORT jvalue JNICALL
JNU_CallMethodByName(JNIEnv * env,jboolean * hasException,jobject obj,const char * name,const char * signature,...)267 JNU_CallMethodByName(JNIEnv *env,
268                      jboolean *hasException,
269                      jobject obj,
270                      const char *name,
271                      const char *signature,
272                      ...)
273 {
274     jvalue result;
275     va_list args;
276 
277     va_start(args, signature);
278     result = JNU_CallMethodByNameV(env, hasException, obj, name, signature,
279                                    args);
280     va_end(args);
281 
282     return result;
283 }
284 
285 
286 JNIEXPORT jvalue JNICALL
JNU_CallMethodByNameV(JNIEnv * env,jboolean * hasException,jobject obj,const char * name,const char * signature,va_list args)287 JNU_CallMethodByNameV(JNIEnv *env,
288                       jboolean *hasException,
289                       jobject obj,
290                       const char *name,
291                       const char *signature,
292                       va_list args)
293 {
294     jclass clazz;
295     jmethodID mid;
296     jvalue result;
297     const char *p = signature;
298 
299     /* find out the return type */
300     while (*p && *p != ')')
301         p++;
302     p++;
303 
304     result.i = 0;
305 
306     if ((*env)->EnsureLocalCapacity(env, 3) < 0)
307         goto done2;
308 
309     clazz = (*env)->GetObjectClass(env, obj);
310     mid = (*env)->GetMethodID(env, clazz, name, signature);
311     if (mid == 0)
312         goto done1;
313 
314     switch (*p) {
315     case 'V':
316         (*env)->CallVoidMethodV(env, obj, mid, args);
317         break;
318     case '[':
319     case 'L':
320         result.l = (*env)->CallObjectMethodV(env, obj, mid, args);
321         break;
322     case 'Z':
323         result.z = (*env)->CallBooleanMethodV(env, obj, mid, args);
324         break;
325     case 'B':
326         result.b = (*env)->CallByteMethodV(env, obj, mid, args);
327         break;
328     case 'C':
329         result.c = (*env)->CallCharMethodV(env, obj, mid, args);
330         break;
331     case 'S':
332         result.s = (*env)->CallShortMethodV(env, obj, mid, args);
333         break;
334     case 'I':
335         result.i = (*env)->CallIntMethodV(env, obj, mid, args);
336         break;
337     case 'J':
338         result.j = (*env)->CallLongMethodV(env, obj, mid, args);
339         break;
340     case 'F':
341         result.f = (*env)->CallFloatMethodV(env, obj, mid, args);
342         break;
343     case 'D':
344         result.d = (*env)->CallDoubleMethodV(env, obj, mid, args);
345         break;
346     default:
347         (*env)->FatalError(env, "JNU_CallMethodByNameV: illegal signature");
348     }
349  done1:
350     (*env)->DeleteLocalRef(env, clazz);
351  done2:
352     if (hasException) {
353         *hasException = (*env)->ExceptionCheck(env);
354     }
355     return result;
356 }
357 
358 JNIEXPORT jobject JNICALL
JNU_NewObjectByName(JNIEnv * env,const char * class_name,const char * constructor_sig,...)359 JNU_NewObjectByName(JNIEnv *env, const char *class_name,
360                     const char *constructor_sig, ...)
361 {
362     jobject obj = NULL;
363 
364     jclass cls = 0;
365     jmethodID cls_initMID;
366     va_list args;
367 
368     if ((*env)->EnsureLocalCapacity(env, 2) < 0)
369         goto done;
370 
371     cls = (*env)->FindClass(env, class_name);
372     if (cls == 0) {
373         goto done;
374     }
375     cls_initMID  = (*env)->GetMethodID(env, cls,
376                                        "<init>", constructor_sig);
377     if (cls_initMID == NULL) {
378         goto done;
379     }
380     va_start(args, constructor_sig);
381     obj = (*env)->NewObjectV(env, cls, cls_initMID, args);
382     va_end(args);
383 
384  done:
385     (*env)->DeleteLocalRef(env, cls);
386     return obj;
387 }
388 
389 // Android removed: Deal solely in UTF-8
390 //
391 // /* Optimized for char set ISO_8559_1 */
392 // static jstring
393 // newString8859_1(JNIEnv *env, const char *str)
394 // {
395 //     int len = (int)strlen(str);
396 //     jchar buf[512];
397 //     jchar *str1;
398 //     jstring result;
399 //     int i;
400 //
401 //     if (len > 512) {
402 //         str1 = (jchar *)malloc(len * sizeof(jchar));
403 //         if (str1 == 0) {
404 //             JNU_ThrowOutOfMemoryError(env, 0);
405 //             return 0;
406 //         }
407 //     } else
408 //         str1 = buf;
409 //
410 //     for (i=0;i<len;i++)
411 //         str1[i] = (unsigned char)str[i];
412 //     result = (*env)->NewString(env, str1, len);
413 //     if (str1 != buf)
414 //         free(str1);
415 //     return result;
416 // }
417 //
418 // static const char*
419 // getString8859_1Chars(JNIEnv *env, jstring jstr)
420 // {
421 //     int i;
422 //     char *result;
423 //     jint len = (*env)->GetStringLength(env, jstr);
424 //     const jchar *str = (*env)->GetStringCritical(env, jstr, 0);
425 //     if (str == 0) {
426 //         return 0;
427 //     }
428 //
429 //     result = MALLOC_MIN4(len);
430 //     if (result == 0) {
431 //         (*env)->ReleaseStringCritical(env, jstr, str);
432 //         JNU_ThrowOutOfMemoryError(env, 0);
433 //         return 0;
434 //     }
435 //
436 //     for (i=0; i<len; i++) {
437 //         jchar unicode = str[i];
438 //         if (unicode <= 0x00ff)
439 //             result[i] = (char)unicode;
440 //         else
441 //             result[i] = '?';
442 //     }
443 //
444 //     result[len] = 0;
445 //     (*env)->ReleaseStringCritical(env, jstr, str);
446 //     return result;
447 // }
448 //
449 //
450 // /* Optimized for char set ISO646-US (us-ascii) */
451 // static jstring
452 // newString646_US(JNIEnv *env, const char *str)
453 // {
454 //     int len = strlen(str);
455 //     jchar buf[512];
456 //     jchar *str1;
457 //     jstring result;
458 //     int i;
459 //
460 //     if (len > 512) {
461 //         str1 = (jchar *)malloc(len * sizeof(jchar));
462 //         if (str1 == 0) {
463 //             JNU_ThrowOutOfMemoryError(env, 0);
464 //             return 0;
465 //         }
466 //     } else
467 //         str1 = buf;
468 //
469 //     for (i=0; i<len; i++) {
470 //         unsigned char c = (unsigned char)str[i];
471 //         if (c <= 0x7f)
472 //             str1[i] = c;
473 //         else
474 //             str1[i] = '?';
475 //     }
476 //
477 //     result = (*env)->NewString(env, str1, len);
478 //     if (str1 != buf)
479 //         free(str1);
480 //     return result;
481 // }
482 //
483 // static const char*
484 // getString646_USChars(JNIEnv *env, jstring jstr)
485 // {
486 //     int i;
487 //     char *result;
488 //     jint len = (*env)->GetStringLength(env, jstr);
489 //     const jchar *str = (*env)->GetStringCritical(env, jstr, 0);
490 //     if (str == 0) {
491 //         return 0;
492 //     }
493 //
494 //     result = MALLOC_MIN4(len);
495 //     if (result == 0) {
496 //         (*env)->ReleaseStringCritical(env, jstr, str);
497 //         JNU_ThrowOutOfMemoryError(env, 0);
498 //         return 0;
499 //     }
500 //
501 //     for (i=0; i<len; i++) {
502 //         jchar unicode = str[i];
503 //         if (unicode <= 0x007f )
504 //             result[i] = (char)unicode;
505 //         else
506 //             result[i] = '?';
507 //     }
508 //
509 //     result[len] = 0;
510 //     (*env)->ReleaseStringCritical(env, jstr, str);
511 //     return result;
512 // }
513 //
514 // /* enumeration of c1 row from Cp1252 */
515 // static int cp1252c1chars[32] = {
516 //     0x20AC,0xFFFD,0x201A,0x0192,0x201E,0x2026,0x2020,0x2021,
517 //     0x02C6,0x2030,0x0160,0x2039,0x0152,0xFFFD,0x017D,0xFFFD,
518 //     0xFFFD,0x2018,0x2019,0x201C,0x201D,0x2022,0x2013,0x2014,
519 //     0x02Dc,0x2122,0x0161,0x203A,0x0153,0xFFFD,0x017E,0x0178
520 // };
521 //
522 // /* Optimized for char set Cp1252 */
523 // static jstring
524 // newStringCp1252(JNIEnv *env, const char *str)
525 // {
526 //     int len = (int) strlen(str);
527 //     jchar buf[512];
528 //     jchar *str1;
529 //     jstring result;
530 //     int i;
531 //     if (len > 512) {
532 //         str1 = (jchar *)malloc(len * sizeof(jchar));
533 //         if (str1 == 0) {
534 //             JNU_ThrowOutOfMemoryError(env, 0);
535 //             return 0;
536 //         }
537 //     } else
538 //         str1 = buf;
539 //
540 //     for (i=0; i<len; i++) {
541 //         unsigned char c = (unsigned char)str[i];
542 //         if ((c >= 0x80) && (c <= 0x9f))
543 //             str1[i] = cp1252c1chars[c-128];
544 //         else
545 //             str1[i] = c;
546 //     }
547 //
548 //     result = (*env)->NewString(env, str1, len);
549 //     if (str1 != buf)
550 //         free(str1);
551 //     return result;
552 // }
553 //
554 // static const char*
555 // getStringCp1252Chars(JNIEnv *env, jstring jstr)
556 // {
557 //     int i;
558 //     char *result;
559 //     jint len = (*env)->GetStringLength(env, jstr);
560 //     const jchar *str = (*env)->GetStringCritical(env, jstr, 0);
561 //     if (str == 0) {
562 //         return 0;
563 //     }
564 //
565 //     result = MALLOC_MIN4(len);
566 //     if (result == 0) {
567 //         (*env)->ReleaseStringCritical(env, jstr, str);
568 //         JNU_ThrowOutOfMemoryError(env, 0);
569 //         return 0;
570 //     }
571 //
572 //     for (i=0; i<len; i++) {
573 //         jchar c = str[i];
574 //         if (c < 256)
575 //             result[i] = (char)c;
576 //         else switch(c) {
577 //             case 0x20AC: result[i] = (char)0x80; break;
578 //             case 0x201A: result[i] = (char)0x82; break;
579 //             case 0x0192: result[i] = (char)0x83; break;
580 //             case 0x201E: result[i] = (char)0x84; break;
581 //             case 0x2026: result[i] = (char)0x85; break;
582 //             case 0x2020: result[i] = (char)0x86; break;
583 //             case 0x2021: result[i] = (char)0x87; break;
584 //             case 0x02C6: result[i] = (char)0x88; break;
585 //             case 0x2030: result[i] = (char)0x89; break;
586 //             case 0x0160: result[i] = (char)0x8A; break;
587 //             case 0x2039: result[i] = (char)0x8B; break;
588 //             case 0x0152: result[i] = (char)0x8C; break;
589 //             case 0x017D: result[i] = (char)0x8E; break;
590 //             case 0x2018: result[i] = (char)0x91; break;
591 //             case 0x2019: result[i] = (char)0x92; break;
592 //             case 0x201C: result[i] = (char)0x93; break;
593 //             case 0x201D: result[i] = (char)0x94; break;
594 //             case 0x2022: result[i] = (char)0x95; break;
595 //             case 0x2013: result[i] = (char)0x96; break;
596 //             case 0x2014: result[i] = (char)0x97; break;
597 //             case 0x02DC: result[i] = (char)0x98; break;
598 //             case 0x2122: result[i] = (char)0x99; break;
599 //             case 0x0161: result[i] = (char)0x9A; break;
600 //             case 0x203A: result[i] = (char)0x9B; break;
601 //             case 0x0153: result[i] = (char)0x9C; break;
602 //             case 0x017E: result[i] = (char)0x9E; break;
603 //             case 0x0178: result[i] = (char)0x9F; break;
604 //             default:     result[i] = '?';  break;
605 //         }
606 //     }
607 //
608 //     result[len] = 0;
609 //     (*env)->ReleaseStringCritical(env, jstr, str);
610 //     return result;
611 // }
612 //
613 // static int fastEncoding = NO_ENCODING_YET;
614 // static jstring jnuEncoding = NULL;
615 //
616 // /* Cached method IDs */
617 // static jmethodID String_init_ID;        /* String(byte[], enc) */
618 // static jmethodID String_getBytes_ID;    /* String.getBytes(enc) */
619 //
620 // int getFastEncoding() {
621 //     return fastEncoding;
622 // }
623 //
624 // /* Initialize the fast encoding.  If the "sun.jnu.encoding" property
625 //  * has not yet been set, we leave fastEncoding == NO_ENCODING_YET.
626 //  */
627 // void
628 // initializeEncoding(JNIEnv *env)
629 // {
630 //     jstring propname = 0;
631 //     jstring enc = 0;
632 //
633 //     if ((*env)->EnsureLocalCapacity(env, 3) < 0)
634 //         return;
635 //
636 //     propname = (*env)->NewStringUTF(env, "sun.jnu.encoding");
637 //     if (propname) {
638 //         jboolean exc;
639 //         enc = JNU_CallStaticMethodByName
640 //                        (env,
641 //                         &exc,
642 //                         "java/lang/System",
643 //                         "getProperty",
644 //                         "(Ljava/lang/String;)Ljava/lang/String;",
645 //                         propname).l;
646 //         if (!exc) {
647 //             if (enc) {
648 //                 const char* encname = (*env)->GetStringUTFChars(env, enc, 0);
649 //                 if (encname) {
650 //            /*
651 //             * On Solaris with nl_langinfo() called in GetJavaProperties():
652 //             *
653 //             *   locale undefined -> NULL -> hardcoded default
654 //             *   "C" locale       -> "" -> hardcoded default     (on 2.6)
655 //             *   "C" locale       -> "ISO646-US"                 (on Sol 7/8)
656 //             *   "en_US" locale -> "ISO8859-1"
657 //             *   "en_GB" locale -> "ISO8859-1"                   (on Sol 7/8)
658 //             *   "en_UK" locale -> "ISO8859-1"                   (on 2.6)
659 //             */
660 //                     if ((strcmp(encname, "8859_1") == 0) ||
661 //                         (strcmp(encname, "ISO8859-1") == 0) ||
662 //                         (strcmp(encname, "ISO8859_1") == 0))
663 //                         fastEncoding = FAST_8859_1;
664 //                     else if (strcmp(encname, "ISO646-US") == 0)
665 //                         fastEncoding = FAST_646_US;
666 //                     else if (strcmp(encname, "Cp1252") == 0 ||
667 //                              /* This is a temporary fix until we move */
668 //                              /* to wide character versions of all Windows */
669 //                              /* calls. */
670 //                              strcmp(encname, "utf-16le") == 0)
671 //                         fastEncoding = FAST_CP1252;
672 //                     else {
673 //                         fastEncoding = NO_FAST_ENCODING;
674 //                         jnuEncoding = (jstring)(*env)->NewGlobalRef(env, enc);
675 //                     }
676 //                     (*env)->ReleaseStringUTFChars(env, enc, encname);
677 //                 }
678 //             }
679 //         } else {
680 //             (*env)->ExceptionClear(env);
681 //         }
682 //     } else {
683 //         (*env)->ExceptionClear(env);
684 //     }
685 //     (*env)->DeleteLocalRef(env, propname);
686 //     (*env)->DeleteLocalRef(env, enc);
687 //
688 //     /* Initialize method-id cache */
689 //     String_getBytes_ID = (*env)->GetMethodID(env, JNU_ClassString(env),
690 //                                              "getBytes", "(Ljava/lang/String;)[B");
691 //     String_init_ID = (*env)->GetMethodID(env, JNU_ClassString(env),
692 //                                          "<init>", "([BLjava/lang/String;)V");
693 // }
694 //
695 // static jboolean isJNUEncodingSupported = JNI_FALSE;
696 // static jboolean jnuEncodingSupported(JNIEnv *env) {
697 //     jboolean exe;
698 //     if (isJNUEncodingSupported == JNI_TRUE) {
699 //         return JNI_TRUE;
700 //     }
701 //     isJNUEncodingSupported = (jboolean) JNU_CallStaticMethodByName (
702 //                                     env, &exe,
703 //                                     "java/nio/charset/Charset",
704 //                                     "isSupported",
705 //                                     "(Ljava/lang/String;)Z",
706 //                                     jnuEncoding).z;
707 //     return isJNUEncodingSupported;
708 // }
709 
710 
711 JNIEXPORT jstring
JNU_NewStringPlatform(JNIEnv * env,const char * str)712 JNU_NewStringPlatform(JNIEnv *env, const char *str)
713 {
714     // Android-changed: Always use nativeNewStringPlatform.
715     // return JNU_NewStringPlatform(env, str);
716     return nativeNewStringPlatform(env, str);
717 }
718 
719 JNIEXPORT const char *
JNU_GetStringPlatformChars(JNIEnv * env,jstring jstr,jboolean * isCopy)720 JNU_GetStringPlatformChars(JNIEnv *env, jstring jstr, jboolean *isCopy)
721 {
722     // Android-changed: Always use nativeGetStringPlatformChars.
723     // return JNU_GetStringPlatformChars(env, jstr, isCopy);
724     return nativeGetStringPlatformChars(env, jstr, isCopy);
725 }
726 
727 JNIEXPORT void JNICALL
JNU_ReleaseStringPlatformChars(JNIEnv * env,jstring jstr,const char * str)728 JNU_ReleaseStringPlatformChars(JNIEnv *env, jstring jstr, const char *str)
729 {
730     // Android changed : Use the "native" code from the platform a chance
731     // to handle this first.
732     //
733     // free((void *)str);
734     nativeReleaseStringPlatformChars(env, jstr, str);
735 }
736 
737 // Android removed: Deal solely in UTF-8
738 //
739 // JNIEXPORT jstring JNICALL
740 // JNU_NewStringPlatform(JNIEnv *env, const char *str)
741 // {
742 //    jstring result;
743 //    result = nativeNewStringPlatform(env, str);
744 //    if (result == NULL) {
745 //        jbyteArray hab = 0;
746 //        int len;
747 //
748 //        if (fastEncoding == NO_ENCODING_YET)
749 //            initializeEncoding(env);
750 //
751 //        if ((fastEncoding == FAST_8859_1) || (fastEncoding == NO_ENCODING_YET))
752 //            return newString8859_1(env, str);
753 //        if (fastEncoding == FAST_646_US)
754 //            return newString646_US(env, str);
755 //        if (fastEncoding == FAST_CP1252)
756 //            return newStringCp1252(env, str);
757 //
758 //        if ((*env)->EnsureLocalCapacity(env, 2) < 0)
759 //            return NULL;
760 //
761 //        len = (int)strlen(str);
762 //        hab = (*env)->NewByteArray(env, len);
763 //        if (hab != 0) {
764 //            (*env)->SetByteArrayRegion(env, hab, 0, len, (jbyte *)str);
765 //            if (jnuEncodingSupported(env)) {
766 //                result = (*env)->NewObject(env, JNU_ClassString(env),
767 //                                           String_init_ID, hab, jnuEncoding);
768 //            } else {
769 //                /*If the encoding specified in sun.jnu.encoding is not endorsed
770 //                  by "Charset.isSupported" we have to fall back to use String(byte[])
771 //                  explicitly here without specifying the encoding name, in which the
772 //                  StringCoding class will pickup the iso-8859-1 as the fallback
773 //                  converter for us.
774 //                 */
775 //                jmethodID mid = (*env)->GetMethodID(env, JNU_ClassString(env),
776 //                                                    "<init>", "([B)V");
777 //                result = (*env)->NewObject(env, JNU_ClassString(env), mid, hab);
778 //            }
779 //            (*env)->DeleteLocalRef(env, hab);
780 //            return result;
781 //        }
782 //    }
783 //    return NULL;
784 //}
785 //
786 //
787 //JNIEXPORT const char * JNICALL
788 //JNU_GetStringPlatformChars(JNIEnv *env, jstring jstr, jboolean *isCopy)
789 //{
790 //    char *result = nativeGetStringPlatformChars(env, jstr, isCopy);
791 //    if (result == NULL) {
792 //
793 //        jbyteArray hab = 0;
794 //
795 //        if (isCopy)
796 //            *isCopy = JNI_TRUE;
797 //
798 //        if (fastEncoding == NO_ENCODING_YET)
799 //            initializeEncoding(env);
800 //
801 //        if ((fastEncoding == FAST_8859_1) || (fastEncoding == NO_ENCODING_YET))
802 //            return getString8859_1Chars(env, jstr);
803 //        if (fastEncoding == FAST_646_US)
804 //            return getString646_USChars(env, jstr);
805 //        if (fastEncoding == FAST_CP1252)
806 //            return getStringCp1252Chars(env, jstr);
807 //
808 //        if ((*env)->EnsureLocalCapacity(env, 2) < 0)
809 //            return 0;
810 //
811 //        if (jnuEncodingSupported(env)) {
812 //            hab = (*env)->CallObjectMethod(env, jstr, String_getBytes_ID, jnuEncoding);
813 //        } else {
814 //            jmethodID mid = (*env)->GetMethodID(env, JNU_ClassString(env),
815 //                                                "getBytes", "()[B");
816 //            hab = (*env)->CallObjectMethod(env, jstr, mid);
817 //        }
818 //
819 //        if (!(*env)->ExceptionCheck(env)) {
820 //            jint len = (*env)->GetArrayLength(env, hab);
821 //            result = MALLOC_MIN4(len);
822 //            if (result == 0) {
823 //                JNU_ThrowOutOfMemoryError(env, 0);
824 //                (*env)->DeleteLocalRef(env, hab);
825 //                return 0;
826 //            }
827 //            (*env)->GetByteArrayRegion(env, hab, 0, len, (jbyte *)result);
828 //            result[len] = 0; /* NULL-terminate */
829 //        }
830 //
831 //        (*env)->DeleteLocalRef(env, hab);
832 //    }
833 //    return result;
834 //}
835 
836 
837 /*
838  * Export the platform dependent path canonicalization so that
839  * VM can find it when loading system classes.
840  *
841  */
842 // Android-removed: Remove unused  Canonicalize().
843 /*
844 // Android-changed: hidden to avoid conflict with libm (b/135018555)
845 __attribute__((visibility("hidden")))
846 extern int canonicalize(char *path, const char *out, int len);
847 
848 JNIEXPORT int
849 Canonicalize(JNIEnv *env, char *orig, char *out, int len)
850 {
851     * canonicalize an already natived path *
852     return canonicalize(orig, out, len);
853 }
854 */
855 
856 JNIEXPORT jclass JNICALL
JNU_ClassString(JNIEnv * env)857 JNU_ClassString(JNIEnv *env)
858 {
859     static jclass cls = 0;
860     if (cls == 0) {
861         jclass c;
862         if ((*env)->EnsureLocalCapacity(env, 1) < 0)
863             return 0;
864         c = (*env)->FindClass(env, "java/lang/String");
865         cls = (*env)->NewGlobalRef(env, c);
866         (*env)->DeleteLocalRef(env, c);
867     }
868     return cls;
869 }
870 
871 JNIEXPORT jclass JNICALL
JNU_ClassClass(JNIEnv * env)872 JNU_ClassClass(JNIEnv *env)
873 {
874     static jclass cls = 0;
875     if (cls == 0) {
876         jclass c;
877         if ((*env)->EnsureLocalCapacity(env, 1) < 0)
878             return 0;
879         c = (*env)->FindClass(env, "java/lang/Class");
880         cls = (*env)->NewGlobalRef(env, c);
881         (*env)->DeleteLocalRef(env, c);
882     }
883     return cls;
884 }
885 
886 JNIEXPORT jclass JNICALL
JNU_ClassObject(JNIEnv * env)887 JNU_ClassObject(JNIEnv *env)
888 {
889     static jclass cls = 0;
890     if (cls == 0) {
891         jclass c;
892         if ((*env)->EnsureLocalCapacity(env, 1) < 0)
893             return 0;
894         c = (*env)->FindClass(env, "java/lang/Object");
895         cls = (*env)->NewGlobalRef(env, c);
896         (*env)->DeleteLocalRef(env, c);
897     }
898     return cls;
899 }
900 
901 JNIEXPORT jclass JNICALL
JNU_ClassThrowable(JNIEnv * env)902 JNU_ClassThrowable(JNIEnv *env)
903 {
904     static jclass cls = 0;
905     if (cls == 0) {
906         jclass c;
907         if ((*env)->EnsureLocalCapacity(env, 1) < 0)
908             return 0;
909         c = (*env)->FindClass(env, "java/lang/Throwable");
910         cls = (*env)->NewGlobalRef(env, c);
911         (*env)->DeleteLocalRef(env, c);
912     }
913     return cls;
914 }
915 
916 JNIEXPORT jint JNICALL
JNU_CopyObjectArray(JNIEnv * env,jobjectArray dst,jobjectArray src,jint count)917 JNU_CopyObjectArray(JNIEnv *env, jobjectArray dst, jobjectArray src,
918                          jint count)
919 {
920     int i;
921     if ((*env)->EnsureLocalCapacity(env, 1) < 0)
922         return -1;
923     for (i=0; i<count; i++) {
924         jstring p = (*env)->GetObjectArrayElement(env, src, i);
925         (*env)->SetObjectArrayElement(env, dst, i, p);
926         (*env)->DeleteLocalRef(env, p);
927     }
928     return 0;
929 }
930 
931 JNIEXPORT void * JNICALL
JNU_GetEnv(JavaVM * vm,jint version)932 JNU_GetEnv(JavaVM *vm, jint version)
933 {
934     void *env;
935     (*vm)->GetEnv(vm, &env, version);
936     return env;
937 }
938 
939 JNIEXPORT jint JNICALL
JNU_IsInstanceOfByName(JNIEnv * env,jobject object,char * classname)940 JNU_IsInstanceOfByName(JNIEnv *env, jobject object, char* classname)
941 {
942     jclass cls;
943     if ((*env)->EnsureLocalCapacity(env, 1) < 0)
944         return JNI_ERR;
945     cls = (*env)->FindClass(env, classname);
946     if (cls != NULL) {
947         jint result = (*env)->IsInstanceOf(env, object, cls);
948         (*env)->DeleteLocalRef(env, cls);
949         return result;
950     }
951     return JNI_ERR;
952 }
953 
954 JNIEXPORT jboolean JNICALL
JNU_Equals(JNIEnv * env,jobject object1,jobject object2)955 JNU_Equals(JNIEnv *env, jobject object1, jobject object2)
956 {
957     static jmethodID mid = NULL;
958     if (mid == NULL) {
959         mid = (*env)->GetMethodID(env, JNU_ClassObject(env), "equals",
960                                   "(Ljava/lang/Object;)Z");
961     }
962     return (*env)->CallBooleanMethod(env, object1, mid, object2);
963 }
964 
965 
966 /************************************************************************
967  * Thread calls
968  */
969 
970 static jmethodID Object_waitMID;
971 static jmethodID Object_notifyMID;
972 static jmethodID Object_notifyAllMID;
973 
974 JNIEXPORT void JNICALL
JNU_MonitorWait(JNIEnv * env,jobject object,jlong timeout)975 JNU_MonitorWait(JNIEnv *env, jobject object, jlong timeout)
976 {
977     if (object == NULL) {
978         JNU_ThrowNullPointerException(env, "JNU_MonitorWait argument");
979         return;
980     }
981     if (Object_waitMID == NULL) {
982         jclass cls = JNU_ClassObject(env);
983         if (cls == NULL) {
984             return;
985         }
986         Object_waitMID = (*env)->GetMethodID(env, cls, "wait", "(J)V");
987         if (Object_waitMID == NULL) {
988             return;
989         }
990     }
991     (*env)->CallVoidMethod(env, object, Object_waitMID, timeout);
992 }
993 
994 JNIEXPORT void JNICALL
JNU_Notify(JNIEnv * env,jobject object)995 JNU_Notify(JNIEnv *env, jobject object)
996 {
997     if (object == NULL) {
998         JNU_ThrowNullPointerException(env, "JNU_Notify argument");
999         return;
1000     }
1001     if (Object_notifyMID == NULL) {
1002         jclass cls = JNU_ClassObject(env);
1003         if (cls == NULL) {
1004             return;
1005         }
1006         Object_notifyMID = (*env)->GetMethodID(env, cls, "notify", "()V");
1007         if (Object_notifyMID == NULL) {
1008             return;
1009         }
1010     }
1011     (*env)->CallVoidMethod(env, object, Object_notifyMID);
1012 }
1013 
1014 JNIEXPORT void JNICALL
JNU_NotifyAll(JNIEnv * env,jobject object)1015 JNU_NotifyAll(JNIEnv *env, jobject object)
1016 {
1017     if (object == NULL) {
1018         JNU_ThrowNullPointerException(env, "JNU_NotifyAll argument");
1019         return;
1020     }
1021     if (Object_notifyAllMID == NULL) {
1022         jclass cls = JNU_ClassObject(env);
1023         if (cls == NULL) {
1024             return;
1025         }
1026         Object_notifyAllMID = (*env)->GetMethodID(env, cls,"notifyAll", "()V");
1027         if (Object_notifyAllMID == NULL) {
1028             return;
1029         }
1030     }
1031     (*env)->CallVoidMethod(env, object, Object_notifyAllMID);
1032 }
1033 
1034 
1035 /************************************************************************
1036  * Debugging utilities
1037  */
1038 
1039 JNIEXPORT void JNICALL
JNU_PrintString(JNIEnv * env,char * hdr,jstring string)1040 JNU_PrintString(JNIEnv *env, char *hdr, jstring string)
1041 {
1042     if (string == NULL) {
1043         fprintf(stderr, "%s: is NULL\n", hdr);
1044     } else {
1045         const char *stringPtr = JNU_GetStringPlatformChars(env, string, 0);
1046         if (stringPtr == 0)
1047             return;
1048         fprintf(stderr, "%s: %s\n", hdr, stringPtr);
1049         JNU_ReleaseStringPlatformChars(env, string, stringPtr);
1050     }
1051 }
1052 
1053 JNIEXPORT void JNICALL
JNU_PrintClass(JNIEnv * env,char * hdr,jobject object)1054 JNU_PrintClass(JNIEnv *env, char* hdr, jobject object)
1055 {
1056     if (object == NULL) {
1057         fprintf(stderr, "%s: object is NULL\n", hdr);
1058         return;
1059     } else {
1060         jclass cls = (*env)->GetObjectClass(env, object);
1061         jstring clsName = JNU_ToString(env, cls);
1062         JNU_PrintString(env, hdr, clsName);
1063         (*env)->DeleteLocalRef(env, cls);
1064         (*env)->DeleteLocalRef(env, clsName);
1065     }
1066 }
1067 
1068 JNIEXPORT jstring JNICALL
JNU_ToString(JNIEnv * env,jobject object)1069 JNU_ToString(JNIEnv *env, jobject object)
1070 {
1071     if (object == NULL) {
1072         return (*env)->NewStringUTF(env, "NULL");
1073     } else {
1074         return (jstring)JNU_CallMethodByName(env,
1075                                              NULL,
1076                                              object,
1077                                              "toString",
1078                                              "()Ljava/lang/String;").l;
1079     }
1080 }
1081 
1082 JNIEXPORT jvalue JNICALL
JNU_GetFieldByName(JNIEnv * env,jboolean * hasException,jobject obj,const char * name,const char * signature)1083 JNU_GetFieldByName(JNIEnv *env,
1084                    jboolean *hasException,
1085                    jobject obj,
1086                    const char *name,
1087                    const char *signature)
1088 {
1089     jclass cls;
1090     jfieldID fid;
1091     jvalue result;
1092 
1093     result.i = 0;
1094 
1095     if ((*env)->EnsureLocalCapacity(env, 3) < 0)
1096         goto done2;
1097 
1098     cls = (*env)->GetObjectClass(env, obj);
1099     fid = (*env)->GetFieldID(env, cls, name, signature);
1100     if (fid == 0)
1101         goto done1;
1102 
1103     switch (*signature) {
1104     case '[':
1105     case 'L':
1106         result.l = (*env)->GetObjectField(env, obj, fid);
1107         break;
1108     case 'Z':
1109         result.z = (*env)->GetBooleanField(env, obj, fid);
1110         break;
1111     case 'B':
1112         result.b = (*env)->GetByteField(env, obj, fid);
1113         break;
1114     case 'C':
1115         result.c = (*env)->GetCharField(env, obj, fid);
1116         break;
1117     case 'S':
1118         result.s = (*env)->GetShortField(env, obj, fid);
1119         break;
1120     case 'I':
1121         result.i = (*env)->GetIntField(env, obj, fid);
1122         break;
1123     case 'J':
1124         result.j = (*env)->GetLongField(env, obj, fid);
1125         break;
1126     case 'F':
1127         result.f = (*env)->GetFloatField(env, obj, fid);
1128         break;
1129     case 'D':
1130         result.d = (*env)->GetDoubleField(env, obj, fid);
1131         break;
1132 
1133     default:
1134         (*env)->FatalError(env, "JNU_GetFieldByName: illegal signature");
1135     }
1136 
1137  done1:
1138     (*env)->DeleteLocalRef(env, cls);
1139  done2:
1140     if (hasException) {
1141         *hasException = (*env)->ExceptionCheck(env);
1142     }
1143     return result;
1144 }
1145 
1146 JNIEXPORT void JNICALL
JNU_SetFieldByName(JNIEnv * env,jboolean * hasException,jobject obj,const char * name,const char * signature,...)1147 JNU_SetFieldByName(JNIEnv *env,
1148                    jboolean *hasException,
1149                    jobject obj,
1150                    const char *name,
1151                    const char *signature,
1152                    ...)
1153 {
1154     jclass cls;
1155     jfieldID fid;
1156     va_list args;
1157 
1158     if ((*env)->EnsureLocalCapacity(env, 3) < 0)
1159         goto done2;
1160 
1161     cls = (*env)->GetObjectClass(env, obj);
1162     fid = (*env)->GetFieldID(env, cls, name, signature);
1163     if (fid == 0)
1164         goto done1;
1165 
1166     va_start(args, signature);
1167     switch (*signature) {
1168     case '[':
1169     case 'L':
1170         (*env)->SetObjectField(env, obj, fid, va_arg(args, jobject));
1171         break;
1172     case 'Z':
1173         (*env)->SetBooleanField(env, obj, fid, (jboolean)va_arg(args, int));
1174         break;
1175     case 'B':
1176         (*env)->SetByteField(env, obj, fid, (jbyte)va_arg(args, int));
1177         break;
1178     case 'C':
1179         (*env)->SetCharField(env, obj, fid, (jchar)va_arg(args, int));
1180         break;
1181     case 'S':
1182         (*env)->SetShortField(env, obj, fid, (jshort)va_arg(args, int));
1183         break;
1184     case 'I':
1185         (*env)->SetIntField(env, obj, fid, va_arg(args, jint));
1186         break;
1187     case 'J':
1188         (*env)->SetLongField(env, obj, fid, va_arg(args, jlong));
1189         break;
1190     case 'F':
1191         (*env)->SetFloatField(env, obj, fid, (jfloat)va_arg(args, jdouble));
1192         break;
1193     case 'D':
1194         (*env)->SetDoubleField(env, obj, fid, va_arg(args, jdouble));
1195         break;
1196 
1197     default:
1198         (*env)->FatalError(env, "JNU_SetFieldByName: illegal signature");
1199     }
1200     va_end(args);
1201 
1202  done1:
1203     (*env)->DeleteLocalRef(env, cls);
1204  done2:
1205     if (hasException) {
1206         *hasException = (*env)->ExceptionCheck(env);
1207     }
1208 }
1209 
1210 JNIEXPORT jvalue JNICALL
JNU_GetStaticFieldByName(JNIEnv * env,jboolean * hasException,const char * classname,const char * name,const char * signature)1211 JNU_GetStaticFieldByName(JNIEnv *env,
1212                          jboolean *hasException,
1213                          const char *classname,
1214                          const char *name,
1215                          const char *signature)
1216 {
1217     jclass cls;
1218     jfieldID fid;
1219     jvalue result;
1220 
1221     result.i = 0;
1222 
1223     if ((*env)->EnsureLocalCapacity(env, 3) < 0)
1224         goto done2;
1225 
1226     cls = (*env)->FindClass(env, classname);
1227     if (cls == 0)
1228         goto done2;
1229 
1230     fid = (*env)->GetStaticFieldID(env, cls, name, signature);
1231     if (fid == 0)
1232         goto done1;
1233 
1234     switch (*signature) {
1235     case '[':
1236     case 'L':
1237         result.l = (*env)->GetStaticObjectField(env, cls, fid);
1238         break;
1239     case 'Z':
1240         result.z = (*env)->GetStaticBooleanField(env, cls, fid);
1241         break;
1242     case 'B':
1243         result.b = (*env)->GetStaticByteField(env, cls, fid);
1244         break;
1245     case 'C':
1246         result.c = (*env)->GetStaticCharField(env, cls, fid);
1247         break;
1248     case 'S':
1249         result.s = (*env)->GetStaticShortField(env, cls, fid);
1250         break;
1251     case 'I':
1252         result.i = (*env)->GetStaticIntField(env, cls, fid);
1253         break;
1254     case 'J':
1255         result.j = (*env)->GetStaticLongField(env, cls, fid);
1256         break;
1257     case 'F':
1258         result.f = (*env)->GetStaticFloatField(env, cls, fid);
1259         break;
1260     case 'D':
1261         result.d = (*env)->GetStaticDoubleField(env, cls, fid);
1262         break;
1263 
1264     default:
1265         (*env)->FatalError(env, "JNU_GetStaticFieldByName: illegal signature");
1266     }
1267 
1268  done1:
1269     (*env)->DeleteLocalRef(env, cls);
1270  done2:
1271     if (hasException) {
1272         *hasException = (*env)->ExceptionCheck(env);
1273     }
1274     return result;
1275 }
1276 
1277 JNIEXPORT void JNICALL
JNU_SetStaticFieldByName(JNIEnv * env,jboolean * hasException,const char * classname,const char * name,const char * signature,...)1278 JNU_SetStaticFieldByName(JNIEnv *env,
1279                          jboolean *hasException,
1280                          const char *classname,
1281                          const char *name,
1282                          const char *signature,
1283                          ...)
1284 {
1285     jclass cls;
1286     jfieldID fid;
1287     va_list args;
1288 
1289     if ((*env)->EnsureLocalCapacity(env, 3) < 0)
1290         goto done2;
1291 
1292     cls = (*env)->FindClass(env, classname);
1293     if (cls == 0)
1294         goto done2;
1295 
1296     fid = (*env)->GetStaticFieldID(env, cls, name, signature);
1297     if (fid == 0)
1298         goto done1;
1299 
1300     va_start(args, signature);
1301     switch (*signature) {
1302     case '[':
1303     case 'L':
1304         (*env)->SetStaticObjectField(env, cls, fid, va_arg(args, jobject));
1305         break;
1306     case 'Z':
1307         (*env)->SetStaticBooleanField(env, cls, fid, (jboolean)va_arg(args, int));
1308         break;
1309     case 'B':
1310         (*env)->SetStaticByteField(env, cls, fid, (jbyte)va_arg(args, int));
1311         break;
1312     case 'C':
1313         (*env)->SetStaticCharField(env, cls, fid, (jchar)va_arg(args, int));
1314         break;
1315     case 'S':
1316         (*env)->SetStaticShortField(env, cls, fid, (jshort)va_arg(args, int));
1317         break;
1318     case 'I':
1319         (*env)->SetStaticIntField(env, cls, fid, va_arg(args, jint));
1320         break;
1321     case 'J':
1322         (*env)->SetStaticLongField(env, cls, fid, va_arg(args, jlong));
1323         break;
1324     case 'F':
1325         (*env)->SetStaticFloatField(env, cls, fid, (jfloat)va_arg(args, jdouble));
1326         break;
1327     case 'D':
1328         (*env)->SetStaticDoubleField(env, cls, fid, va_arg(args, jdouble));
1329         break;
1330 
1331     default:
1332         (*env)->FatalError(env, "JNU_SetStaticFieldByName: illegal signature");
1333     }
1334     va_end(args);
1335 
1336  done1:
1337     (*env)->DeleteLocalRef(env, cls);
1338  done2:
1339     if (hasException) {
1340         *hasException = (*env)->ExceptionCheck(env);
1341     }
1342 }
1343