8201246: Export native function to set platform encoding
Reviewed-by: rriggs
Contributed-by: andrew_m_leonard@uk.ibm.com
/*
* Copyright (c) 1997, 2018, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* This code is free software; you can redistribute it and/or modify it
* under the terms of the GNU General Public License version 2 only, as
* published by the Free Software Foundation. Oracle designates this
* particular file as subject to the "Classpath" exception as provided
* by Oracle in the LICENSE file that accompanied this code.
*
* This code is distributed in the hope that it will be useful, but WITHOUT
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
* FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License
* version 2 for more details (a copy is included in the LICENSE file that
* accompanied this code).
*
* You should have received a copy of the GNU General Public License version
* 2 along with this work; if not, write to the Free Software Foundation,
* Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
*
* Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
* or visit www.oracle.com if you need additional information or have any
* questions.
*/
#include <stdlib.h>
#include <string.h>
#include "jvm.h"
#include "jni.h"
#include "jni_util.h"
#include "java_lang_String.h"
/* Due to a bug in the win32 C runtime library strings
* such as "z:" need to be appended with a "." so we
* must allocate at least 4 bytes to allow room for
* this expansion. See 4235353 for details.
*/
#define MALLOC_MIN4(len) ((char *)malloc((len) + 1 < 4 ? 4 : (len) + 1))
/**
* Throw a Java exception by name. Similar to SignalError.
*/
JNIEXPORT void JNICALL
JNU_ThrowByName(JNIEnv *env, const char *name, const char *msg)
{
jclass cls = (*env)->FindClass(env, name);
if (cls != 0) /* Otherwise an exception has already been thrown */
(*env)->ThrowNew(env, cls, msg);
}
/* JNU_Throw common exceptions */
JNIEXPORT void JNICALL
JNU_ThrowNullPointerException(JNIEnv *env, const char *msg)
{
JNU_ThrowByName(env, "java/lang/NullPointerException", msg);
}
JNIEXPORT void JNICALL
JNU_ThrowArrayIndexOutOfBoundsException(JNIEnv *env, const char *msg)
{
JNU_ThrowByName(env, "java/lang/ArrayIndexOutOfBoundsException", msg);
}
JNIEXPORT void JNICALL
JNU_ThrowOutOfMemoryError(JNIEnv *env, const char *msg)
{
JNU_ThrowByName(env, "java/lang/OutOfMemoryError", msg);
}
JNIEXPORT void JNICALL
JNU_ThrowIllegalArgumentException(JNIEnv *env, const char *msg)
{
JNU_ThrowByName(env, "java/lang/IllegalArgumentException", msg);
}
JNIEXPORT void JNICALL
JNU_ThrowIllegalAccessError(JNIEnv *env, const char *msg)
{
JNU_ThrowByName(env, "java/lang/IllegalAccessError", msg);
}
JNIEXPORT void JNICALL
JNU_ThrowIllegalAccessException(JNIEnv *env, const char *msg)
{
JNU_ThrowByName(env, "java/lang/IllegalAccessException", msg);
}
JNIEXPORT void JNICALL
JNU_ThrowInternalError(JNIEnv *env, const char *msg)
{
JNU_ThrowByName(env, "java/lang/InternalError", msg);
}
JNIEXPORT void JNICALL
JNU_ThrowNoSuchFieldException(JNIEnv *env, const char *msg)
{
JNU_ThrowByName(env, "java/lang/NoSuchFieldException", msg);
}
JNIEXPORT void JNICALL
JNU_ThrowNoSuchMethodException(JNIEnv *env, const char *msg)
{
JNU_ThrowByName(env, "java/lang/NoSuchMethodException", msg);
}
JNIEXPORT void JNICALL
JNU_ThrowClassNotFoundException(JNIEnv *env, const char *msg)
{
JNU_ThrowByName(env, "java/lang/ClassNotFoundException", msg);
}
JNIEXPORT void JNICALL
JNU_ThrowNumberFormatException(JNIEnv *env, const char *msg)
{
JNU_ThrowByName(env, "java/lang/NumberFormatException", msg);
}
JNIEXPORT void JNICALL
JNU_ThrowIOException(JNIEnv *env, const char *msg)
{
JNU_ThrowByName(env, "java/io/IOException", msg);
}
JNIEXPORT void JNICALL
JNU_ThrowNoSuchFieldError(JNIEnv *env, const char *msg)
{
JNU_ThrowByName(env, "java/lang/NoSuchFieldError", msg);
}
JNIEXPORT void JNICALL
JNU_ThrowNoSuchMethodError(JNIEnv *env, const char *msg)
{
JNU_ThrowByName(env, "java/lang/NoSuchMethodError", msg);
}
JNIEXPORT void JNICALL
JNU_ThrowStringIndexOutOfBoundsException(JNIEnv *env, const char *msg)
{
JNU_ThrowByName(env, "java/lang/StringIndexOutOfBoundsException", msg);
}
JNIEXPORT void JNICALL
JNU_ThrowInstantiationException(JNIEnv *env, const char *msg)
{
JNU_ThrowByName(env, "java/lang/InstantiationException", msg);
}
/*
* Throw an exception by name, using the string returned by
* getLastErrorString for the detail string. If the last-error
* string is NULL, use the given default detail string.
*/
JNIEXPORT void JNICALL
JNU_ThrowByNameWithLastError(JNIEnv *env, const char *name,
const char *defaultDetail)
{
char buf[256];
size_t n = getLastErrorString(buf, sizeof(buf));
if (n > 0) {
jstring s = JNU_NewStringPlatform(env, buf);
if (s != NULL) {
jobject x = JNU_NewObjectByName(env, name,
"(Ljava/lang/String;)V", s);
if (x != NULL) {
(*env)->Throw(env, x);
}
}
}
if (!(*env)->ExceptionOccurred(env)) {
JNU_ThrowByName(env, name, defaultDetail);
}
}
/*
* Throw an exception by name, using a given message and the string
* returned by getLastErrorString to construct the detail string.
*/
JNIEXPORT void JNICALL
JNU_ThrowByNameWithMessageAndLastError
(JNIEnv *env, const char *name, const char *message)
{
char buf[256];
size_t n = getLastErrorString(buf, sizeof(buf));
size_t messagelen = message == NULL ? 0 : strlen(message);
if (n > 0) {
jstring s = JNU_NewStringPlatform(env, buf);
if (s != NULL) {
jobject x = NULL;
if (messagelen) {
jstring s2 = NULL;
size_t messageextlen = messagelen + 4;
char *str1 = (char *)malloc((messageextlen) * sizeof(char));
if (str1 == 0) {
JNU_ThrowOutOfMemoryError(env, 0);
return;
}
jio_snprintf(str1, messageextlen, " (%s)", message);
s2 = (*env)->NewStringUTF(env, str1);
free(str1);
JNU_CHECK_EXCEPTION(env);
if (s2 != NULL) {
jstring s3 = JNU_CallMethodByName(
env, NULL, s, "concat",
"(Ljava/lang/String;)Ljava/lang/String;",
s2).l;
(*env)->DeleteLocalRef(env, s2);
JNU_CHECK_EXCEPTION(env);
if (s3 != NULL) {
(*env)->DeleteLocalRef(env, s);
s = s3;
}
}
}
x = JNU_NewObjectByName(env, name, "(Ljava/lang/String;)V", s);
if (x != NULL) {
(*env)->Throw(env, x);
}
}
}
if (!(*env)->ExceptionOccurred(env)) {
if (messagelen) {
JNU_ThrowByName(env, name, message);
} else {
JNU_ThrowByName(env, name, "no further information");
}
}
}
/*
* Convenience method.
* Call JNU_ThrowByNameWithLastError for java.io.IOException.
*/
JNIEXPORT void JNICALL
JNU_ThrowIOExceptionWithLastError(JNIEnv *env, const char *defaultDetail)
{
JNU_ThrowByNameWithLastError(env, "java/io/IOException", defaultDetail);
}
JNIEXPORT jvalue JNICALL
JNU_CallStaticMethodByName(JNIEnv *env,
jboolean *hasException,
const char *class_name,
const char *name,
const char *signature,
...)
{
jclass clazz;
jmethodID mid;
va_list args;
jvalue result;
const char *p = signature;
/* find out the return type */
while (*p && *p != ')')
p++;
p++;
result.i = 0;
if ((*env)->EnsureLocalCapacity(env, 3) < 0)
goto done2;
clazz = (*env)->FindClass(env, class_name);
if (clazz == 0)
goto done2;
mid = (*env)->GetStaticMethodID(env, clazz, name, signature);
if (mid == 0)
goto done1;
va_start(args, signature);
switch (*p) {
case 'V':
(*env)->CallStaticVoidMethodV(env, clazz, mid, args);
break;
case '[':
case 'L':
result.l = (*env)->CallStaticObjectMethodV(env, clazz, mid, args);
break;
case 'Z':
result.z = (*env)->CallStaticBooleanMethodV(env, clazz, mid, args);
break;
case 'B':
result.b = (*env)->CallStaticByteMethodV(env, clazz, mid, args);
break;
case 'C':
result.c = (*env)->CallStaticCharMethodV(env, clazz, mid, args);
break;
case 'S':
result.s = (*env)->CallStaticShortMethodV(env, clazz, mid, args);
break;
case 'I':
result.i = (*env)->CallStaticIntMethodV(env, clazz, mid, args);
break;
case 'J':
result.j = (*env)->CallStaticLongMethodV(env, clazz, mid, args);
break;
case 'F':
result.f = (*env)->CallStaticFloatMethodV(env, clazz, mid, args);
break;
case 'D':
result.d = (*env)->CallStaticDoubleMethodV(env, clazz, mid, args);
break;
default:
(*env)->FatalError(env, "JNU_CallStaticMethodByName: illegal signature");
}
va_end(args);
done1:
(*env)->DeleteLocalRef(env, clazz);
done2:
if (hasException) {
*hasException = (*env)->ExceptionCheck(env);
}
return result;
}
JNIEXPORT jvalue JNICALL
JNU_CallMethodByName(JNIEnv *env,
jboolean *hasException,
jobject obj,
const char *name,
const char *signature,
...)
{
jvalue result;
va_list args;
va_start(args, signature);
result = JNU_CallMethodByNameV(env, hasException, obj, name, signature,
args);
va_end(args);
return result;
}
JNIEXPORT jvalue JNICALL
JNU_CallMethodByNameV(JNIEnv *env,
jboolean *hasException,
jobject obj,
const char *name,
const char *signature,
va_list args)
{
jclass clazz;
jmethodID mid;
jvalue result;
const char *p = signature;
/* find out the return type */
while (*p && *p != ')')
p++;
p++;
result.i = 0;
if ((*env)->EnsureLocalCapacity(env, 3) < 0)
goto done2;
clazz = (*env)->GetObjectClass(env, obj);
mid = (*env)->GetMethodID(env, clazz, name, signature);
if (mid == 0)
goto done1;
switch (*p) {
case 'V':
(*env)->CallVoidMethodV(env, obj, mid, args);
break;
case '[':
case 'L':
result.l = (*env)->CallObjectMethodV(env, obj, mid, args);
break;
case 'Z':
result.z = (*env)->CallBooleanMethodV(env, obj, mid, args);
break;
case 'B':
result.b = (*env)->CallByteMethodV(env, obj, mid, args);
break;
case 'C':
result.c = (*env)->CallCharMethodV(env, obj, mid, args);
break;
case 'S':
result.s = (*env)->CallShortMethodV(env, obj, mid, args);
break;
case 'I':
result.i = (*env)->CallIntMethodV(env, obj, mid, args);
break;
case 'J':
result.j = (*env)->CallLongMethodV(env, obj, mid, args);
break;
case 'F':
result.f = (*env)->CallFloatMethodV(env, obj, mid, args);
break;
case 'D':
result.d = (*env)->CallDoubleMethodV(env, obj, mid, args);
break;
default:
(*env)->FatalError(env, "JNU_CallMethodByNameV: illegal signature");
}
done1:
(*env)->DeleteLocalRef(env, clazz);
done2:
if (hasException) {
*hasException = (*env)->ExceptionCheck(env);
}
return result;
}
JNIEXPORT jobject JNICALL
JNU_NewObjectByName(JNIEnv *env, const char *class_name,
const char *constructor_sig, ...)
{
jobject obj = NULL;
jclass cls = 0;
jmethodID cls_initMID;
va_list args;
if ((*env)->EnsureLocalCapacity(env, 2) < 0)
goto done;
cls = (*env)->FindClass(env, class_name);
if (cls == 0) {
goto done;
}
cls_initMID = (*env)->GetMethodID(env, cls,
"<init>", constructor_sig);
if (cls_initMID == NULL) {
goto done;
}
va_start(args, constructor_sig);
obj = (*env)->NewObjectV(env, cls, cls_initMID, args);
va_end(args);
done:
(*env)->DeleteLocalRef(env, cls);
return obj;
}
/* Optimized for charset ISO_8559_1 */
static jstring
newSizedString8859_1(JNIEnv *env, const char *str, const int len)
{
jchar buf[512];
jchar *str1;
jstring result;
int i;
if ((*env)->EnsureLocalCapacity(env, 1) < 0)
return NULL;
if (len > 512) {
str1 = (jchar *)malloc(len * sizeof(jchar));
if (str1 == 0) {
JNU_ThrowOutOfMemoryError(env, 0);
return 0;
}
} else
str1 = buf;
for (i=0;i<len;i++)
str1[i] = (unsigned char)str[i];
result = (*env)->NewString(env, str1, len);
if (str1 != buf)
free(str1);
return result;
}
static jstring
newString8859_1(JNIEnv *env, const char *str)
{
int len = (int)strlen(str);
return newSizedString8859_1(env, str, len);
}
static const char*
getString8859_1Chars(JNIEnv *env, jstring jstr)
{
int i;
char *result;
jint len = (*env)->GetStringLength(env, jstr);
const jchar *str = (*env)->GetStringCritical(env, jstr, 0);
if (str == 0) {
return 0;
}
result = MALLOC_MIN4(len);
if (result == 0) {
(*env)->ReleaseStringCritical(env, jstr, str);
JNU_ThrowOutOfMemoryError(env, 0);
return 0;
}
for (i=0; i<len; i++) {
jchar unicode = str[i];
if (unicode <= 0x00ff)
result[i] = (char)unicode;
else
result[i] = '?';
}
result[len] = 0;
(*env)->ReleaseStringCritical(env, jstr, str);
return result;
}
/* Optimized for charset ISO646-US (us-ascii) */
static jstring
newString646_US(JNIEnv *env, const char *str)
{
int len = (int)strlen(str);
jchar buf[512];
jchar *str1;
jstring result;
int i;
if (len > 512) {
str1 = (jchar *)malloc(len * sizeof(jchar));
if (str1 == 0) {
JNU_ThrowOutOfMemoryError(env, 0);
return 0;
}
} else
str1 = buf;
for (i=0; i<len; i++) {
unsigned char c = (unsigned char)str[i];
if (c <= 0x7f)
str1[i] = c;
else
str1[i] = '?';
}
result = (*env)->NewString(env, str1, len);
if (str1 != buf)
free(str1);
return result;
}
static const char*
getString646_USChars(JNIEnv *env, jstring jstr)
{
int i;
char *result;
jint len = (*env)->GetStringLength(env, jstr);
const jchar *str = (*env)->GetStringCritical(env, jstr, 0);
if (str == 0) {
return 0;
}
result = MALLOC_MIN4(len);
if (result == 0) {
(*env)->ReleaseStringCritical(env, jstr, str);
JNU_ThrowOutOfMemoryError(env, 0);
return 0;
}
for (i=0; i<len; i++) {
jchar unicode = str[i];
if (unicode <= 0x007f )
result[i] = (char)unicode;
else
result[i] = '?';
}
result[len] = 0;
(*env)->ReleaseStringCritical(env, jstr, str);
return result;
}
/* enumeration of c1 row from Cp1252 */
static int cp1252c1chars[32] = {
0x20AC,0xFFFD,0x201A,0x0192,0x201E,0x2026,0x2020,0x2021,
0x02C6,0x2030,0x0160,0x2039,0x0152,0xFFFD,0x017D,0xFFFD,
0xFFFD,0x2018,0x2019,0x201C,0x201D,0x2022,0x2013,0x2014,
0x02Dc,0x2122,0x0161,0x203A,0x0153,0xFFFD,0x017E,0x0178
};
/* Optimized for charset Cp1252 */
static jstring
newStringCp1252(JNIEnv *env, const char *str)
{
int len = (int) strlen(str);
jchar buf[512];
jchar *str1;
jstring result;
int i;
if ((*env)->EnsureLocalCapacity(env, 1) < 0)
return NULL;
if (len > 512) {
str1 = (jchar *)malloc(len * sizeof(jchar));
if (str1 == 0) {
JNU_ThrowOutOfMemoryError(env, 0);
return 0;
}
} else
str1 = buf;
for (i=0; i<len; i++) {
unsigned char c = (unsigned char)str[i];
if ((c >= 0x80) && (c <= 0x9f))
str1[i] = cp1252c1chars[c-128];
else
str1[i] = c;
}
result = (*env)->NewString(env, str1, len);
if (str1 != buf)
free(str1);
return result;
}
static const char*
getStringCp1252Chars(JNIEnv *env, jstring jstr)
{
int i;
char *result;
jint len = (*env)->GetStringLength(env, jstr);
const jchar *str = (*env)->GetStringCritical(env, jstr, 0);
if (str == 0) {
return 0;
}
result = MALLOC_MIN4(len);
if (result == 0) {
(*env)->ReleaseStringCritical(env, jstr, str);
JNU_ThrowOutOfMemoryError(env, 0);
return 0;
}
for (i=0; i<len; i++) {
jchar c = str[i];
if (c < 256) {
if ((c >= 0x80) && (c <= 0x9f)) {
result[i] = '?';
} else {
result[i] = (char)c;
}
} else switch(c) {
case 0x20AC: result[i] = (char)0x80; break;
case 0x201A: result[i] = (char)0x82; break;
case 0x0192: result[i] = (char)0x83; break;
case 0x201E: result[i] = (char)0x84; break;
case 0x2026: result[i] = (char)0x85; break;
case 0x2020: result[i] = (char)0x86; break;
case 0x2021: result[i] = (char)0x87; break;
case 0x02C6: result[i] = (char)0x88; break;
case 0x2030: result[i] = (char)0x89; break;
case 0x0160: result[i] = (char)0x8A; break;
case 0x2039: result[i] = (char)0x8B; break;
case 0x0152: result[i] = (char)0x8C; break;
case 0x017D: result[i] = (char)0x8E; break;
case 0x2018: result[i] = (char)0x91; break;
case 0x2019: result[i] = (char)0x92; break;
case 0x201C: result[i] = (char)0x93; break;
case 0x201D: result[i] = (char)0x94; break;
case 0x2022: result[i] = (char)0x95; break;
case 0x2013: result[i] = (char)0x96; break;
case 0x2014: result[i] = (char)0x97; break;
case 0x02DC: result[i] = (char)0x98; break;
case 0x2122: result[i] = (char)0x99; break;
case 0x0161: result[i] = (char)0x9A; break;
case 0x203A: result[i] = (char)0x9B; break;
case 0x0153: result[i] = (char)0x9C; break;
case 0x017E: result[i] = (char)0x9E; break;
case 0x0178: result[i] = (char)0x9F; break;
default: result[i] = '?'; break;
}
}
result[len] = 0;
(*env)->ReleaseStringCritical(env, jstr, str);
return result;
}
static int fastEncoding = NO_ENCODING_YET;
static jstring jnuEncoding = NULL;
/* Cached method IDs */
static jmethodID String_init_ID; /* String(byte[], enc) */
static jmethodID String_getBytes_ID; /* String.getBytes(enc) */
/* Cached field IDs */
static jfieldID String_coder_ID; /* String.coder */
static jfieldID String_value_ID; /* String.value */
static jboolean isJNUEncodingSupported = JNI_FALSE;
static jboolean jnuEncodingSupported(JNIEnv *env) {
jboolean exe;
if (isJNUEncodingSupported == JNI_TRUE) {
return JNI_TRUE;
}
isJNUEncodingSupported = (jboolean) JNU_CallStaticMethodByName (
env, &exe,
"java/nio/charset/Charset",
"isSupported",
"(Ljava/lang/String;)Z",
jnuEncoding).z;
return isJNUEncodingSupported;
}
/* Create a new string by converting str to a heap-allocated byte array and
* calling the appropriate String constructor.
*/
static jstring
newSizedStringJava(JNIEnv *env, const char *str, const int len)
{
jstring result = NULL;
jbyteArray bytes = 0;
if ((*env)->EnsureLocalCapacity(env, 2) < 0)
return NULL;
bytes = (*env)->NewByteArray(env, len);
if (bytes != NULL) {
jclass strClazz = JNU_ClassString(env);
CHECK_NULL_RETURN(strClazz, 0);
(*env)->SetByteArrayRegion(env, bytes, 0, len, (jbyte *)str);
if (jnuEncodingSupported(env)) {
result = (*env)->NewObject(env, strClazz,
String_init_ID, bytes, jnuEncoding);
} else {
/*If the encoding specified in sun.jnu.encoding is not endorsed
by "Charset.isSupported" we have to fall back to use String(byte[])
explicitly here without specifying the encoding name, in which the
StringCoding class will pickup the iso-8859-1 as the fallback
converter for us.
*/
jmethodID mid = (*env)->GetMethodID(env, strClazz,
"<init>", "([B)V");
if (mid != NULL) {
result = (*env)->NewObject(env, strClazz, mid, bytes);
}
}
(*env)->DeleteLocalRef(env, bytes);
return result;
}
return NULL;
}
static jstring
newStringJava(JNIEnv *env, const char *str)
{
int len = (int)strlen(str);
return newSizedStringJava(env, str, len);
}
/* Optimized for charset UTF-8 */
static jstring
newStringUTF8(JNIEnv *env, const char *str)
{
int len;
const unsigned char *p;
unsigned char asciiCheck;
for (asciiCheck = 0, p = (const unsigned char*)str; *p != '\0'; p++) {
asciiCheck |= *p;
}
len = (int)((const char*)p - str);
if (asciiCheck < 0x80) {
// ascii fast-path
return newSizedString8859_1(env, str, len);
}
return newSizedStringJava(env, str, len);
}
/* Initialize the fast encoding from the encoding name.
* Export InitializeEncoding so that the VM can initialize it if required.
*/
JNIEXPORT void
InitializeEncoding(JNIEnv *env, const char *encname)
{
jclass strClazz = NULL;
if ((*env)->EnsureLocalCapacity(env, 3) < 0)
return;
strClazz = JNU_ClassString(env);
CHECK_NULL(strClazz);
if (encname) {
/*
* On Solaris with nl_langinfo() called in GetJavaProperties():
*
* locale undefined -> NULL -> hardcoded default
* "C" locale -> "" -> hardcoded default (on 2.6)
* "C" locale -> "ISO646-US" (on Sol 7/8)
* "en_US" locale -> "ISO8859-1"
* "en_GB" locale -> "ISO8859-1" (on Sol 7/8)
* "en_UK" locale -> "ISO8859-1" (on 2.6)
*/
if ((strcmp(encname, "8859_1") == 0) ||
(strcmp(encname, "ISO8859-1") == 0) ||
(strcmp(encname, "ISO8859_1") == 0) ||
(strcmp(encname, "ISO-8859-1") == 0)) {
fastEncoding = FAST_8859_1;
} else if (strcmp(encname, "UTF-8") == 0) {
jstring enc = (*env)->NewStringUTF(env, encname);
if (enc == NULL)
return;
fastEncoding = FAST_UTF_8;
jnuEncoding = (jstring)(*env)->NewGlobalRef(env, enc);
(*env)->DeleteLocalRef(env, enc);
} else if (strcmp(encname, "ISO646-US") == 0) {
fastEncoding = FAST_646_US;
} else if (strcmp(encname, "Cp1252") == 0 ||
/* This is a temporary fix until we move */
/* to wide character versions of all Windows */
/* calls. */
strcmp(encname, "utf-16le") == 0) {
fastEncoding = FAST_CP1252;
} else {
jstring enc = (*env)->NewStringUTF(env, encname);
if (enc == NULL)
return;
fastEncoding = NO_FAST_ENCODING;
jnuEncoding = (jstring)(*env)->NewGlobalRef(env, enc);
(*env)->DeleteLocalRef(env, enc);
}
} else {
JNU_ThrowInternalError(env, "platform encoding undefined");
return;
}
/* Initialize method-id cache */
String_getBytes_ID = (*env)->GetMethodID(env, strClazz,
"getBytes", "(Ljava/lang/String;)[B");
CHECK_NULL(String_getBytes_ID);
String_init_ID = (*env)->GetMethodID(env, strClazz,
"<init>", "([BLjava/lang/String;)V");
String_coder_ID = (*env)->GetFieldID(env, strClazz, "coder", "B");
String_value_ID = (*env)->GetFieldID(env, strClazz, "value", "[B");
}
JNIEXPORT jstring
NewStringPlatform(JNIEnv *env, const char *str)
{
return JNU_NewStringPlatform(env, str);
}
JNIEXPORT jstring JNICALL
JNU_NewStringPlatform(JNIEnv *env, const char *str)
{
if (fastEncoding == FAST_UTF_8)
return newStringUTF8(env, str);
if (fastEncoding == FAST_8859_1)
return newString8859_1(env, str);
if (fastEncoding == FAST_646_US)
return newString646_US(env, str);
if (fastEncoding == FAST_CP1252)
return newStringCp1252(env, str);
if (fastEncoding == NO_ENCODING_YET) {
JNU_ThrowInternalError(env, "platform encoding not initialized");
return NULL;
}
return newStringJava(env, str);
}
JNIEXPORT const char *
GetStringPlatformChars(JNIEnv *env, jstring jstr, jboolean *isCopy)
{
return JNU_GetStringPlatformChars(env, jstr, isCopy);
}
static const char* getStringBytes(JNIEnv *env, jstring jstr) {
char *result = NULL;
jbyteArray hab = 0;
if ((*env)->EnsureLocalCapacity(env, 2) < 0)
return 0;
if (jnuEncodingSupported(env)) {
hab = (*env)->CallObjectMethod(env, jstr, String_getBytes_ID, jnuEncoding);
} else {
jmethodID mid;
jclass strClazz = JNU_ClassString(env);
CHECK_NULL_RETURN(strClazz, 0);
mid = (*env)->GetMethodID(env, strClazz,
"getBytes", "()[B");
if (mid != NULL) {
hab = (*env)->CallObjectMethod(env, jstr, mid);
}
}
if (!(*env)->ExceptionCheck(env)) {
jint len = (*env)->GetArrayLength(env, hab);
result = MALLOC_MIN4(len);
if (result == 0) {
JNU_ThrowOutOfMemoryError(env, 0);
(*env)->DeleteLocalRef(env, hab);
return 0;
}
(*env)->GetByteArrayRegion(env, hab, 0, len, (jbyte *)result);
result[len] = 0; /* NULL-terminate */
}
(*env)->DeleteLocalRef(env, hab);
return result;
}
static const char*
getStringUTF8(JNIEnv *env, jstring jstr)
{
int i;
char *result;
jbyteArray value;
jint len;
jbyte *str;
jint rlen;
int ri;
jbyte coder = (*env)->GetByteField(env, jstr, String_coder_ID);
if (coder != java_lang_String_LATIN1) {
return getStringBytes(env, jstr);
}
if ((*env)->EnsureLocalCapacity(env, 2) < 0) {
return NULL;
}
value = (*env)->GetObjectField(env, jstr, String_value_ID);
if (value == NULL)
return NULL;
len = (*env)->GetArrayLength(env, value);
str = (*env)->GetPrimitiveArrayCritical(env, value, NULL);
if (str == NULL) {
return NULL;
}
rlen = len;
// we need two bytes for each latin-1 char above 127 (negative jbytes)
for (i = 0; i < len; i++) {
if (str[i] < 0) {
rlen++;
}
}
result = MALLOC_MIN4(rlen);
if (result == NULL) {
(*env)->ReleasePrimitiveArrayCritical(env, value, str, 0);
JNU_ThrowOutOfMemoryError(env, 0);
return NULL;
}
for (ri = 0, i = 0; i < len; i++) {
jbyte c = str[i];
if (c < 0) {
result[ri++] = (char)(0xc0 | ((c & 0xff) >> 6));
result[ri++] = (char)(0x80 | (c & 0x3f));
} else {
result[ri++] = c;
}
}
(*env)->ReleasePrimitiveArrayCritical(env, value, str, 0);
result[rlen] = '\0';
return result;
}
JNIEXPORT const char * JNICALL
JNU_GetStringPlatformChars(JNIEnv *env, jstring jstr, jboolean *isCopy)
{
if (isCopy)
*isCopy = JNI_TRUE;
if (fastEncoding == FAST_UTF_8)
return getStringUTF8(env, jstr);
if (fastEncoding == FAST_8859_1)
return getString8859_1Chars(env, jstr);
if (fastEncoding == FAST_646_US)
return getString646_USChars(env, jstr);
if (fastEncoding == FAST_CP1252)
return getStringCp1252Chars(env, jstr);
if (fastEncoding == NO_ENCODING_YET) {
JNU_ThrowInternalError(env, "platform encoding not initialized");
return 0;
} else
return getStringBytes(env, jstr);
}
JNIEXPORT void JNICALL
JNU_ReleaseStringPlatformChars(JNIEnv *env, jstring jstr, const char *str)
{
free((void *)str);
}
/*
* Export the platform dependent path canonicalization so that
* VM can find it when loading system classes.
* This function is also used by the instrumentation agent.
*/
extern int canonicalize(char *path, const char *out, int len);
JNIEXPORT int
Canonicalize(JNIEnv *unused, char *orig, char *out, int len)
{
/* canonicalize an already natived path */
return canonicalize(orig, out, len);
}
JNIEXPORT jclass JNICALL
JNU_ClassString(JNIEnv *env)
{
static jclass cls = 0;
if (cls == 0) {
jclass c;
if ((*env)->EnsureLocalCapacity(env, 1) < 0)
return 0;
c = (*env)->FindClass(env, "java/lang/String");
CHECK_NULL_RETURN(c, NULL);
cls = (*env)->NewGlobalRef(env, c);
(*env)->DeleteLocalRef(env, c);
}
return cls;
}
JNIEXPORT jclass JNICALL
JNU_ClassClass(JNIEnv *env)
{
static jclass cls = 0;
if (cls == 0) {
jclass c;
if ((*env)->EnsureLocalCapacity(env, 1) < 0)
return 0;
c = (*env)->FindClass(env, "java/lang/Class");
CHECK_NULL_RETURN(c, NULL);
cls = (*env)->NewGlobalRef(env, c);
(*env)->DeleteLocalRef(env, c);
}
return cls;
}
JNIEXPORT jclass JNICALL
JNU_ClassObject(JNIEnv *env)
{
static jclass cls = 0;
if (cls == 0) {
jclass c;
if ((*env)->EnsureLocalCapacity(env, 1) < 0)
return 0;
c = (*env)->FindClass(env, "java/lang/Object");
CHECK_NULL_RETURN(c, NULL);
cls = (*env)->NewGlobalRef(env, c);
(*env)->DeleteLocalRef(env, c);
}
return cls;
}
JNIEXPORT jclass JNICALL
JNU_ClassThrowable(JNIEnv *env)
{
static jclass cls = 0;
if (cls == 0) {
jclass c;
if ((*env)->EnsureLocalCapacity(env, 1) < 0)
return 0;
c = (*env)->FindClass(env, "java/lang/Throwable");
CHECK_NULL_RETURN(c, NULL);
cls = (*env)->NewGlobalRef(env, c);
(*env)->DeleteLocalRef(env, c);
}
return cls;
}
JNIEXPORT jint JNICALL
JNU_CopyObjectArray(JNIEnv *env, jobjectArray dst, jobjectArray src,
jint count)
{
int i;
if ((*env)->EnsureLocalCapacity(env, 1) < 0)
return -1;
for (i=0; i<count; i++) {
jstring p = (*env)->GetObjectArrayElement(env, src, i);
(*env)->SetObjectArrayElement(env, dst, i, p);
(*env)->DeleteLocalRef(env, p);
}
return 0;
}
JNIEXPORT void * JNICALL
JNU_GetEnv(JavaVM *vm, jint version)
{
void *env;
(*vm)->GetEnv(vm, &env, version);
return env;
}
JNIEXPORT jint JNICALL
JNU_IsInstanceOfByName(JNIEnv *env, jobject object, char* classname)
{
jclass cls;
if ((*env)->EnsureLocalCapacity(env, 1) < 0)
return JNI_ERR;
cls = (*env)->FindClass(env, classname);
if (cls != NULL) {
jint result = (*env)->IsInstanceOf(env, object, cls);
(*env)->DeleteLocalRef(env, cls);
return result;
}
return JNI_ERR;
}
JNIEXPORT jboolean JNICALL
JNU_Equals(JNIEnv *env, jobject object1, jobject object2)
{
static jmethodID mid = NULL;
if (mid == NULL) {
jclass objClazz = JNU_ClassObject(env);
CHECK_NULL_RETURN(objClazz, JNI_FALSE);
mid = (*env)->GetMethodID(env, objClazz, "equals",
"(Ljava/lang/Object;)Z");
CHECK_NULL_RETURN(mid, JNI_FALSE);
}
return (*env)->CallBooleanMethod(env, object1, mid, object2);
}
/************************************************************************
* Thread calls
*/
static jmethodID Object_waitMID;
static jmethodID Object_notifyMID;
static jmethodID Object_notifyAllMID;
JNIEXPORT void JNICALL
JNU_MonitorWait(JNIEnv *env, jobject object, jlong timeout)
{
if (object == NULL) {
JNU_ThrowNullPointerException(env, "JNU_MonitorWait argument");
return;
}
if (Object_waitMID == NULL) {
jclass cls = JNU_ClassObject(env);
if (cls == NULL) {
return;
}
Object_waitMID = (*env)->GetMethodID(env, cls, "wait", "(J)V");
if (Object_waitMID == NULL) {
return;
}
}
(*env)->CallVoidMethod(env, object, Object_waitMID, timeout);
}
JNIEXPORT void JNICALL
JNU_Notify(JNIEnv *env, jobject object)
{
if (object == NULL) {
JNU_ThrowNullPointerException(env, "JNU_Notify argument");
return;
}
if (Object_notifyMID == NULL) {
jclass cls = JNU_ClassObject(env);
if (cls == NULL) {
return;
}
Object_notifyMID = (*env)->GetMethodID(env, cls, "notify", "()V");
if (Object_notifyMID == NULL) {
return;
}
}
(*env)->CallVoidMethod(env, object, Object_notifyMID);
}
JNIEXPORT void JNICALL
JNU_NotifyAll(JNIEnv *env, jobject object)
{
if (object == NULL) {
JNU_ThrowNullPointerException(env, "JNU_NotifyAll argument");
return;
}
if (Object_notifyAllMID == NULL) {
jclass cls = JNU_ClassObject(env);
if (cls == NULL) {
return;
}
Object_notifyAllMID = (*env)->GetMethodID(env, cls,"notifyAll", "()V");
if (Object_notifyAllMID == NULL) {
return;
}
}
(*env)->CallVoidMethod(env, object, Object_notifyAllMID);
}
/************************************************************************
* Debugging utilities
*/
JNIEXPORT void JNICALL
JNU_PrintString(JNIEnv *env, char *hdr, jstring string)
{
if (string == NULL) {
fprintf(stderr, "%s: is NULL\n", hdr);
} else {
const char *stringPtr = JNU_GetStringPlatformChars(env, string, 0);
if (stringPtr == 0)
return;
fprintf(stderr, "%s: %s\n", hdr, stringPtr);
JNU_ReleaseStringPlatformChars(env, string, stringPtr);
}
}
JNIEXPORT void JNICALL
JNU_PrintClass(JNIEnv *env, char* hdr, jobject object)
{
if (object == NULL) {
fprintf(stderr, "%s: object is NULL\n", hdr);
return;
} else {
jclass cls = (*env)->GetObjectClass(env, object);
jstring clsName = JNU_ToString(env, cls);
if (clsName == NULL) {
JNU_PrintString(env, hdr, clsName);
}
(*env)->DeleteLocalRef(env, cls);
(*env)->DeleteLocalRef(env, clsName);
}
}
JNIEXPORT jstring JNICALL
JNU_ToString(JNIEnv *env, jobject object)
{
if (object == NULL) {
return (*env)->NewStringUTF(env, "NULL");
} else {
return (jstring)JNU_CallMethodByName(env,
NULL,
object,
"toString",
"()Ljava/lang/String;").l;
}
}
JNIEXPORT jvalue JNICALL
JNU_GetFieldByName(JNIEnv *env,
jboolean *hasException,
jobject obj,
const char *name,
const char *signature)
{
jclass cls;
jfieldID fid;
jvalue result;
result.i = 0;
if ((*env)->EnsureLocalCapacity(env, 3) < 0)
goto done2;
cls = (*env)->GetObjectClass(env, obj);
fid = (*env)->GetFieldID(env, cls, name, signature);
if (fid == 0)
goto done1;
switch (*signature) {
case '[':
case 'L':
result.l = (*env)->GetObjectField(env, obj, fid);
break;
case 'Z':
result.z = (*env)->GetBooleanField(env, obj, fid);
break;
case 'B':
result.b = (*env)->GetByteField(env, obj, fid);
break;
case 'C':
result.c = (*env)->GetCharField(env, obj, fid);
break;
case 'S':
result.s = (*env)->GetShortField(env, obj, fid);
break;
case 'I':
result.i = (*env)->GetIntField(env, obj, fid);
break;
case 'J':
result.j = (*env)->GetLongField(env, obj, fid);
break;
case 'F':
result.f = (*env)->GetFloatField(env, obj, fid);
break;
case 'D':
result.d = (*env)->GetDoubleField(env, obj, fid);
break;
default:
(*env)->FatalError(env, "JNU_GetFieldByName: illegal signature");
}
done1:
(*env)->DeleteLocalRef(env, cls);
done2:
if (hasException) {
*hasException = (*env)->ExceptionCheck(env);
}
return result;
}
JNIEXPORT void JNICALL
JNU_SetFieldByName(JNIEnv *env,
jboolean *hasException,
jobject obj,
const char *name,
const char *signature,
...)
{
jclass cls;
jfieldID fid;
va_list args;
if ((*env)->EnsureLocalCapacity(env, 3) < 0)
goto done2;
cls = (*env)->GetObjectClass(env, obj);
fid = (*env)->GetFieldID(env, cls, name, signature);
if (fid == 0)
goto done1;
va_start(args, signature);
switch (*signature) {
case '[':
case 'L':
(*env)->SetObjectField(env, obj, fid, va_arg(args, jobject));
break;
case 'Z':
(*env)->SetBooleanField(env, obj, fid, (jboolean)va_arg(args, int));
break;
case 'B':
(*env)->SetByteField(env, obj, fid, (jbyte)va_arg(args, int));
break;
case 'C':
(*env)->SetCharField(env, obj, fid, (jchar)va_arg(args, int));
break;
case 'S':
(*env)->SetShortField(env, obj, fid, (jshort)va_arg(args, int));
break;
case 'I':
(*env)->SetIntField(env, obj, fid, va_arg(args, jint));
break;
case 'J':
(*env)->SetLongField(env, obj, fid, va_arg(args, jlong));
break;
case 'F':
(*env)->SetFloatField(env, obj, fid, (jfloat)va_arg(args, jdouble));
break;
case 'D':
(*env)->SetDoubleField(env, obj, fid, va_arg(args, jdouble));
break;
default:
(*env)->FatalError(env, "JNU_SetFieldByName: illegal signature");
}
va_end(args);
done1:
(*env)->DeleteLocalRef(env, cls);
done2:
if (hasException) {
*hasException = (*env)->ExceptionCheck(env);
}
}
JNIEXPORT jvalue JNICALL
JNU_GetStaticFieldByName(JNIEnv *env,
jboolean *hasException,
const char *classname,
const char *name,
const char *signature)
{
jclass cls;
jfieldID fid;
jvalue result;
result.i = 0;
if ((*env)->EnsureLocalCapacity(env, 3) < 0)
goto done2;
cls = (*env)->FindClass(env, classname);
if (cls == 0)
goto done2;
fid = (*env)->GetStaticFieldID(env, cls, name, signature);
if (fid == 0)
goto done1;
switch (*signature) {
case '[':
case 'L':
result.l = (*env)->GetStaticObjectField(env, cls, fid);
break;
case 'Z':
result.z = (*env)->GetStaticBooleanField(env, cls, fid);
break;
case 'B':
result.b = (*env)->GetStaticByteField(env, cls, fid);
break;
case 'C':
result.c = (*env)->GetStaticCharField(env, cls, fid);
break;
case 'S':
result.s = (*env)->GetStaticShortField(env, cls, fid);
break;
case 'I':
result.i = (*env)->GetStaticIntField(env, cls, fid);
break;
case 'J':
result.j = (*env)->GetStaticLongField(env, cls, fid);
break;
case 'F':
result.f = (*env)->GetStaticFloatField(env, cls, fid);
break;
case 'D':
result.d = (*env)->GetStaticDoubleField(env, cls, fid);
break;
default:
(*env)->FatalError(env, "JNU_GetStaticFieldByName: illegal signature");
}
done1:
(*env)->DeleteLocalRef(env, cls);
done2:
if (hasException) {
*hasException = (*env)->ExceptionCheck(env);
}
return result;
}
JNIEXPORT void JNICALL
JNU_SetStaticFieldByName(JNIEnv *env,
jboolean *hasException,
const char *classname,
const char *name,
const char *signature,
...)
{
jclass cls;
jfieldID fid;
va_list args;
if ((*env)->EnsureLocalCapacity(env, 3) < 0)
goto done2;
cls = (*env)->FindClass(env, classname);
if (cls == 0)
goto done2;
fid = (*env)->GetStaticFieldID(env, cls, name, signature);
if (fid == 0)
goto done1;
va_start(args, signature);
switch (*signature) {
case '[':
case 'L':
(*env)->SetStaticObjectField(env, cls, fid, va_arg(args, jobject));
break;
case 'Z':
(*env)->SetStaticBooleanField(env, cls, fid, (jboolean)va_arg(args, int));
break;
case 'B':
(*env)->SetStaticByteField(env, cls, fid, (jbyte)va_arg(args, int));
break;
case 'C':
(*env)->SetStaticCharField(env, cls, fid, (jchar)va_arg(args, int));
break;
case 'S':
(*env)->SetStaticShortField(env, cls, fid, (jshort)va_arg(args, int));
break;
case 'I':
(*env)->SetStaticIntField(env, cls, fid, va_arg(args, jint));
break;
case 'J':
(*env)->SetStaticLongField(env, cls, fid, va_arg(args, jlong));
break;
case 'F':
(*env)->SetStaticFloatField(env, cls, fid, (jfloat)va_arg(args, jdouble));
break;
case 'D':
(*env)->SetStaticDoubleField(env, cls, fid, va_arg(args, jdouble));
break;
default:
(*env)->FatalError(env, "JNU_SetStaticFieldByName: illegal signature");
}
va_end(args);
done1:
(*env)->DeleteLocalRef(env, cls);
done2:
if (hasException) {
*hasException = (*env)->ExceptionCheck(env);
}
}