Path: blob/aarch64-shenandoah-jdk8u272-b10/jdk/src/share/native/common/jni_util.c
38825 views
/*1* Copyright (c) 1997, 2010, Oracle and/or its affiliates. All rights reserved.2* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.3*4* This code is free software; you can redistribute it and/or modify it5* under the terms of the GNU General Public License version 2 only, as6* published by the Free Software Foundation. Oracle designates this7* particular file as subject to the "Classpath" exception as provided8* by Oracle in the LICENSE file that accompanied this code.9*10* This code is distributed in the hope that it will be useful, but WITHOUT11* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or12* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License13* version 2 for more details (a copy is included in the LICENSE file that14* accompanied this code).15*16* You should have received a copy of the GNU General Public License version17* 2 along with this work; if not, write to the Free Software Foundation,18* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.19*20* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA21* or visit www.oracle.com if you need additional information or have any22* questions.23*/2425#include <stdlib.h>26#include <string.h>2728#include "jvm.h"29#include "io_util.h"3031/* Due to a bug in the win32 C runtime library strings32* such as "z:" need to be appended with a "." so we33* must allocate at least 4 bytes to allow room for34* this expansion. See 4235353 for details.35*/36#define MALLOC_MIN4(len) ((char *)malloc((len) + 1 < 4 ? 4 : (len) + 1))3738/**39* Throw a Java exception by name. Similar to SignalError.40*/41JNIEXPORT void JNICALL42JNU_ThrowByName(JNIEnv *env, const char *name, const char *msg)43{44jclass cls = (*env)->FindClass(env, name);4546if (cls != 0) /* Otherwise an exception has already been thrown */47(*env)->ThrowNew(env, cls, msg);48}4950/* JNU_Throw common exceptions */5152JNIEXPORT void JNICALL53JNU_ThrowNullPointerException(JNIEnv *env, const char *msg)54{55JNU_ThrowByName(env, "java/lang/NullPointerException", msg);56}5758JNIEXPORT void JNICALL59JNU_ThrowArrayIndexOutOfBoundsException(JNIEnv *env, const char *msg)60{61JNU_ThrowByName(env, "java/lang/ArrayIndexOutOfBoundsException", msg);62}6364JNIEXPORT void JNICALL65JNU_ThrowOutOfMemoryError(JNIEnv *env, const char *msg)66{67JNU_ThrowByName(env, "java/lang/OutOfMemoryError", msg);68}6970JNIEXPORT void JNICALL71JNU_ThrowIllegalArgumentException(JNIEnv *env, const char *msg)72{73JNU_ThrowByName(env, "java/lang/IllegalArgumentException", msg);74}7576JNIEXPORT void JNICALL77JNU_ThrowIllegalAccessError(JNIEnv *env, const char *msg)78{79JNU_ThrowByName(env, "java/lang/IllegalAccessError", msg);80}8182JNIEXPORT void JNICALL83JNU_ThrowIllegalAccessException(JNIEnv *env, const char *msg)84{85JNU_ThrowByName(env, "java/lang/IllegalAccessException", msg);86}8788JNIEXPORT void JNICALL89JNU_ThrowInternalError(JNIEnv *env, const char *msg)90{91JNU_ThrowByName(env, "java/lang/InternalError", msg);92}9394JNIEXPORT void JNICALL95JNU_ThrowNoSuchFieldException(JNIEnv *env, const char *msg)96{97JNU_ThrowByName(env, "java/lang/NoSuchFieldException", msg);98}99100JNIEXPORT void JNICALL101JNU_ThrowNoSuchMethodException(JNIEnv *env, const char *msg)102{103JNU_ThrowByName(env, "java/lang/NoSuchMethodException", msg);104}105106JNIEXPORT void JNICALL107JNU_ThrowClassNotFoundException(JNIEnv *env, const char *msg)108{109JNU_ThrowByName(env, "java/lang/ClassNotFoundException", msg);110}111112JNIEXPORT void JNICALL113JNU_ThrowNumberFormatException(JNIEnv *env, const char *msg)114{115JNU_ThrowByName(env, "java/lang/NumberFormatException", msg);116}117118JNIEXPORT void JNICALL119JNU_ThrowIOException(JNIEnv *env, const char *msg)120{121JNU_ThrowByName(env, "java/io/IOException", msg);122}123124JNIEXPORT void JNICALL125JNU_ThrowNoSuchFieldError(JNIEnv *env, const char *msg)126{127JNU_ThrowByName(env, "java/lang/NoSuchFieldError", msg);128}129130JNIEXPORT void JNICALL131JNU_ThrowNoSuchMethodError(JNIEnv *env, const char *msg)132{133JNU_ThrowByName(env, "java/lang/NoSuchMethodError", msg);134}135136JNIEXPORT void JNICALL137JNU_ThrowStringIndexOutOfBoundsException(JNIEnv *env, const char *msg)138{139JNU_ThrowByName(env, "java/lang/StringIndexOutOfBoundsException", msg);140}141142JNIEXPORT void JNICALL143JNU_ThrowInstantiationException(JNIEnv *env, const char *msg)144{145JNU_ThrowByName(env, "java/lang/InstantiationException", msg);146}147148149/*150* Throw an exception by name, using a given message and the string151* returned by getLastErrorString to construct the detail string.152*/153JNIEXPORT void JNICALL154JNU_ThrowByNameWithMessageAndLastError155(JNIEnv *env, const char *name, const char *message)156{157char buf[256];158size_t n = getLastErrorString(buf, sizeof(buf));159size_t messagelen = message == NULL ? 0 : strlen(message);160161if (n > 0) {162jstring s = JNU_NewStringPlatform(env, buf);163if (s != NULL) {164jobject x = NULL;165if (messagelen) {166jstring s2 = NULL;167size_t messageextlen = messagelen + 4;168char *str1 = (char *)malloc((messageextlen) * sizeof(char));169if (str1 == 0) {170JNU_ThrowOutOfMemoryError(env, 0);171return;172}173jio_snprintf(str1, messageextlen, " (%s)", message);174s2 = (*env)->NewStringUTF(env, str1);175free(str1);176JNU_CHECK_EXCEPTION(env);177if (s2 != NULL) {178jstring s3 = JNU_CallMethodByName(179env, NULL, s, "concat",180"(Ljava/lang/String;)Ljava/lang/String;",181s2).l;182(*env)->DeleteLocalRef(env, s2);183JNU_CHECK_EXCEPTION(env);184if (s3 != NULL) {185(*env)->DeleteLocalRef(env, s);186s = s3;187}188}189}190x = JNU_NewObjectByName(env, name, "(Ljava/lang/String;)V", s);191if (x != NULL) {192(*env)->Throw(env, x);193}194}195}196197if (!(*env)->ExceptionOccurred(env)) {198if (messagelen) {199JNU_ThrowByName(env, name, message);200} else {201JNU_ThrowByName(env, name, "no further information");202}203}204}205206/* Throw an exception by name, using the string returned by207* JVM_LastErrorString for the detail string. If the last-error208* string is NULL, use the given default detail string.209*/210JNIEXPORT void JNICALL211JNU_ThrowByNameWithLastError(JNIEnv *env, const char *name,212const char *defaultDetail)213{214char buf[256];215int n = JVM_GetLastErrorString(buf, sizeof(buf));216217if (n > 0) {218jstring s = JNU_NewStringPlatform(env, buf);219if (s != NULL) {220jobject x = JNU_NewObjectByName(env, name,221"(Ljava/lang/String;)V", s);222if (x != NULL) {223(*env)->Throw(env, x);224}225}226}227if (!(*env)->ExceptionOccurred(env)) {228JNU_ThrowByName(env, name, defaultDetail);229}230}231232/* Throw an IOException, using the last-error string for the detail233* string. If the last-error string is NULL, use the given default234* detail string.235*/236JNIEXPORT void JNICALL237JNU_ThrowIOExceptionWithLastError(JNIEnv *env, const char *defaultDetail)238{239JNU_ThrowByNameWithLastError(env, "java/io/IOException", defaultDetail);240}241242243JNIEXPORT jvalue JNICALL244JNU_CallStaticMethodByName(JNIEnv *env,245jboolean *hasException,246const char *class_name,247const char *name,248const char *signature,249...)250{251jclass clazz;252jmethodID mid;253va_list args;254jvalue result;255const char *p = signature;256257/* find out the return type */258while (*p && *p != ')')259p++;260p++;261262result.i = 0;263264if ((*env)->EnsureLocalCapacity(env, 3) < 0)265goto done2;266267clazz = (*env)->FindClass(env, class_name);268if (clazz == 0)269goto done2;270mid = (*env)->GetStaticMethodID(env, clazz, name, signature);271if (mid == 0)272goto done1;273va_start(args, signature);274switch (*p) {275case 'V':276(*env)->CallStaticVoidMethodV(env, clazz, mid, args);277break;278case '[':279case 'L':280result.l = (*env)->CallStaticObjectMethodV(env, clazz, mid, args);281break;282case 'Z':283result.z = (*env)->CallStaticBooleanMethodV(env, clazz, mid, args);284break;285case 'B':286result.b = (*env)->CallStaticByteMethodV(env, clazz, mid, args);287break;288case 'C':289result.c = (*env)->CallStaticCharMethodV(env, clazz, mid, args);290break;291case 'S':292result.s = (*env)->CallStaticShortMethodV(env, clazz, mid, args);293break;294case 'I':295result.i = (*env)->CallStaticIntMethodV(env, clazz, mid, args);296break;297case 'J':298result.j = (*env)->CallStaticLongMethodV(env, clazz, mid, args);299break;300case 'F':301result.f = (*env)->CallStaticFloatMethodV(env, clazz, mid, args);302break;303case 'D':304result.d = (*env)->CallStaticDoubleMethodV(env, clazz, mid, args);305break;306default:307(*env)->FatalError(env, "JNU_CallStaticMethodByName: illegal signature");308}309va_end(args);310311done1:312(*env)->DeleteLocalRef(env, clazz);313done2:314if (hasException) {315*hasException = (*env)->ExceptionCheck(env);316}317return result;318}319320JNIEXPORT jvalue JNICALL321JNU_CallMethodByName(JNIEnv *env,322jboolean *hasException,323jobject obj,324const char *name,325const char *signature,326...)327{328jvalue result;329va_list args;330331va_start(args, signature);332result = JNU_CallMethodByNameV(env, hasException, obj, name, signature,333args);334va_end(args);335336return result;337}338339340JNIEXPORT jvalue JNICALL341JNU_CallMethodByNameV(JNIEnv *env,342jboolean *hasException,343jobject obj,344const char *name,345const char *signature,346va_list args)347{348jclass clazz;349jmethodID mid;350jvalue result;351const char *p = signature;352353/* find out the return type */354while (*p && *p != ')')355p++;356p++;357358result.i = 0;359360if ((*env)->EnsureLocalCapacity(env, 3) < 0)361goto done2;362363clazz = (*env)->GetObjectClass(env, obj);364mid = (*env)->GetMethodID(env, clazz, name, signature);365if (mid == 0)366goto done1;367368switch (*p) {369case 'V':370(*env)->CallVoidMethodV(env, obj, mid, args);371break;372case '[':373case 'L':374result.l = (*env)->CallObjectMethodV(env, obj, mid, args);375break;376case 'Z':377result.z = (*env)->CallBooleanMethodV(env, obj, mid, args);378break;379case 'B':380result.b = (*env)->CallByteMethodV(env, obj, mid, args);381break;382case 'C':383result.c = (*env)->CallCharMethodV(env, obj, mid, args);384break;385case 'S':386result.s = (*env)->CallShortMethodV(env, obj, mid, args);387break;388case 'I':389result.i = (*env)->CallIntMethodV(env, obj, mid, args);390break;391case 'J':392result.j = (*env)->CallLongMethodV(env, obj, mid, args);393break;394case 'F':395result.f = (*env)->CallFloatMethodV(env, obj, mid, args);396break;397case 'D':398result.d = (*env)->CallDoubleMethodV(env, obj, mid, args);399break;400default:401(*env)->FatalError(env, "JNU_CallMethodByNameV: illegal signature");402}403done1:404(*env)->DeleteLocalRef(env, clazz);405done2:406if (hasException) {407*hasException = (*env)->ExceptionCheck(env);408}409return result;410}411412JNIEXPORT jobject JNICALL413JNU_NewObjectByName(JNIEnv *env, const char *class_name,414const char *constructor_sig, ...)415{416jobject obj = NULL;417418jclass cls = 0;419jmethodID cls_initMID;420va_list args;421422if ((*env)->EnsureLocalCapacity(env, 2) < 0)423goto done;424425cls = (*env)->FindClass(env, class_name);426if (cls == 0) {427goto done;428}429cls_initMID = (*env)->GetMethodID(env, cls,430"<init>", constructor_sig);431if (cls_initMID == NULL) {432goto done;433}434va_start(args, constructor_sig);435obj = (*env)->NewObjectV(env, cls, cls_initMID, args);436va_end(args);437438done:439(*env)->DeleteLocalRef(env, cls);440return obj;441}442443/* Optimized for char set ISO_8559_1 */444static jstring445newString8859_1(JNIEnv *env, const char *str)446{447int len = (int)strlen(str);448jchar buf[512];449jchar *str1;450jstring result;451int i;452453if (len > 512) {454str1 = (jchar *)malloc(len * sizeof(jchar));455if (str1 == 0) {456JNU_ThrowOutOfMemoryError(env, 0);457return 0;458}459} else460str1 = buf;461462for (i=0;i<len;i++)463str1[i] = (unsigned char)str[i];464result = (*env)->NewString(env, str1, len);465if (str1 != buf)466free(str1);467return result;468}469470static const char*471getString8859_1Chars(JNIEnv *env, jstring jstr)472{473int i;474char *result;475jint len = (*env)->GetStringLength(env, jstr);476const jchar *str = (*env)->GetStringCritical(env, jstr, 0);477if (str == 0) {478return 0;479}480481result = MALLOC_MIN4(len);482if (result == 0) {483(*env)->ReleaseStringCritical(env, jstr, str);484JNU_ThrowOutOfMemoryError(env, 0);485return 0;486}487488for (i=0; i<len; i++) {489jchar unicode = str[i];490if (unicode <= 0x00ff)491result[i] = (char)unicode;492else493result[i] = '?';494}495496result[len] = 0;497(*env)->ReleaseStringCritical(env, jstr, str);498return result;499}500501502/* Optimized for char set ISO646-US (us-ascii) */503static jstring504newString646_US(JNIEnv *env, const char *str)505{506int len = strlen(str);507jchar buf[512];508jchar *str1;509jstring result;510int i;511512if (len > 512) {513str1 = (jchar *)malloc(len * sizeof(jchar));514if (str1 == 0) {515JNU_ThrowOutOfMemoryError(env, 0);516return 0;517}518} else519str1 = buf;520521for (i=0; i<len; i++) {522unsigned char c = (unsigned char)str[i];523if (c <= 0x7f)524str1[i] = c;525else526str1[i] = '?';527}528529result = (*env)->NewString(env, str1, len);530if (str1 != buf)531free(str1);532return result;533}534535static const char*536getString646_USChars(JNIEnv *env, jstring jstr)537{538int i;539char *result;540jint len = (*env)->GetStringLength(env, jstr);541const jchar *str = (*env)->GetStringCritical(env, jstr, 0);542if (str == 0) {543return 0;544}545546result = MALLOC_MIN4(len);547if (result == 0) {548(*env)->ReleaseStringCritical(env, jstr, str);549JNU_ThrowOutOfMemoryError(env, 0);550return 0;551}552553for (i=0; i<len; i++) {554jchar unicode = str[i];555if (unicode <= 0x007f )556result[i] = (char)unicode;557else558result[i] = '?';559}560561result[len] = 0;562(*env)->ReleaseStringCritical(env, jstr, str);563return result;564}565566/* enumeration of c1 row from Cp1252 */567static int cp1252c1chars[32] = {5680x20AC,0xFFFD,0x201A,0x0192,0x201E,0x2026,0x2020,0x2021,5690x02C6,0x2030,0x0160,0x2039,0x0152,0xFFFD,0x017D,0xFFFD,5700xFFFD,0x2018,0x2019,0x201C,0x201D,0x2022,0x2013,0x2014,5710x02Dc,0x2122,0x0161,0x203A,0x0153,0xFFFD,0x017E,0x0178572};573574/* Optimized for char set Cp1252 */575static jstring576newStringCp1252(JNIEnv *env, const char *str)577{578int len = (int) strlen(str);579jchar buf[512];580jchar *str1;581jstring result;582int i;583if (len > 512) {584str1 = (jchar *)malloc(len * sizeof(jchar));585if (str1 == 0) {586JNU_ThrowOutOfMemoryError(env, 0);587return 0;588}589} else590str1 = buf;591592for (i=0; i<len; i++) {593unsigned char c = (unsigned char)str[i];594if ((c >= 0x80) && (c <= 0x9f))595str1[i] = cp1252c1chars[c-128];596else597str1[i] = c;598}599600result = (*env)->NewString(env, str1, len);601if (str1 != buf)602free(str1);603return result;604}605606static const char*607getStringCp1252Chars(JNIEnv *env, jstring jstr)608{609int i;610char *result;611jint len = (*env)->GetStringLength(env, jstr);612const jchar *str = (*env)->GetStringCritical(env, jstr, 0);613if (str == 0) {614return 0;615}616617result = MALLOC_MIN4(len);618if (result == 0) {619(*env)->ReleaseStringCritical(env, jstr, str);620JNU_ThrowOutOfMemoryError(env, 0);621return 0;622}623624for (i=0; i<len; i++) {625jchar c = str[i];626if (c < 256)627result[i] = (char)c;628else switch(c) {629case 0x20AC: result[i] = (char)0x80; break;630case 0x201A: result[i] = (char)0x82; break;631case 0x0192: result[i] = (char)0x83; break;632case 0x201E: result[i] = (char)0x84; break;633case 0x2026: result[i] = (char)0x85; break;634case 0x2020: result[i] = (char)0x86; break;635case 0x2021: result[i] = (char)0x87; break;636case 0x02C6: result[i] = (char)0x88; break;637case 0x2030: result[i] = (char)0x89; break;638case 0x0160: result[i] = (char)0x8A; break;639case 0x2039: result[i] = (char)0x8B; break;640case 0x0152: result[i] = (char)0x8C; break;641case 0x017D: result[i] = (char)0x8E; break;642case 0x2018: result[i] = (char)0x91; break;643case 0x2019: result[i] = (char)0x92; break;644case 0x201C: result[i] = (char)0x93; break;645case 0x201D: result[i] = (char)0x94; break;646case 0x2022: result[i] = (char)0x95; break;647case 0x2013: result[i] = (char)0x96; break;648case 0x2014: result[i] = (char)0x97; break;649case 0x02DC: result[i] = (char)0x98; break;650case 0x2122: result[i] = (char)0x99; break;651case 0x0161: result[i] = (char)0x9A; break;652case 0x203A: result[i] = (char)0x9B; break;653case 0x0153: result[i] = (char)0x9C; break;654case 0x017E: result[i] = (char)0x9E; break;655case 0x0178: result[i] = (char)0x9F; break;656default: result[i] = '?'; break;657}658}659660result[len] = 0;661(*env)->ReleaseStringCritical(env, jstr, str);662return result;663}664665static int fastEncoding = NO_ENCODING_YET;666static jstring jnuEncoding = NULL;667668/* Cached method IDs */669static jmethodID String_init_ID; /* String(byte[], enc) */670static jmethodID String_getBytes_ID; /* String.getBytes(enc) */671672int getFastEncoding() {673return fastEncoding;674}675676/* Initialize the fast encoding. If the "sun.jnu.encoding" property677* has not yet been set, we leave fastEncoding == NO_ENCODING_YET.678*/679void680initializeEncoding(JNIEnv *env)681{682jstring propname = 0;683jstring enc = 0;684jclass strClazz = NULL;685686if ((*env)->EnsureLocalCapacity(env, 3) < 0)687return;688689strClazz = JNU_ClassString(env);690CHECK_NULL(strClazz);691692propname = (*env)->NewStringUTF(env, "sun.jnu.encoding");693if (propname) {694jboolean exc;695enc = JNU_CallStaticMethodByName696(env,697&exc,698"java/lang/System",699"getProperty",700"(Ljava/lang/String;)Ljava/lang/String;",701propname).l;702if (!exc) {703if (enc) {704const char* encname = (*env)->GetStringUTFChars(env, enc, 0);705if (encname) {706/*707* On Solaris with nl_langinfo() called in GetJavaProperties():708*709* locale undefined -> NULL -> hardcoded default710* "C" locale -> "" -> hardcoded default (on 2.6)711* "C" locale -> "ISO646-US" (on Sol 7/8)712* "en_US" locale -> "ISO8859-1"713* "en_GB" locale -> "ISO8859-1" (on Sol 7/8)714* "en_UK" locale -> "ISO8859-1" (on 2.6)715*/716if ((strcmp(encname, "8859_1") == 0) ||717(strcmp(encname, "ISO8859-1") == 0) ||718(strcmp(encname, "ISO8859_1") == 0))719fastEncoding = FAST_8859_1;720else if (strcmp(encname, "ISO646-US") == 0)721fastEncoding = FAST_646_US;722else if (strcmp(encname, "Cp1252") == 0 ||723/* This is a temporary fix until we move */724/* to wide character versions of all Windows */725/* calls. */726strcmp(encname, "utf-16le") == 0)727fastEncoding = FAST_CP1252;728else {729fastEncoding = NO_FAST_ENCODING;730jnuEncoding = (jstring)(*env)->NewGlobalRef(env, enc);731}732(*env)->ReleaseStringUTFChars(env, enc, encname);733}734}735} else {736(*env)->ExceptionClear(env);737}738} else {739(*env)->ExceptionClear(env);740}741(*env)->DeleteLocalRef(env, propname);742(*env)->DeleteLocalRef(env, enc);743744/* Initialize method-id cache */745String_getBytes_ID = (*env)->GetMethodID(env, strClazz,746"getBytes", "(Ljava/lang/String;)[B");747CHECK_NULL(String_getBytes_ID);748String_init_ID = (*env)->GetMethodID(env, strClazz,749"<init>", "([BLjava/lang/String;)V");750CHECK_NULL(String_init_ID);751}752753static jboolean isJNUEncodingSupported = JNI_FALSE;754static jboolean jnuEncodingSupported(JNIEnv *env) {755jboolean exe;756if (isJNUEncodingSupported == JNI_TRUE) {757return JNI_TRUE;758}759isJNUEncodingSupported = (jboolean) JNU_CallStaticMethodByName (760env, &exe,761"java/nio/charset/Charset",762"isSupported",763"(Ljava/lang/String;)Z",764jnuEncoding).z;765return isJNUEncodingSupported;766}767768769JNIEXPORT jstring770NewStringPlatform(JNIEnv *env, const char *str)771{772return JNU_NewStringPlatform(env, str);773}774775JNIEXPORT jstring JNICALL776JNU_NewStringPlatform(JNIEnv *env, const char *str)777{778jstring result;779result = nativeNewStringPlatform(env, str);780if (result == NULL) {781jbyteArray hab = 0;782int len;783784if (fastEncoding == NO_ENCODING_YET) {785initializeEncoding(env);786JNU_CHECK_EXCEPTION_RETURN(env, NULL);787}788789if ((fastEncoding == FAST_8859_1) || (fastEncoding == NO_ENCODING_YET))790return newString8859_1(env, str);791if (fastEncoding == FAST_646_US)792return newString646_US(env, str);793if (fastEncoding == FAST_CP1252)794return newStringCp1252(env, str);795796if ((*env)->EnsureLocalCapacity(env, 2) < 0)797return NULL;798799len = (int)strlen(str);800hab = (*env)->NewByteArray(env, len);801if (hab != 0) {802jclass strClazz = JNU_ClassString(env);803CHECK_NULL_RETURN(strClazz, 0);804(*env)->SetByteArrayRegion(env, hab, 0, len, (jbyte *)str);805if (jnuEncodingSupported(env)) {806result = (*env)->NewObject(env, strClazz,807String_init_ID, hab, jnuEncoding);808} else {809/*If the encoding specified in sun.jnu.encoding is not endorsed810by "Charset.isSupported" we have to fall back to use String(byte[])811explicitly here without specifying the encoding name, in which the812StringCoding class will pickup the iso-8859-1 as the fallback813converter for us.814*/815jmethodID mid = (*env)->GetMethodID(env, strClazz,816"<init>", "([B)V");817if (mid != NULL) {818result = (*env)->NewObject(env, strClazz, mid, hab);819}820}821(*env)->DeleteLocalRef(env, hab);822return result;823}824}825return NULL;826}827828JNIEXPORT const char *829GetStringPlatformChars(JNIEnv *env, jstring jstr, jboolean *isCopy)830{831return JNU_GetStringPlatformChars(env, jstr, isCopy);832}833834JNIEXPORT const char * JNICALL835JNU_GetStringPlatformChars(JNIEnv *env, jstring jstr, jboolean *isCopy)836{837char *result = nativeGetStringPlatformChars(env, jstr, isCopy);838if (result == NULL) {839840jbyteArray hab = 0;841842if (isCopy)843*isCopy = JNI_TRUE;844845if (fastEncoding == NO_ENCODING_YET) {846initializeEncoding(env);847JNU_CHECK_EXCEPTION_RETURN(env, 0);848}849850if ((fastEncoding == FAST_8859_1) || (fastEncoding == NO_ENCODING_YET))851return getString8859_1Chars(env, jstr);852if (fastEncoding == FAST_646_US)853return getString646_USChars(env, jstr);854if (fastEncoding == FAST_CP1252)855return getStringCp1252Chars(env, jstr);856857if ((*env)->EnsureLocalCapacity(env, 2) < 0)858return 0;859860if (jnuEncodingSupported(env)) {861hab = (*env)->CallObjectMethod(env, jstr, String_getBytes_ID, jnuEncoding);862} else {863jmethodID mid;864jclass strClazz = JNU_ClassString(env);865CHECK_NULL_RETURN(strClazz, 0);866mid = (*env)->GetMethodID(env, strClazz,867"getBytes", "()[B");868if (mid != NULL) {869hab = (*env)->CallObjectMethod(env, jstr, mid);870}871}872873if (!(*env)->ExceptionCheck(env)) {874jint len = (*env)->GetArrayLength(env, hab);875result = MALLOC_MIN4(len);876if (result == 0) {877JNU_ThrowOutOfMemoryError(env, 0);878(*env)->DeleteLocalRef(env, hab);879return 0;880}881(*env)->GetByteArrayRegion(env, hab, 0, len, (jbyte *)result);882result[len] = 0; /* NULL-terminate */883}884885(*env)->DeleteLocalRef(env, hab);886}887return result;888}889890JNIEXPORT void JNICALL891JNU_ReleaseStringPlatformChars(JNIEnv *env, jstring jstr, const char *str)892{893free((void *)str);894}895896/*897* Export the platform dependent path canonicalization so that898* VM can find it when loading system classes.899*900*/901extern int canonicalize(char *path, const char *out, int len);902903JNIEXPORT int904Canonicalize(JNIEnv *env, char *orig, char *out, int len)905{906/* canonicalize an already natived path */907return canonicalize(orig, out, len);908}909910JNIEXPORT jclass JNICALL911JNU_ClassString(JNIEnv *env)912{913static jclass cls = 0;914if (cls == 0) {915jclass c;916if ((*env)->EnsureLocalCapacity(env, 1) < 0)917return 0;918c = (*env)->FindClass(env, "java/lang/String");919CHECK_NULL_RETURN(c, NULL);920cls = (*env)->NewGlobalRef(env, c);921(*env)->DeleteLocalRef(env, c);922}923return cls;924}925926JNIEXPORT jclass JNICALL927JNU_ClassClass(JNIEnv *env)928{929static jclass cls = 0;930if (cls == 0) {931jclass c;932if ((*env)->EnsureLocalCapacity(env, 1) < 0)933return 0;934c = (*env)->FindClass(env, "java/lang/Class");935CHECK_NULL_RETURN(c, NULL);936cls = (*env)->NewGlobalRef(env, c);937(*env)->DeleteLocalRef(env, c);938}939return cls;940}941942JNIEXPORT jclass JNICALL943JNU_ClassObject(JNIEnv *env)944{945static jclass cls = 0;946if (cls == 0) {947jclass c;948if ((*env)->EnsureLocalCapacity(env, 1) < 0)949return 0;950c = (*env)->FindClass(env, "java/lang/Object");951CHECK_NULL_RETURN(c, NULL);952cls = (*env)->NewGlobalRef(env, c);953(*env)->DeleteLocalRef(env, c);954}955return cls;956}957958JNIEXPORT jclass JNICALL959JNU_ClassThrowable(JNIEnv *env)960{961static jclass cls = 0;962if (cls == 0) {963jclass c;964if ((*env)->EnsureLocalCapacity(env, 1) < 0)965return 0;966c = (*env)->FindClass(env, "java/lang/Throwable");967CHECK_NULL_RETURN(c, NULL);968cls = (*env)->NewGlobalRef(env, c);969(*env)->DeleteLocalRef(env, c);970}971return cls;972}973974JNIEXPORT jint JNICALL975JNU_CopyObjectArray(JNIEnv *env, jobjectArray dst, jobjectArray src,976jint count)977{978int i;979if ((*env)->EnsureLocalCapacity(env, 1) < 0)980return -1;981for (i=0; i<count; i++) {982jstring p = (*env)->GetObjectArrayElement(env, src, i);983(*env)->SetObjectArrayElement(env, dst, i, p);984(*env)->DeleteLocalRef(env, p);985}986return 0;987}988989JNIEXPORT void * JNICALL990JNU_GetEnv(JavaVM *vm, jint version)991{992void *env;993(*vm)->GetEnv(vm, &env, version);994return env;995}996997JNIEXPORT jint JNICALL998JNU_IsInstanceOfByName(JNIEnv *env, jobject object, char* classname)999{1000jclass cls;1001if ((*env)->EnsureLocalCapacity(env, 1) < 0)1002return JNI_ERR;1003cls = (*env)->FindClass(env, classname);1004if (cls != NULL) {1005jint result = (*env)->IsInstanceOf(env, object, cls);1006(*env)->DeleteLocalRef(env, cls);1007return result;1008}1009return JNI_ERR;1010}10111012JNIEXPORT jboolean JNICALL1013JNU_Equals(JNIEnv *env, jobject object1, jobject object2)1014{1015static jmethodID mid = NULL;1016if (mid == NULL) {1017jclass objClazz = JNU_ClassObject(env);1018CHECK_NULL_RETURN(objClazz, JNI_FALSE);1019mid = (*env)->GetMethodID(env, objClazz, "equals",1020"(Ljava/lang/Object;)Z");1021CHECK_NULL_RETURN(mid, JNI_FALSE);1022}1023return (*env)->CallBooleanMethod(env, object1, mid, object2);1024}102510261027/************************************************************************1028* Thread calls1029*/10301031static jmethodID Object_waitMID;1032static jmethodID Object_notifyMID;1033static jmethodID Object_notifyAllMID;10341035JNIEXPORT void JNICALL1036JNU_MonitorWait(JNIEnv *env, jobject object, jlong timeout)1037{1038if (object == NULL) {1039JNU_ThrowNullPointerException(env, "JNU_MonitorWait argument");1040return;1041}1042if (Object_waitMID == NULL) {1043jclass cls = JNU_ClassObject(env);1044if (cls == NULL) {1045return;1046}1047Object_waitMID = (*env)->GetMethodID(env, cls, "wait", "(J)V");1048if (Object_waitMID == NULL) {1049return;1050}1051}1052(*env)->CallVoidMethod(env, object, Object_waitMID, timeout);1053}10541055JNIEXPORT void JNICALL1056JNU_Notify(JNIEnv *env, jobject object)1057{1058if (object == NULL) {1059JNU_ThrowNullPointerException(env, "JNU_Notify argument");1060return;1061}1062if (Object_notifyMID == NULL) {1063jclass cls = JNU_ClassObject(env);1064if (cls == NULL) {1065return;1066}1067Object_notifyMID = (*env)->GetMethodID(env, cls, "notify", "()V");1068if (Object_notifyMID == NULL) {1069return;1070}1071}1072(*env)->CallVoidMethod(env, object, Object_notifyMID);1073}10741075JNIEXPORT void JNICALL1076JNU_NotifyAll(JNIEnv *env, jobject object)1077{1078if (object == NULL) {1079JNU_ThrowNullPointerException(env, "JNU_NotifyAll argument");1080return;1081}1082if (Object_notifyAllMID == NULL) {1083jclass cls = JNU_ClassObject(env);1084if (cls == NULL) {1085return;1086}1087Object_notifyAllMID = (*env)->GetMethodID(env, cls,"notifyAll", "()V");1088if (Object_notifyAllMID == NULL) {1089return;1090}1091}1092(*env)->CallVoidMethod(env, object, Object_notifyAllMID);1093}109410951096/************************************************************************1097* Debugging utilities1098*/10991100JNIEXPORT void JNICALL1101JNU_PrintString(JNIEnv *env, char *hdr, jstring string)1102{1103if (string == NULL) {1104fprintf(stderr, "%s: is NULL\n", hdr);1105} else {1106const char *stringPtr = JNU_GetStringPlatformChars(env, string, 0);1107if (stringPtr == 0)1108return;1109fprintf(stderr, "%s: %s\n", hdr, stringPtr);1110JNU_ReleaseStringPlatformChars(env, string, stringPtr);1111}1112}11131114JNIEXPORT void JNICALL1115JNU_PrintClass(JNIEnv *env, char* hdr, jobject object)1116{1117if (object == NULL) {1118fprintf(stderr, "%s: object is NULL\n", hdr);1119return;1120} else {1121jclass cls = (*env)->GetObjectClass(env, object);1122jstring clsName = JNU_ToString(env, cls);1123if (clsName == NULL) {1124JNU_PrintString(env, hdr, clsName);1125}1126(*env)->DeleteLocalRef(env, cls);1127(*env)->DeleteLocalRef(env, clsName);1128}1129}11301131JNIEXPORT jstring JNICALL1132JNU_ToString(JNIEnv *env, jobject object)1133{1134if (object == NULL) {1135return (*env)->NewStringUTF(env, "NULL");1136} else {1137return (jstring)JNU_CallMethodByName(env,1138NULL,1139object,1140"toString",1141"()Ljava/lang/String;").l;1142}1143}11441145JNIEXPORT jvalue JNICALL1146JNU_GetFieldByName(JNIEnv *env,1147jboolean *hasException,1148jobject obj,1149const char *name,1150const char *signature)1151{1152jclass cls;1153jfieldID fid;1154jvalue result;11551156result.i = 0;11571158if ((*env)->EnsureLocalCapacity(env, 3) < 0)1159goto done2;11601161cls = (*env)->GetObjectClass(env, obj);1162fid = (*env)->GetFieldID(env, cls, name, signature);1163if (fid == 0)1164goto done1;11651166switch (*signature) {1167case '[':1168case 'L':1169result.l = (*env)->GetObjectField(env, obj, fid);1170break;1171case 'Z':1172result.z = (*env)->GetBooleanField(env, obj, fid);1173break;1174case 'B':1175result.b = (*env)->GetByteField(env, obj, fid);1176break;1177case 'C':1178result.c = (*env)->GetCharField(env, obj, fid);1179break;1180case 'S':1181result.s = (*env)->GetShortField(env, obj, fid);1182break;1183case 'I':1184result.i = (*env)->GetIntField(env, obj, fid);1185break;1186case 'J':1187result.j = (*env)->GetLongField(env, obj, fid);1188break;1189case 'F':1190result.f = (*env)->GetFloatField(env, obj, fid);1191break;1192case 'D':1193result.d = (*env)->GetDoubleField(env, obj, fid);1194break;11951196default:1197(*env)->FatalError(env, "JNU_GetFieldByName: illegal signature");1198}11991200done1:1201(*env)->DeleteLocalRef(env, cls);1202done2:1203if (hasException) {1204*hasException = (*env)->ExceptionCheck(env);1205}1206return result;1207}12081209JNIEXPORT void JNICALL1210JNU_SetFieldByName(JNIEnv *env,1211jboolean *hasException,1212jobject obj,1213const char *name,1214const char *signature,1215...)1216{1217jclass cls;1218jfieldID fid;1219va_list args;12201221if ((*env)->EnsureLocalCapacity(env, 3) < 0)1222goto done2;12231224cls = (*env)->GetObjectClass(env, obj);1225fid = (*env)->GetFieldID(env, cls, name, signature);1226if (fid == 0)1227goto done1;12281229va_start(args, signature);1230switch (*signature) {1231case '[':1232case 'L':1233(*env)->SetObjectField(env, obj, fid, va_arg(args, jobject));1234break;1235case 'Z':1236(*env)->SetBooleanField(env, obj, fid, (jboolean)va_arg(args, int));1237break;1238case 'B':1239(*env)->SetByteField(env, obj, fid, (jbyte)va_arg(args, int));1240break;1241case 'C':1242(*env)->SetCharField(env, obj, fid, (jchar)va_arg(args, int));1243break;1244case 'S':1245(*env)->SetShortField(env, obj, fid, (jshort)va_arg(args, int));1246break;1247case 'I':1248(*env)->SetIntField(env, obj, fid, va_arg(args, jint));1249break;1250case 'J':1251(*env)->SetLongField(env, obj, fid, va_arg(args, jlong));1252break;1253case 'F':1254(*env)->SetFloatField(env, obj, fid, (jfloat)va_arg(args, jdouble));1255break;1256case 'D':1257(*env)->SetDoubleField(env, obj, fid, va_arg(args, jdouble));1258break;12591260default:1261(*env)->FatalError(env, "JNU_SetFieldByName: illegal signature");1262}1263va_end(args);12641265done1:1266(*env)->DeleteLocalRef(env, cls);1267done2:1268if (hasException) {1269*hasException = (*env)->ExceptionCheck(env);1270}1271}12721273JNIEXPORT jvalue JNICALL1274JNU_GetStaticFieldByName(JNIEnv *env,1275jboolean *hasException,1276const char *classname,1277const char *name,1278const char *signature)1279{1280jclass cls;1281jfieldID fid;1282jvalue result;12831284result.i = 0;12851286if ((*env)->EnsureLocalCapacity(env, 3) < 0)1287goto done2;12881289cls = (*env)->FindClass(env, classname);1290if (cls == 0)1291goto done2;12921293fid = (*env)->GetStaticFieldID(env, cls, name, signature);1294if (fid == 0)1295goto done1;12961297switch (*signature) {1298case '[':1299case 'L':1300result.l = (*env)->GetStaticObjectField(env, cls, fid);1301break;1302case 'Z':1303result.z = (*env)->GetStaticBooleanField(env, cls, fid);1304break;1305case 'B':1306result.b = (*env)->GetStaticByteField(env, cls, fid);1307break;1308case 'C':1309result.c = (*env)->GetStaticCharField(env, cls, fid);1310break;1311case 'S':1312result.s = (*env)->GetStaticShortField(env, cls, fid);1313break;1314case 'I':1315result.i = (*env)->GetStaticIntField(env, cls, fid);1316break;1317case 'J':1318result.j = (*env)->GetStaticLongField(env, cls, fid);1319break;1320case 'F':1321result.f = (*env)->GetStaticFloatField(env, cls, fid);1322break;1323case 'D':1324result.d = (*env)->GetStaticDoubleField(env, cls, fid);1325break;13261327default:1328(*env)->FatalError(env, "JNU_GetStaticFieldByName: illegal signature");1329}13301331done1:1332(*env)->DeleteLocalRef(env, cls);1333done2:1334if (hasException) {1335*hasException = (*env)->ExceptionCheck(env);1336}1337return result;1338}13391340JNIEXPORT void JNICALL1341JNU_SetStaticFieldByName(JNIEnv *env,1342jboolean *hasException,1343const char *classname,1344const char *name,1345const char *signature,1346...)1347{1348jclass cls;1349jfieldID fid;1350va_list args;13511352if ((*env)->EnsureLocalCapacity(env, 3) < 0)1353goto done2;13541355cls = (*env)->FindClass(env, classname);1356if (cls == 0)1357goto done2;13581359fid = (*env)->GetStaticFieldID(env, cls, name, signature);1360if (fid == 0)1361goto done1;13621363va_start(args, signature);1364switch (*signature) {1365case '[':1366case 'L':1367(*env)->SetStaticObjectField(env, cls, fid, va_arg(args, jobject));1368break;1369case 'Z':1370(*env)->SetStaticBooleanField(env, cls, fid, (jboolean)va_arg(args, int));1371break;1372case 'B':1373(*env)->SetStaticByteField(env, cls, fid, (jbyte)va_arg(args, int));1374break;1375case 'C':1376(*env)->SetStaticCharField(env, cls, fid, (jchar)va_arg(args, int));1377break;1378case 'S':1379(*env)->SetStaticShortField(env, cls, fid, (jshort)va_arg(args, int));1380break;1381case 'I':1382(*env)->SetStaticIntField(env, cls, fid, va_arg(args, jint));1383break;1384case 'J':1385(*env)->SetStaticLongField(env, cls, fid, va_arg(args, jlong));1386break;1387case 'F':1388(*env)->SetStaticFloatField(env, cls, fid, (jfloat)va_arg(args, jdouble));1389break;1390case 'D':1391(*env)->SetStaticDoubleField(env, cls, fid, va_arg(args, jdouble));1392break;13931394default:1395(*env)->FatalError(env, "JNU_SetStaticFieldByName: illegal signature");1396}1397va_end(args);13981399done1:1400(*env)->DeleteLocalRef(env, cls);1401done2:1402if (hasException) {1403*hasException = (*env)->ExceptionCheck(env);1404}1405}140614071408