From f378a448336d4fb3547b53c0712fd7307fc0ba3a Mon Sep 17 00:00:00 2001 From: Jari Vetoniemi Date: Sat, 17 Feb 2018 23:04:55 +0200 Subject: Add fake jvm --- Makefile | 6 +- src/app.c | 26 +- src/fakejvm/jni.h | 1140 ++++++++++++++++++++++++++++++++++ src/fakejvm/jvm.c | 1778 +++++++++++++++++++++++++++++++++++++++++++++++++++++ src/fakejvm/jvm.h | 25 + 5 files changed, 2964 insertions(+), 11 deletions(-) create mode 100644 src/fakejvm/jni.h create mode 100644 src/fakejvm/jvm.c create mode 100644 src/fakejvm/jvm.h diff --git a/Makefile b/Makefile index c9c552a..13688e7 100644 --- a/Makefile +++ b/Makefile @@ -40,9 +40,11 @@ runtime/liblog.so: runtime src/liblog.c linker.a: CFLAGS += -D_GNU_SOURCE -DANDROID_X86_LINKER -DLINKER_DEBUG=1 linker.a: CFLAGS += -Wno-pedantic -Wno-variadic-macros -Wno-pointer-to-int-cast -Wno-int-to-pointer-cast linker.a: src/linker/dlfcn.o src/linker/linker.o src/linker/linker_environ.o src/linker/rt.o src/linker/strlcpy.o +jvm.a: private CFLAGS += -Wno-unused-variable +jvm.a: src/fakejvm/jvm.o app: private LDLIBS += -ldl -Wl,-rpath,runtime -app: src/app.c linker.a runtime/libc.so runtime/libpthread.so runtime/libandroid.so runtime/liblog.so +app: src/app.c linker.a jvm.a runtime/libc.so runtime/libpthread.so runtime/libandroid.so runtime/liblog.so install-bin: $(bins) install -Dm755 $^ -t "$(DESTDIR)$(PREFIX)$(BINDIR)" @@ -50,7 +52,7 @@ install-bin: $(bins) install: install-bin clean: - $(RM) $(bins) *.a src/linker/*.o + $(RM) $(bins) *.a src/linker/*.o src/fakejvm/*.o $(RM) -r runtime .PHONY: all clean install diff --git a/src/app.c b/src/app.c index 6dd8e10..5d23a3d 100644 --- a/src/app.c +++ b/src/app.c @@ -4,6 +4,7 @@ #include #include #include +#include "fakejvm/jvm.h" extern void *apkenv_android_dlopen(const char*, int); extern void *apkenv_android_dlclose(void*); @@ -16,25 +17,32 @@ main(int argc, const char *argv[]) if (argc < 2) errx(EXIT_FAILURE, "usage: so-file"); + printf("loading runtime\n"); dlopen("libpthread.so", RTLD_NOW | RTLD_GLOBAL); - printf("loading: %s\n", argv[1]); + printf("loading module: %s\n", argv[1]); - char path[4096]; - snprintf(path, sizeof(path), "%s", argv[1]); - apkenv_parse_library_path(dirname(path), ";"); + { + char path[4096]; + snprintf(path, sizeof(path), "%s", argv[1]); + apkenv_parse_library_path(dirname(path), ";"); + } void *handle; if (!(handle = apkenv_android_dlopen(argv[1], RTLD_NOW | RTLD_LOCAL))) errx(EXIT_FAILURE, "dlopen failed: %s", dlerror()); - printf("loaded: %s\n", argv[1]); + printf("trying JNI_OnLoad from: %s\n", argv[1]); - void* (*JNI_OnLoad)(void*, void*) = apkenv_android_dlsym(handle, "JNI_OnLoad"); - assert(JNI_OnLoad); + { + struct jvm jvm; + jvm_init(&jvm); - char data[1024]; - JNI_OnLoad(data, NULL); + void* (*JNI_OnLoad)(void*, void*) = apkenv_android_dlsym(handle, "JNI_OnLoad"); + assert(JNI_OnLoad); + + JNI_OnLoad(&jvm.vm, NULL); + } // apkenv_android_dlclose(handle); return EXIT_SUCCESS; diff --git a/src/fakejvm/jni.h b/src/fakejvm/jni.h new file mode 100644 index 0000000..ad954c8 --- /dev/null +++ b/src/fakejvm/jni.h @@ -0,0 +1,1140 @@ +/* + * Copyright 2006 The Android Open Source Project + * + * JNI specification, as defined by Sun: + * http://java.sun.com/javase/6/docs/technotes/guides/jni/spec/jniTOC.html + * + * Everything here is expected to be VM-neutral. + */ +#ifndef _JNI_H +#define _JNI_H + +#include + +/* + * Primitive types that match up with Java equivalents. + */ +#ifdef HAVE_INTTYPES_H +# include /* C99 */ +typedef uint8_t jboolean; /* unsigned 8 bits */ +typedef int8_t jbyte; /* signed 8 bits */ +typedef uint16_t jchar; /* unsigned 16 bits */ +typedef int16_t jshort; /* signed 16 bits */ +typedef int32_t jint; /* signed 32 bits */ +typedef int64_t jlong; /* signed 64 bits */ +typedef float jfloat; /* 32-bit IEEE 754 */ +typedef double jdouble; /* 64-bit IEEE 754 */ +#else +typedef unsigned char jboolean; /* unsigned 8 bits */ +typedef signed char jbyte; /* signed 8 bits */ +typedef unsigned short jchar; /* unsigned 16 bits */ +typedef short jshort; /* signed 16 bits */ +typedef int jint; /* signed 32 bits */ +typedef long long jlong; /* signed 64 bits */ +typedef float jfloat; /* 32-bit IEEE 754 */ +typedef double jdouble; /* 64-bit IEEE 754 */ +#endif + +/* "cardinal indices and sizes" */ +typedef jint jsize; + +#ifdef __cplusplus +/* + * Reference types, in C++ + */ +class _jobject {}; +class _jclass : public _jobject {}; +class _jstring : public _jobject {}; +class _jarray : public _jobject {}; +class _jobjectArray : public _jarray {}; +class _jbooleanArray : public _jarray {}; +class _jbyteArray : public _jarray {}; +class _jcharArray : public _jarray {}; +class _jshortArray : public _jarray {}; +class _jintArray : public _jarray {}; +class _jlongArray : public _jarray {}; +class _jfloatArray : public _jarray {}; +class _jdoubleArray : public _jarray {}; +class _jthrowable : public _jobject {}; + +typedef _jobject* jobject; +typedef _jclass* jclass; +typedef _jstring* jstring; +typedef _jarray* jarray; +typedef _jobjectArray* jobjectArray; +typedef _jbooleanArray* jbooleanArray; +typedef _jbyteArray* jbyteArray; +typedef _jcharArray* jcharArray; +typedef _jshortArray* jshortArray; +typedef _jintArray* jintArray; +typedef _jlongArray* jlongArray; +typedef _jfloatArray* jfloatArray; +typedef _jdoubleArray* jdoubleArray; +typedef _jthrowable* jthrowable; +typedef _jobject* jweak; + + +#else /* not __cplusplus */ + +/* + * Reference types, in C. + */ +typedef void* jobject; +typedef jobject jclass; +typedef jobject jstring; +typedef jobject jarray; +typedef jarray jobjectArray; +typedef jarray jbooleanArray; +typedef jarray jbyteArray; +typedef jarray jcharArray; +typedef jarray jshortArray; +typedef jarray jintArray; +typedef jarray jlongArray; +typedef jarray jfloatArray; +typedef jarray jdoubleArray; +typedef jobject jthrowable; +typedef jobject jweak; + +#endif /* not __cplusplus */ + +struct _jfieldID; /* opaque structure */ +typedef struct _jfieldID* jfieldID; /* field IDs */ + +struct _jmethodID; /* opaque structure */ +typedef struct _jmethodID* jmethodID; /* method IDs */ + +struct JNIInvokeInterface; + +typedef union jvalue { + jboolean z; + jbyte b; + jchar c; + jshort s; + jint i; + jlong j; + jfloat f; + jdouble d; + jobject l; +} jvalue; + +typedef enum jobjectRefType { + JNIInvalidRefType = 0, + JNILocalRefType = 1, + JNIGlobalRefType = 2, + JNIWeakGlobalRefType = 3 +} jobjectRefType; + +typedef struct { + const char* name; + const char* signature; + void* fnPtr; +} JNINativeMethod; + +struct _JNIEnv; +struct _JavaVM; +typedef const struct JNINativeInterface* C_JNIEnv; + +#if defined(__cplusplus) +typedef _JNIEnv JNIEnv; +typedef _JavaVM JavaVM; +#else +typedef const struct JNINativeInterface* JNIEnv; +typedef const struct JNIInvokeInterface* JavaVM; +#endif + +/* + * Table of interface function pointers. + */ +struct JNINativeInterface { + void* reserved0; + void* reserved1; + void* reserved2; + void* reserved3; + + jint (*GetVersion)(JNIEnv *); + + jclass (*DefineClass)(JNIEnv*, const char*, jobject, const jbyte*, + jsize); + jclass (*FindClass)(JNIEnv*, const char*); + + jmethodID (*FromReflectedMethod)(JNIEnv*, jobject); + jfieldID (*FromReflectedField)(JNIEnv*, jobject); + /* spec doesn't show jboolean parameter */ + jobject (*ToReflectedMethod)(JNIEnv*, jclass, jmethodID, jboolean); + + jclass (*GetSuperclass)(JNIEnv*, jclass); + jboolean (*IsAssignableFrom)(JNIEnv*, jclass, jclass); + + /* spec doesn't show jboolean parameter */ + jobject (*ToReflectedField)(JNIEnv*, jclass, jfieldID, jboolean); + + jint (*Throw)(JNIEnv*, jthrowable); + jint (*ThrowNew)(JNIEnv *, jclass, const char *); + jthrowable (*ExceptionOccurred)(JNIEnv*); + void (*ExceptionDescribe)(JNIEnv*); + void (*ExceptionClear)(JNIEnv*); + void (*FatalError)(JNIEnv*, const char*); + + jint (*PushLocalFrame)(JNIEnv*, jint); + jobject (*PopLocalFrame)(JNIEnv*, jobject); + + jobject (*NewGlobalRef)(JNIEnv*, jobject); + void (*DeleteGlobalRef)(JNIEnv*, jobject); + void (*DeleteLocalRef)(JNIEnv*, jobject); + jboolean (*IsSameObject)(JNIEnv*, jobject, jobject); + + jobject (*NewLocalRef)(JNIEnv*, jobject); + jint (*EnsureLocalCapacity)(JNIEnv*, jint); + + jobject (*AllocObject)(JNIEnv*, jclass); + jobject (*NewObject)(JNIEnv*, jclass, jmethodID, ...); + jobject (*NewObjectV)(JNIEnv*, jclass, jmethodID, va_list); + jobject (*NewObjectA)(JNIEnv*, jclass, jmethodID, jvalue*); + + jclass (*GetObjectClass)(JNIEnv*, jobject); + jboolean (*IsInstanceOf)(JNIEnv*, jobject, jclass); + jmethodID (*GetMethodID)(JNIEnv*, jclass, const char*, const char*); + + jobject (*CallObjectMethod)(JNIEnv*, jobject, jmethodID, ...); + jobject (*CallObjectMethodV)(JNIEnv*, jobject, jmethodID, va_list); + jobject (*CallObjectMethodA)(JNIEnv*, jobject, jmethodID, jvalue*); + jboolean (*CallBooleanMethod)(JNIEnv*, jobject, jmethodID, ...); + jboolean (*CallBooleanMethodV)(JNIEnv*, jobject, jmethodID, va_list); + jboolean (*CallBooleanMethodA)(JNIEnv*, jobject, jmethodID, jvalue*); + jbyte (*CallByteMethod)(JNIEnv*, jobject, jmethodID, ...); + jbyte (*CallByteMethodV)(JNIEnv*, jobject, jmethodID, va_list); + jbyte (*CallByteMethodA)(JNIEnv*, jobject, jmethodID, jvalue*); + jchar (*CallCharMethod)(JNIEnv*, jobject, jmethodID, ...); + jchar (*CallCharMethodV)(JNIEnv*, jobject, jmethodID, va_list); + jchar (*CallCharMethodA)(JNIEnv*, jobject, jmethodID, jvalue*); + jshort (*CallShortMethod)(JNIEnv*, jobject, jmethodID, ...); + jshort (*CallShortMethodV)(JNIEnv*, jobject, jmethodID, va_list); + jshort (*CallShortMethodA)(JNIEnv*, jobject, jmethodID, jvalue*); + jint (*CallIntMethod)(JNIEnv*, jobject, jmethodID, ...); + jint (*CallIntMethodV)(JNIEnv*, jobject, jmethodID, va_list); + jint (*CallIntMethodA)(JNIEnv*, jobject, jmethodID, jvalue*); + jlong (*CallLongMethod)(JNIEnv*, jobject, jmethodID, ...); + jlong (*CallLongMethodV)(JNIEnv*, jobject, jmethodID, va_list); + jlong (*CallLongMethodA)(JNIEnv*, jobject, jmethodID, jvalue*); + jfloat (*CallFloatMethod)(JNIEnv*, jobject, jmethodID, ...); + jfloat (*CallFloatMethodV)(JNIEnv*, jobject, jmethodID, va_list); + jfloat (*CallFloatMethodA)(JNIEnv*, jobject, jmethodID, jvalue*); + jdouble (*CallDoubleMethod)(JNIEnv*, jobject, jmethodID, ...); + jdouble (*CallDoubleMethodV)(JNIEnv*, jobject, jmethodID, va_list); + jdouble (*CallDoubleMethodA)(JNIEnv*, jobject, jmethodID, jvalue*); + void (*CallVoidMethod)(JNIEnv*, jobject, jmethodID, ...); + void (*CallVoidMethodV)(JNIEnv*, jobject, jmethodID, va_list); + void (*CallVoidMethodA)(JNIEnv*, jobject, jmethodID, jvalue*); + + jobject (*CallNonvirtualObjectMethod)(JNIEnv*, jobject, jclass, + jmethodID, ...); + jobject (*CallNonvirtualObjectMethodV)(JNIEnv*, jobject, jclass, + jmethodID, va_list); + jobject (*CallNonvirtualObjectMethodA)(JNIEnv*, jobject, jclass, + jmethodID, jvalue*); + jboolean (*CallNonvirtualBooleanMethod)(JNIEnv*, jobject, jclass, + jmethodID, ...); + jboolean (*CallNonvirtualBooleanMethodV)(JNIEnv*, jobject, jclass, + jmethodID, va_list); + jboolean (*CallNonvirtualBooleanMethodA)(JNIEnv*, jobject, jclass, + jmethodID, jvalue*); + jbyte (*CallNonvirtualByteMethod)(JNIEnv*, jobject, jclass, + jmethodID, ...); + jbyte (*CallNonvirtualByteMethodV)(JNIEnv*, jobject, jclass, + jmethodID, va_list); + jbyte (*CallNonvirtualByteMethodA)(JNIEnv*, jobject, jclass, + jmethodID, jvalue*); + jchar (*CallNonvirtualCharMethod)(JNIEnv*, jobject, jclass, + jmethodID, ...); + jchar (*CallNonvirtualCharMethodV)(JNIEnv*, jobject, jclass, + jmethodID, va_list); + jchar (*CallNonvirtualCharMethodA)(JNIEnv*, jobject, jclass, + jmethodID, jvalue*); + jshort (*CallNonvirtualShortMethod)(JNIEnv*, jobject, jclass, + jmethodID, ...); + jshort (*CallNonvirtualShortMethodV)(JNIEnv*, jobject, jclass, + jmethodID, va_list); + jshort (*CallNonvirtualShortMethodA)(JNIEnv*, jobject, jclass, + jmethodID, jvalue*); + jint (*CallNonvirtualIntMethod)(JNIEnv*, jobject, jclass, + jmethodID, ...); + jint (*CallNonvirtualIntMethodV)(JNIEnv*, jobject, jclass, + jmethodID, va_list); + jint (*CallNonvirtualIntMethodA)(JNIEnv*, jobject, jclass, + jmethodID, jvalue*); + jlong (*CallNonvirtualLongMethod)(JNIEnv*, jobject, jclass, + jmethodID, ...); + jlong (*CallNonvirtualLongMethodV)(JNIEnv*, jobject, jclass, + jmethodID, va_list); + jlong (*CallNonvirtualLongMethodA)(JNIEnv*, jobject, jclass, + jmethodID, jvalue*); + jfloat (*CallNonvirtualFloatMethod)(JNIEnv*, jobject, jclass, + jmethodID, ...); + jfloat (*CallNonvirtualFloatMethodV)(JNIEnv*, jobject, jclass, + jmethodID, va_list); + jfloat (*CallNonvirtualFloatMethodA)(JNIEnv*, jobject, jclass, + jmethodID, jvalue*); + jdouble (*CallNonvirtualDoubleMethod)(JNIEnv*, jobject, jclass, + jmethodID, ...); + jdouble (*CallNonvirtualDoubleMethodV)(JNIEnv*, jobject, jclass, + jmethodID, va_list); + jdouble (*CallNonvirtualDoubleMethodA)(JNIEnv*, jobject, jclass, + jmethodID, jvalue*); + void (*CallNonvirtualVoidMethod)(JNIEnv*, jobject, jclass, + jmethodID, ...); + void (*CallNonvirtualVoidMethodV)(JNIEnv*, jobject, jclass, + jmethodID, va_list); + void (*CallNonvirtualVoidMethodA)(JNIEnv*, jobject, jclass, + jmethodID, jvalue*); + + jfieldID (*GetFieldID)(JNIEnv*, jclass, const char*, const char*); + + jobject (*GetObjectField)(JNIEnv*, jobject, jfieldID); + jboolean (*GetBooleanField)(JNIEnv*, jobject, jfieldID); + jbyte (*GetByteField)(JNIEnv*, jobject, jfieldID); + jchar (*GetCharField)(JNIEnv*, jobject, jfieldID); + jshort (*GetShortField)(JNIEnv*, jobject, jfieldID); + jint (*GetIntField)(JNIEnv*, jobject, jfieldID); + jlong (*GetLongField)(JNIEnv*, jobject, jfieldID); + jfloat (*GetFloatField)(JNIEnv*, jobject, jfieldID); + jdouble (*GetDoubleField)(JNIEnv*, jobject, jfieldID); + + void (*SetObjectField)(JNIEnv*, jobject, jfieldID, jobject); + void (*SetBooleanField)(JNIEnv*, jobject, jfieldID, jboolean); + void (*SetByteField)(JNIEnv*, jobject, jfieldID, jbyte); + void (*SetCharField)(JNIEnv*, jobject, jfieldID, jchar); + void (*SetShortField)(JNIEnv*, jobject, jfieldID, jshort); + void (*SetIntField)(JNIEnv*, jobject, jfieldID, jint); + void (*SetLongField)(JNIEnv*, jobject, jfieldID, jlong); + void (*SetFloatField)(JNIEnv*, jobject, jfieldID, jfloat); + void (*SetDoubleField)(JNIEnv*, jobject, jfieldID, jdouble); + + jmethodID (*GetStaticMethodID)(JNIEnv*, jclass, const char*, const char*); + + jobject (*CallStaticObjectMethod)(JNIEnv*, jclass, jmethodID, ...); + jobject (*CallStaticObjectMethodV)(JNIEnv*, jclass, jmethodID, va_list); + jobject (*CallStaticObjectMethodA)(JNIEnv*, jclass, jmethodID, jvalue*); + jboolean (*CallStaticBooleanMethod)(JNIEnv*, jclass, jmethodID, ...); + jboolean (*CallStaticBooleanMethodV)(JNIEnv*, jclass, jmethodID, + va_list); + jboolean (*CallStaticBooleanMethodA)(JNIEnv*, jclass, jmethodID, + jvalue*); + jbyte (*CallStaticByteMethod)(JNIEnv*, jclass, jmethodID, ...); + jbyte (*CallStaticByteMethodV)(JNIEnv*, jclass, jmethodID, va_list); + jbyte (*CallStaticByteMethodA)(JNIEnv*, jclass, jmethodID, jvalue*); + jchar (*CallStaticCharMethod)(JNIEnv*, jclass, jmethodID, ...); + jchar (*CallStaticCharMethodV)(JNIEnv*, jclass, jmethodID, va_list); + jchar (*CallStaticCharMethodA)(JNIEnv*, jclass, jmethodID, jvalue*); + jshort (*CallStaticShortMethod)(JNIEnv*, jclass, jmethodID, ...); + jshort (*CallStaticShortMethodV)(JNIEnv*, jclass, jmethodID, va_list); + jshort (*CallStaticShortMethodA)(JNIEnv*, jclass, jmethodID, jvalue*); + jint (*CallStaticIntMethod)(JNIEnv*, jclass, jmethodID, ...); + jint (*CallStaticIntMethodV)(JNIEnv*, jclass, jmethodID, va_list); + jint (*CallStaticIntMethodA)(JNIEnv*, jclass, jmethodID, jvalue*); + jlong (*CallStaticLongMethod)(JNIEnv*, jclass, jmethodID, ...); + jlong (*CallStaticLongMethodV)(JNIEnv*, jclass, jmethodID, va_list); + jlong (*CallStaticLongMethodA)(JNIEnv*, jclass, jmethodID, jvalue*); + jfloat (*CallStaticFloatMethod)(JNIEnv*, jclass, jmethodID, ...); + jfloat (*CallStaticFloatMethodV)(JNIEnv*, jclass, jmethodID, va_list); + jfloat (*CallStaticFloatMethodA)(JNIEnv*, jclass, jmethodID, jvalue*); + jdouble (*CallStaticDoubleMethod)(JNIEnv*, jclass, jmethodID, ...); + jdouble (*CallStaticDoubleMethodV)(JNIEnv*, jclass, jmethodID, va_list); + jdouble (*CallStaticDoubleMethodA)(JNIEnv*, jclass, jmethodID, jvalue*); + void (*CallStaticVoidMethod)(JNIEnv*, jclass, jmethodID, ...); + void (*CallStaticVoidMethodV)(JNIEnv*, jclass, jmethodID, va_list); + void (*CallStaticVoidMethodA)(JNIEnv*, jclass, jmethodID, jvalue*); + + jfieldID (*GetStaticFieldID)(JNIEnv*, jclass, const char*, + const char*); + + jobject (*GetStaticObjectField)(JNIEnv*, jclass, jfieldID); + jboolean (*GetStaticBooleanField)(JNIEnv*, jclass, jfieldID); + jbyte (*GetStaticByteField)(JNIEnv*, jclass, jfieldID); + jchar (*GetStaticCharField)(JNIEnv*, jclass, jfieldID); + jshort (*GetStaticShortField)(JNIEnv*, jclass, jfieldID); + jint (*GetStaticIntField)(JNIEnv*, jclass, jfieldID); + jlong (*GetStaticLongField)(JNIEnv*, jclass, jfieldID); + jfloat (*GetStaticFloatField)(JNIEnv*, jclass, jfieldID); + jdouble (*GetStaticDoubleField)(JNIEnv*, jclass, jfieldID); + + void (*SetStaticObjectField)(JNIEnv*, jclass, jfieldID, jobject); + void (*SetStaticBooleanField)(JNIEnv*, jclass, jfieldID, jboolean); + void (*SetStaticByteField)(JNIEnv*, jclass, jfieldID, jbyte); + void (*SetStaticCharField)(JNIEnv*, jclass, jfieldID, jchar); + void (*SetStaticShortField)(JNIEnv*, jclass, jfieldID, jshort); + void (*SetStaticIntField)(JNIEnv*, jclass, jfieldID, jint); + void (*SetStaticLongField)(JNIEnv*, jclass, jfieldID, jlong); + void (*SetStaticFloatField)(JNIEnv*, jclass, jfieldID, jfloat); + void (*SetStaticDoubleField)(JNIEnv*, jclass, jfieldID, jdouble); + + jstring (*NewString)(JNIEnv*, const jchar*, jsize); + jsize (*GetStringLength)(JNIEnv*, jstring); + const jchar* (*GetStringChars)(JNIEnv*, jstring, jboolean*); + void (*ReleaseStringChars)(JNIEnv*, jstring, const jchar*); + jstring (*NewStringUTF)(JNIEnv*, const char*); + jsize (*GetStringUTFLength)(JNIEnv*, jstring); + /* JNI spec says this returns const jbyte*, but that's inconsistent */ + const char* (*GetStringUTFChars)(JNIEnv*, jstring, jboolean*); + void (*ReleaseStringUTFChars)(JNIEnv*, jstring, const char*); + jsize (*GetArrayLength)(JNIEnv*, jarray); + jobjectArray (*NewObjectArray)(JNIEnv*, jsize, jclass, jobject); + jobject (*GetObjectArrayElement)(JNIEnv*, jobjectArray, jsize); + void (*SetObjectArrayElement)(JNIEnv*, jobjectArray, jsize, jobject); + + jbooleanArray (*NewBooleanArray)(JNIEnv*, jsize); + jbyteArray (*NewByteArray)(JNIEnv*, jsize); + jcharArray (*NewCharArray)(JNIEnv*, jsize); + jshortArray (*NewShortArray)(JNIEnv*, jsize); + jintArray (*NewIntArray)(JNIEnv*, jsize); + jlongArray (*NewLongArray)(JNIEnv*, jsize); + jfloatArray (*NewFloatArray)(JNIEnv*, jsize); + jdoubleArray (*NewDoubleArray)(JNIEnv*, jsize); + + jboolean* (*GetBooleanArrayElements)(JNIEnv*, jbooleanArray, jboolean*); + jbyte* (*GetByteArrayElements)(JNIEnv*, jbyteArray, jboolean*); + jchar* (*GetCharArrayElements)(JNIEnv*, jcharArray, jboolean*); + jshort* (*GetShortArrayElements)(JNIEnv*, jshortArray, jboolean*); + jint* (*GetIntArrayElements)(JNIEnv*, jintArray, jboolean*); + jlong* (*GetLongArrayElements)(JNIEnv*, jlongArray, jboolean*); + jfloat* (*GetFloatArrayElements)(JNIEnv*, jfloatArray, jboolean*); + jdouble* (*GetDoubleArrayElements)(JNIEnv*, jdoubleArray, jboolean*); + + void (*ReleaseBooleanArrayElements)(JNIEnv*, jbooleanArray, + jboolean*, jint); + void (*ReleaseByteArrayElements)(JNIEnv*, jbyteArray, + jbyte*, jint); + void (*ReleaseCharArrayElements)(JNIEnv*, jcharArray, + jchar*, jint); + void (*ReleaseShortArrayElements)(JNIEnv*, jshortArray, + jshort*, jint); + void (*ReleaseIntArrayElements)(JNIEnv*, jintArray, + jint*, jint); + void (*ReleaseLongArrayElements)(JNIEnv*, jlongArray, + jlong*, jint); + void (*ReleaseFloatArrayElements)(JNIEnv*, jfloatArray, + jfloat*, jint); + void (*ReleaseDoubleArrayElements)(JNIEnv*, jdoubleArray, + jdouble*, jint); + + void (*GetBooleanArrayRegion)(JNIEnv*, jbooleanArray, + jsize, jsize, jboolean*); + void (*GetByteArrayRegion)(JNIEnv*, jbyteArray, + jsize, jsize, jbyte*); + void (*GetCharArrayRegion)(JNIEnv*, jcharArray, + jsize, jsize, jchar*); + void (*GetShortArrayRegion)(JNIEnv*, jshortArray, + jsize, jsize, jshort*); + void (*GetIntArrayRegion)(JNIEnv*, jintArray, + jsize, jsize, jint*); + void (*GetLongArrayRegion)(JNIEnv*, jlongArray, + jsize, jsize, jlong*); + void (*GetFloatArrayRegion)(JNIEnv*, jfloatArray, + jsize, jsize, jfloat*); + void (*GetDoubleArrayRegion)(JNIEnv*, jdoubleArray, + jsize, jsize, jdouble*); + + /* spec shows these without const; some jni.h do, some don't */ + void (*SetBooleanArrayRegion)(JNIEnv*, jbooleanArray, + jsize, jsize, const jboolean*); + void (*SetByteArrayRegion)(JNIEnv*, jbyteArray, + jsize, jsize, const jbyte*); + void (*SetCharArrayRegion)(JNIEnv*, jcharArray, + jsize, jsize, const jchar*); + void (*SetShortArrayRegion)(JNIEnv*, jshortArray, + jsize, jsize, const jshort*); + void (*SetIntArrayRegion)(JNIEnv*, jintArray, + jsize, jsize, const jint*); + void (*SetLongArrayRegion)(JNIEnv*, jlongArray, + jsize, jsize, const jlong*); + void (*SetFloatArrayRegion)(JNIEnv*, jfloatArray, + jsize, jsize, const jfloat*); + void (*SetDoubleArrayRegion)(JNIEnv*, jdoubleArray, + jsize, jsize, const jdouble*); + + jint (*RegisterNatives)(JNIEnv*, jclass, const JNINativeMethod*, + jint); + jint (*UnregisterNatives)(JNIEnv*, jclass); + jint (*MonitorEnter)(JNIEnv*, jobject); + jint (*MonitorExit)(JNIEnv*, jobject); + jint (*GetJavaVM)(JNIEnv*, JavaVM**); + + void (*GetStringRegion)(JNIEnv*, jstring, jsize, jsize, jchar*); + void (*GetStringUTFRegion)(JNIEnv*, jstring, jsize, jsize, char*); + + void* (*GetPrimitiveArrayCritical)(JNIEnv*, jarray, jboolean*); + void (*ReleasePrimitiveArrayCritical)(JNIEnv*, jarray, void*, jint); + + const jchar* (*GetStringCritical)(JNIEnv*, jstring, jboolean*); + void (*ReleaseStringCritical)(JNIEnv*, jstring, const jchar*); + + jweak (*NewWeakGlobalRef)(JNIEnv*, jobject); + void (*DeleteWeakGlobalRef)(JNIEnv*, jweak); + + jboolean (*ExceptionCheck)(JNIEnv*); + + jobject (*NewDirectByteBuffer)(JNIEnv*, void*, jlong); + void* (*GetDirectBufferAddress)(JNIEnv*, jobject); + jlong (*GetDirectBufferCapacity)(JNIEnv*, jobject); + + /* added in JNI 1.6 */ + jobjectRefType (*GetObjectRefType)(JNIEnv*, jobject); +}; + +/* + * C++ object wrapper. + * + * This is usually overlaid on a C struct whose first element is a + * JNINativeInterface*. We rely somewhat on compiler behavior. + */ +struct _JNIEnv { + /* do not rename this; it does not seem to be entirely opaque */ + const struct JNINativeInterface* functions; + +#if defined(__cplusplus) + + jint GetVersion() + { return functions->GetVersion(this); } + + jclass DefineClass(const char *name, jobject loader, const jbyte* buf, + jsize bufLen) + { return functions->DefineClass(this, name, loader, buf, bufLen); } + + jclass FindClass(const char* name) + { return functions->FindClass(this, name); } + + jmethodID FromReflectedMethod(jobject method) + { return functions->FromReflectedMethod(this, method); } + + jfieldID FromReflectedField(jobject field) + { return functions->FromReflectedField(this, field); } + + jobject ToReflectedMethod(jclass cls, jmethodID methodID, jboolean isStatic) + { return functions->ToReflectedMethod(this, cls, methodID, isStatic); } + + jclass GetSuperclass(jclass clazz) + { return functions->GetSuperclass(this, clazz); } + + jboolean IsAssignableFrom(jclass clazz1, jclass clazz2) + { return functions->IsAssignableFrom(this, clazz1, clazz2); } + + jobject ToReflectedField(jclass cls, jfieldID fieldID, jboolean isStatic) + { return functions->ToReflectedField(this, cls, fieldID, isStatic); } + + jint Throw(jthrowable obj) + { return functions->Throw(this, obj); } + + jint ThrowNew(jclass clazz, const char* message) + { return functions->ThrowNew(this, clazz, message); } + + jthrowable ExceptionOccurred() + { return functions->ExceptionOccurred(this); } + + void ExceptionDescribe() + { functions->ExceptionDescribe(this); } + + void ExceptionClear() + { functions->ExceptionClear(this); } + + void FatalError(const char* msg) + { functions->FatalError(this, msg); } + + jint PushLocalFrame(jint capacity) + { return functions->PushLocalFrame(this, capacity); } + + jobject PopLocalFrame(jobject result) + { return functions->PopLocalFrame(this, result); } + + jobject NewGlobalRef(jobject obj) + { return functions->NewGlobalRef(this, obj); } + + void DeleteGlobalRef(jobject globalRef) + { functions->DeleteGlobalRef(this, globalRef); } + + void DeleteLocalRef(jobject localRef) + { functions->DeleteLocalRef(this, localRef); } + + jboolean IsSameObject(jobject ref1, jobject ref2) + { return functions->IsSameObject(this, ref1, ref2); } + + jobject NewLocalRef(jobject ref) + { return functions->NewLocalRef(this, ref); } + + jint EnsureLocalCapacity(jint capacity) + { return functions->EnsureLocalCapacity(this, capacity); } + + jobject AllocObject(jclass clazz) + { return functions->AllocObject(this, clazz); } + + jobject NewObject(jclass clazz, jmethodID methodID, ...) + { + va_list args; + va_start(args, methodID); + jobject result = functions->NewObjectV(this, clazz, methodID, args); + va_end(args); + return result; + } + + jobject NewObjectV(jclass clazz, jmethodID methodID, va_list args) + { return functions->NewObjectV(this, clazz, methodID, args); } + + jobject NewObjectA(jclass clazz, jmethodID methodID, jvalue* args) + { return functions->NewObjectA(this, clazz, methodID, args); } + + jclass GetObjectClass(jobject obj) + { return functions->GetObjectClass(this, obj); } + + jboolean IsInstanceOf(jobject obj, jclass clazz) + { return functions->IsInstanceOf(this, obj, clazz); } + + jmethodID GetMethodID(jclass clazz, const char* name, const char* sig) + { return functions->GetMethodID(this, clazz, name, sig); } + +#define CALL_TYPE_METHOD(_jtype, _jname) \ + _jtype Call##_jname##Method(jobject obj, jmethodID methodID, ...) \ + { \ + _jtype result; \ + va_list args; \ + va_start(args, methodID); \ + result = functions->Call##_jname##MethodV(this, obj, methodID, \ + args); \ + va_end(args); \ + return result; \ + } +#define CALL_TYPE_METHODV(_jtype, _jname) \ + _jtype Call##_jname##MethodV(jobject obj, jmethodID methodID, \ + va_list args) \ + { return functions->Call##_jname##MethodV(this, obj, methodID, args); } +#define CALL_TYPE_METHODA(_jtype, _jname) \ + _jtype Call##_jname##MethodA(jobject obj, jmethodID methodID, \ + jvalue* args) \ + { return functions->Call##_jname##MethodA(this, obj, methodID, args); } + +#define CALL_TYPE(_jtype, _jname) \ + CALL_TYPE_METHOD(_jtype, _jname) \ + CALL_TYPE_METHODV(_jtype, _jname) \ + CALL_TYPE_METHODA(_jtype, _jname) + + CALL_TYPE(jobject, Object) + CALL_TYPE(jboolean, Boolean) + CALL_TYPE(jbyte, Byte) + CALL_TYPE(jchar, Char) + CALL_TYPE(jshort, Short) + CALL_TYPE(jint, Int) + CALL_TYPE(jlong, Long) + CALL_TYPE(jfloat, Float) + CALL_TYPE(jdouble, Double) + + void CallVoidMethod(jobject obj, jmethodID methodID, ...) + { + va_list args; + va_start(args, methodID); + functions->CallVoidMethodV(this, obj, methodID, args); + va_end(args); + } + void CallVoidMethodV(jobject obj, jmethodID methodID, va_list args) + { functions->CallVoidMethodV(this, obj, methodID, args); } + void CallVoidMethodA(jobject obj, jmethodID methodID, jvalue* args) + { functions->CallVoidMethodA(this, obj, methodID, args); } + +#define CALL_NONVIRT_TYPE_METHOD(_jtype, _jname) \ + _jtype CallNonvirtual##_jname##Method(jobject obj, jclass clazz, \ + jmethodID methodID, ...) \ + { \ + _jtype result; \ + va_list args; \ + va_start(args, methodID); \ + result = functions->CallNonvirtual##_jname##MethodV(this, obj, \ + clazz, methodID, args); \ + va_end(args); \ + return result; \ + } +#define CALL_NONVIRT_TYPE_METHODV(_jtype, _jname) \ + _jtype CallNonvirtual##_jname##MethodV(jobject obj, jclass clazz, \ + jmethodID methodID, va_list args) \ + { return functions->CallNonvirtual##_jname##MethodV(this, obj, clazz, \ + methodID, args); } +#define CALL_NONVIRT_TYPE_METHODA(_jtype, _jname) \ + _jtype CallNonvirtual##_jname##MethodA(jobject obj, jclass clazz, \ + jmethodID methodID, jvalue* args) \ + { return functions->CallNonvirtual##_jname##MethodA(this, obj, clazz, \ + methodID, args); } + +#define CALL_NONVIRT_TYPE(_jtype, _jname) \ + CALL_NONVIRT_TYPE_METHOD(_jtype, _jname) \ + CALL_NONVIRT_TYPE_METHODV(_jtype, _jname) \ + CALL_NONVIRT_TYPE_METHODA(_jtype, _jname) + + CALL_NONVIRT_TYPE(jobject, Object) + CALL_NONVIRT_TYPE(jboolean, Boolean) + CALL_NONVIRT_TYPE(jbyte, Byte) + CALL_NONVIRT_TYPE(jchar, Char) + CALL_NONVIRT_TYPE(jshort, Short) + CALL_NONVIRT_TYPE(jint, Int) + CALL_NONVIRT_TYPE(jlong, Long) + CALL_NONVIRT_TYPE(jfloat, Float) + CALL_NONVIRT_TYPE(jdouble, Double) + + void CallNonvirtualVoidMethod(jobject obj, jclass clazz, + jmethodID methodID, ...) + { + va_list args; + va_start(args, methodID); + functions->CallNonvirtualVoidMethodV(this, obj, clazz, methodID, args); + va_end(args); + } + void CallNonvirtualVoidMethodV(jobject obj, jclass clazz, + jmethodID methodID, va_list args) + { functions->CallNonvirtualVoidMethodV(this, obj, clazz, methodID, args); } + void CallNonvirtualVoidMethodA(jobject obj, jclass clazz, + jmethodID methodID, jvalue* args) + { functions->CallNonvirtualVoidMethodA(this, obj, clazz, methodID, args); } + + jfieldID GetFieldID(jclass clazz, const char* name, const char* sig) + { return functions->GetFieldID(this, clazz, name, sig); } + + jobject GetObjectField(jobject obj, jfieldID fieldID) + { return functions->GetObjectField(this, obj, fieldID); } + jboolean GetBooleanField(jobject obj, jfieldID fieldID) + { return functions->GetBooleanField(this, obj, fieldID); } + jbyte GetByteField(jobject obj, jfieldID fieldID) + { return functions->GetByteField(this, obj, fieldID); } + jchar GetCharField(jobject obj, jfieldID fieldID) + { return functions->GetCharField(this, obj, fieldID); } + jshort GetShortField(jobject obj, jfieldID fieldID) + { return functions->GetShortField(this, obj, fieldID); } + jint GetIntField(jobject obj, jfieldID fieldID) + { return functions->GetIntField(this, obj, fieldID); } + jlong GetLongField(jobject obj, jfieldID fieldID) + { return functions->GetLongField(this, obj, fieldID); } + jfloat GetFloatField(jobject obj, jfieldID fieldID) + { return functions->GetFloatField(this, obj, fieldID); } + jdouble GetDoubleField(jobject obj, jfieldID fieldID) + { return functions->GetDoubleField(this, obj, fieldID); } + + void SetObjectField(jobject obj, jfieldID fieldID, jobject value) + { functions->SetObjectField(this, obj, fieldID, value); } + void SetBooleanField(jobject obj, jfieldID fieldID, jboolean value) + { functions->SetBooleanField(this, obj, fieldID, value); } + void SetByteField(jobject obj, jfieldID fieldID, jbyte value) + { functions->SetByteField(this, obj, fieldID, value); } + void SetCharField(jobject obj, jfieldID fieldID, jchar value) + { functions->SetCharField(this, obj, fieldID, value); } + void SetShortField(jobject obj, jfieldID fieldID, jshort value) + { functions->SetShortField(this, obj, fieldID, value); } + void SetIntField(jobject obj, jfieldID fieldID, jint value) + { functions->SetIntField(this, obj, fieldID, value); } + void SetLongField(jobject obj, jfieldID fieldID, jlong value) + { functions->SetLongField(this, obj, fieldID, value); } + void SetFloatField(jobject obj, jfieldID fieldID, jfloat value) + { functions->SetFloatField(this, obj, fieldID, value); } + void SetDoubleField(jobject obj, jfieldID fieldID, jdouble value) + { functions->SetDoubleField(this, obj, fieldID, value); } + + jmethodID GetStaticMethodID(jclass clazz, const char* name, const char* sig) + { return functions->GetStaticMethodID(this, clazz, name, sig); } + +#define CALL_STATIC_TYPE_METHOD(_jtype, _jname) \ + _jtype CallStatic##_jname##Method(jclass clazz, jmethodID methodID, \ + ...) \ + { \ + _jtype result; \ + va_list args; \ + va_start(args, methodID); \ + result = functions->CallStatic##_jname##MethodV(this, clazz, \ + methodID, args); \ + va_end(args); \ + return result; \ + } +#define CALL_STATIC_TYPE_METHODV(_jtype, _jname) \ + _jtype CallStatic##_jname##MethodV(jclass clazz, jmethodID methodID, \ + va_list args) \ + { return functions->CallStatic##_jname##MethodV(this, clazz, methodID, \ + args); } +#define CALL_STATIC_TYPE_METHODA(_jtype, _jname) \ + _jtype CallStatic##_jname##MethodA(jclass clazz, jmethodID methodID, \ + jvalue* args) \ + { return functions->CallStatic##_jname##MethodA(this, clazz, methodID, \ + args); } + +#define CALL_STATIC_TYPE(_jtype, _jname) \ + CALL_STATIC_TYPE_METHOD(_jtype, _jname) \ + CALL_STATIC_TYPE_METHODV(_jtype, _jname) \ + CALL_STATIC_TYPE_METHODA(_jtype, _jname) + + CALL_STATIC_TYPE(jobject, Object) + CALL_STATIC_TYPE(jboolean, Boolean) + CALL_STATIC_TYPE(jbyte, Byte) + CALL_STATIC_TYPE(jchar, Char) + CALL_STATIC_TYPE(jshort, Short) + CALL_STATIC_TYPE(jint, Int) + CALL_STATIC_TYPE(jlong, Long) + CALL_STATIC_TYPE(jfloat, Float) + CALL_STATIC_TYPE(jdouble, Double) + + void CallStaticVoidMethod(jclass clazz, jmethodID methodID, ...) + { + va_list args; + va_start(args, methodID); + functions->CallStaticVoidMethodV(this, clazz, methodID, args); + va_end(args); + } + void CallStaticVoidMethodV(jclass clazz, jmethodID methodID, va_list args) + { functions->CallStaticVoidMethodV(this, clazz, methodID, args); } + void CallStaticVoidMethodA(jclass clazz, jmethodID methodID, jvalue* args) + { functions->CallStaticVoidMethodA(this, clazz, methodID, args); } + + jfieldID GetStaticFieldID(jclass clazz, const char* name, const char* sig) + { return functions->GetStaticFieldID(this, clazz, name, sig); } + + jobject GetStaticObjectField(jclass clazz, jfieldID fieldID) + { return functions->GetStaticObjectField(this, clazz, fieldID); } + jboolean GetStaticBooleanField(jclass clazz, jfieldID fieldID) + { return functions->GetStaticBooleanField(this, clazz, fieldID); } + jbyte GetStaticByteField(jclass clazz, jfieldID fieldID) + { return functions->GetStaticByteField(this, clazz, fieldID); } + jchar GetStaticCharField(jclass clazz, jfieldID fieldID) + { return functions->GetStaticCharField(this, clazz, fieldID); } + jshort GetStaticShortField(jclass clazz, jfieldID fieldID) + { return functions->GetStaticShortField(this, clazz, fieldID); } + jint GetStaticIntField(jclass clazz, jfieldID fieldID) + { return functions->GetStaticIntField(this, clazz, fieldID); } + jlong GetStaticLongField(jclass clazz, jfieldID fieldID) + { return functions->GetStaticLongField(this, clazz, fieldID); } + jfloat GetStaticFloatField(jclass clazz, jfieldID fieldID) + { return functions->GetStaticFloatField(this, clazz, fieldID); } + jdouble GetStaticDoubleField(jclass clazz, jfieldID fieldID) + { return functions->GetStaticDoubleField(this, clazz, fieldID); } + + void SetStaticObjectField(jclass clazz, jfieldID fieldID, jobject value) + { functions->SetStaticObjectField(this, clazz, fieldID, value); } + void SetStaticBooleanField(jclass clazz, jfieldID fieldID, jboolean value) + { functions->SetStaticBooleanField(this, clazz, fieldID, value); } + void SetStaticByteField(jclass clazz, jfieldID fieldID, jbyte value) + { functions->SetStaticByteField(this, clazz, fieldID, value); } + void SetStaticCharField(jclass clazz, jfieldID fieldID, jchar value) + { functions->SetStaticCharField(this, clazz, fieldID, value); } + void SetStaticShortField(jclass clazz, jfieldID fieldID, jshort value) + { functions->SetStaticShortField(this, clazz, fieldID, value); } + void SetStaticIntField(jclass clazz, jfieldID fieldID, jint value) + { functions->SetStaticIntField(this, clazz, fieldID, value); } + void SetStaticLongField(jclass clazz, jfieldID fieldID, jlong value) + { functions->SetStaticLongField(this, clazz, fieldID, value); } + void SetStaticFloatField(jclass clazz, jfieldID fieldID, jfloat value) + { functions->SetStaticFloatField(this, clazz, fieldID, value); } + void SetStaticDoubleField(jclass clazz, jfieldID fieldID, jdouble value) + { functions->SetStaticDoubleField(this, clazz, fieldID, value); } + + jstring NewString(const jchar* unicodeChars, jsize len) + { return functions->NewString(this, unicodeChars, len); } + + jsize GetStringLength(jstring string) + { return functions->GetStringLength(this, string); } + + const jchar* GetStringChars(jstring string, jboolean* isCopy) + { return functions->GetStringChars(this, string, isCopy); } + + void ReleaseStringChars(jstring string, const jchar* chars) + { functions->ReleaseStringChars(this, string, chars); } + + jstring NewStringUTF(const char* bytes) + { return functions->NewStringUTF(this, bytes); } + + jsize GetStringUTFLength(jstring string) + { return functions->GetStringUTFLength(this, string); } + + const char* GetStringUTFChars(jstring string, jboolean* isCopy) + { return functions->GetStringUTFChars(this, string, isCopy); } + + void ReleaseStringUTFChars(jstring string, const char* utf) + { functions->ReleaseStringUTFChars(this, string, utf); } + + jsize GetArrayLength(jarray array) + { return functions->GetArrayLength(this, array); } + + jobjectArray NewObjectArray(jsize length, jclass elementClass, + jobject initialElement) + { return functions->NewObjectArray(this, length, elementClass, + initialElement); } + + jobject GetObjectArrayElement(jobjectArray array, jsize index) + { return functions->GetObjectArrayElement(this, array, index); } + + void SetObjectArrayElement(jobjectArray array, jsize index, jobject value) + { functions->SetObjectArrayElement(this, array, index, value); } + + jbooleanArray NewBooleanArray(jsize length) + { return functions->NewBooleanArray(this, length); } + jbyteArray NewByteArray(jsize length) + { return functions->NewByteArray(this, length); } + jcharArray NewCharArray(jsize length) + { return functions->NewCharArray(this, length); } + jshortArray NewShortArray(jsize length) + { return functions->NewShortArray(this, length); } + jintArray NewIntArray(jsize length) + { return functions->NewIntArray(this, length); } + jlongArray NewLongArray(jsize length) + { return functions->NewLongArray(this, length); } + jfloatArray NewFloatArray(jsize length) + { return functions->NewFloatArray(this, length); } + jdoubleArray NewDoubleArray(jsize length) + { return functions->NewDoubleArray(this, length); } + + jboolean* GetBooleanArrayElements(jbooleanArray array, jboolean* isCopy) + { return functions->GetBooleanArrayElements(this, array, isCopy); } + jbyte* GetByteArrayElements(jbyteArray array, jboolean* isCopy) + { return functions->GetByteArrayElements(this, array, isCopy); } + jchar* GetCharArrayElements(jcharArray array, jboolean* isCopy) + { return functions->GetCharArrayElements(this, array, isCopy); } + jshort* GetShortArrayElements(jshortArray array, jboolean* isCopy) + { return functions->GetShortArrayElements(this, array, isCopy); } + jint* GetIntArrayElements(jintArray array, jboolean* isCopy) + { return functions->GetIntArrayElements(this, array, isCopy); } + jlong* GetLongArrayElements(jlongArray array, jboolean* isCopy) + { return functions->GetLongArrayElements(this, array, isCopy); } + jfloat* GetFloatArrayElements(jfloatArray array, jboolean* isCopy) + { return functions->GetFloatArrayElements(this, array, isCopy); } + jdouble* GetDoubleArrayElements(jdoubleArray array, jboolean* isCopy) + { return functions->GetDoubleArrayElements(this, array, isCopy); } + + void ReleaseBooleanArrayElements(jbooleanArray array, jboolean* elems, + jint mode) + { functions->ReleaseBooleanArrayElements(this, array, elems, mode); } + void ReleaseByteArrayElements(jbyteArray array, jbyte* elems, + jint mode) + { functions->ReleaseByteArrayElements(this, array, elems, mode); } + void ReleaseCharArrayElements(jcharArray array, jchar* elems, + jint mode) + { functions->ReleaseCharArrayElements(this, array, elems, mode); } + void ReleaseShortArrayElements(jshortArray array, jshort* elems, + jint mode) + { functions->ReleaseShortArrayElements(this, array, elems, mode); } + void ReleaseIntArrayElements(jintArray array, jint* elems, + jint mode) + { functions->ReleaseIntArrayElements(this, array, elems, mode); } + void ReleaseLongArrayElements(jlongArray array, jlong* elems, + jint mode) + { functions->ReleaseLongArrayElements(this, array, elems, mode); } + void ReleaseFloatArrayElements(jfloatArray array, jfloat* elems, + jint mode) + { functions->ReleaseFloatArrayElements(this, array, elems, mode); } + void ReleaseDoubleArrayElements(jdoubleArray array, jdouble* elems, + jint mode) + { functions->ReleaseDoubleArrayElements(this, array, elems, mode); } + + void GetBooleanArrayRegion(jbooleanArray array, jsize start, jsize len, + jboolean* buf) + { functions->GetBooleanArrayRegion(this, array, start, len, buf); } + void GetByteArrayRegion(jbyteArray array, jsize start, jsize len, + jbyte* buf) + { functions->GetByteArrayRegion(this, array, start, len, buf); } + void GetCharArrayRegion(jcharArray array, jsize start, jsize len, + jchar* buf) + { functions->GetCharArrayRegion(this, array, start, len, buf); } + void GetShortArrayRegion(jshortArray array, jsize start, jsize len, + jshort* buf) + { functions->GetShortArrayRegion(this, array, start, len, buf); } + void GetIntArrayRegion(jintArray array, jsize start, jsize len, + jint* buf) + { functions->GetIntArrayRegion(this, array, start, len, buf); } + void GetLongArrayRegion(jlongArray array, jsize start, jsize len, + jlong* buf) + { functions->GetLongArrayRegion(this, array, start, len, buf); } + void GetFloatArrayRegion(jfloatArray array, jsize start, jsize len, + jfloat* buf) + { functions->GetFloatArrayRegion(this, array, start, len, buf); } + void GetDoubleArrayRegion(jdoubleArray array, jsize start, jsize len, + jdouble* buf) + { functions->GetDoubleArrayRegion(this, array, start, len, buf); } + + void SetBooleanArrayRegion(jbooleanArray array, jsize start, jsize len, + const jboolean* buf) + { functions->SetBooleanArrayRegion(this, array, start, len, buf); } + void SetByteArrayRegion(jbyteArray array, jsize start, jsize len, + const jbyte* buf) + { functions->SetByteArrayRegion(this, array, start, len, buf); } + void SetCharArrayRegion(jcharArray array, jsize start, jsize len, + const jchar* buf) + { functions->SetCharArrayRegion(this, array, start, len, buf); } + void SetShortArrayRegion(jshortArray array, jsize start, jsize len, + const jshort* buf) + { functions->SetShortArrayRegion(this, array, start, len, buf); } + void SetIntArrayRegion(jintArray array, jsize start, jsize len, + const jint* buf) + { functions->SetIntArrayRegion(this, array, start, len, buf); } + void SetLongArrayRegion(jlongArray array, jsize start, jsize len, + const jlong* buf) + { functions->SetLongArrayRegion(this, array, start, len, buf); } + void SetFloatArrayRegion(jfloatArray array, jsize start, jsize len, + const jfloat* buf) + { functions->SetFloatArrayRegion(this, array, start, len, buf); } + void SetDoubleArrayRegion(jdoubleArray array, jsize start, jsize len, + const jdouble* buf) + { functions->SetDoubleArrayRegion(this, array, start, len, buf); } + + jint RegisterNatives(jclass clazz, const JNINativeMethod* methods, + jint nMethods) + { return functions->RegisterNatives(this, clazz, methods, nMethods); } + + jint UnregisterNatives(jclass clazz) + { return functions->UnregisterNatives(this, clazz); } + + jint MonitorEnter(jobject obj) + { return functions->MonitorEnter(this, obj); } + + jint MonitorExit(jobject obj) + { return functions->MonitorExit(this, obj); } + + jint GetJavaVM(JavaVM** vm) + { return functions->GetJavaVM(this, vm); } + + void GetStringRegion(jstring str, jsize start, jsize len, jchar* buf) + { functions->GetStringRegion(this, str, start, len, buf); } + + void GetStringUTFRegion(jstring str, jsize start, jsize len, char* buf) + { return functions->GetStringUTFRegion(this, str, start, len, buf); } + + void* GetPrimitiveArrayCritical(jarray array, jboolean* isCopy) + { return functions->GetPrimitiveArrayCritical(this, array, isCopy); } + + void ReleasePrimitiveArrayCritical(jarray array, void* carray, jint mode) + { functions->ReleasePrimitiveArrayCritical(this, array, carray, mode); } + + const jchar* GetStringCritical(jstring string, jboolean* isCopy) + { return functions->GetStringCritical(this, string, isCopy); } + + void ReleaseStringCritical(jstring string, const jchar* carray) + { functions->ReleaseStringCritical(this, string, carray); } + + jweak NewWeakGlobalRef(jobject obj) + { return functions->NewWeakGlobalRef(this, obj); } + + void DeleteWeakGlobalRef(jweak obj) + { functions->DeleteWeakGlobalRef(this, obj); } + + jboolean ExceptionCheck() + { return functions->ExceptionCheck(this); } + + jobject NewDirectByteBuffer(void* address, jlong capacity) + { return functions->NewDirectByteBuffer(this, address, capacity); } + + void* GetDirectBufferAddress(jobject buf) + { return functions->GetDirectBufferAddress(this, buf); } + + jlong GetDirectBufferCapacity(jobject buf) + { return functions->GetDirectBufferCapacity(this, buf); } + + /* added in JNI 1.6 */ + jobjectRefType GetObjectRefType(jobject obj) + { return functions->GetObjectRefType(this, obj); } +#endif /*__cplusplus*/ +}; + + +/* + * JNI invocation interface. + */ +struct JNIInvokeInterface { + void* reserved0; + void* reserved1; + void* reserved2; + + jint (*DestroyJavaVM)(JavaVM*); + jint (*AttachCurrentThread)(JavaVM*, JNIEnv**, void*); + jint (*DetachCurrentThread)(JavaVM*); + jint (*GetEnv)(JavaVM*, void**, jint); + jint (*AttachCurrentThreadAsDaemon)(JavaVM*, JNIEnv**, void*); +}; + +/* + * C++ version. + */ +struct _JavaVM { + const struct JNIInvokeInterface* functions; + +#if defined(__cplusplus) + jint DestroyJavaVM() + { return functions->DestroyJavaVM(this); } + jint AttachCurrentThread(JNIEnv** p_env, void* thr_args) + { return functions->AttachCurrentThread(this, p_env, thr_args); } + jint DetachCurrentThread() + { return functions->DetachCurrentThread(this); } + jint GetEnv(void** env, jint version) + { return functions->GetEnv(this, env, version); } + jint AttachCurrentThreadAsDaemon(JNIEnv** p_env, void* thr_args) + { return functions->AttachCurrentThreadAsDaemon(this, p_env, thr_args); } +#endif /*__cplusplus*/ +}; + +struct JavaVMAttachArgs { + jint version; /* must be >= JNI_VERSION_1_2 */ + const char* name; /* NULL or name of thread as modified UTF-8 str */ + jobject group; /* global ref of a ThreadGroup object, or NULL */ +}; +typedef struct JavaVMAttachArgs JavaVMAttachArgs; + +/* + * JNI 1.2+ initialization. (As of 1.6, the pre-1.2 structures are no + * longer supported.) + */ +typedef struct JavaVMOption { + const char* optionString; + void* extraInfo; +} JavaVMOption; + +typedef struct JavaVMInitArgs { + jint version; /* use JNI_VERSION_1_2 or later */ + + jint nOptions; + JavaVMOption* options; + jboolean ignoreUnrecognized; +} JavaVMInitArgs; + +#ifdef __cplusplus +extern "C" { +#endif +/* + * VM initialization functions. + * + * Note these are the only symbols exported for JNI by the VM. + */ +jint JNI_GetDefaultJavaVMInitArgs(void*); +jint JNI_CreateJavaVM(JavaVM**, JNIEnv**, void*); +jint JNI_GetCreatedJavaVMs(JavaVM**, jsize, jsize*); + +/* + * Prototypes for functions exported by loadable shared libs. These are + * called by JNI, not provided by JNI. + */ +jint JNI_OnLoad(JavaVM* vm, void* reserved); +void JNI_OnUnload(JavaVM* vm, void* reserved); + +#ifdef __cplusplus +} +#endif + + +/* + * Manifest constants. + */ +#define JNI_FALSE 0 +#define JNI_TRUE 1 + +#define JNI_VERSION_1_1 0x00010001 +#define JNI_VERSION_1_2 0x00010002 +#define JNI_VERSION_1_4 0x00010004 +#define JNI_VERSION_1_6 0x00010006 + +#define JNI_OK (0) /* no error */ +#define JNI_ERR (-1) /* generic error */ +#define JNI_EDETACHED (-2) /* thread detached from the VM */ +#define JNI_EVERSION (-3) /* JNI version error */ + +#define JNI_COMMIT 1 /* copy content, do not free buffer */ +#define JNI_ABORT 2 /* free buffer w/o copying back */ + +/* need these for Windows-aware headers */ +#define JNIIMPORT +#define JNIEXPORT +#define JNICALL + +#endif /*_JNI_H*/ diff --git a/src/fakejvm/jvm.c b/src/fakejvm/jvm.c new file mode 100644 index 0000000..3ce783b --- /dev/null +++ b/src/fakejvm/jvm.c @@ -0,0 +1,1778 @@ +#include +#include +#include +#include +#include +#include +#include "jvm.h" + +#define ARRAY_SIZE(x) (sizeof(x) / sizeof(x[0])) +#define container_of(ptr, type, member) ((type *)((char *)(1 ? (ptr) : &((type *)0)->member) - offsetof(type, member))) + +struct jstring { + char *data; +}; + +struct array { + void *data; + long element_size; + long length; // in elements +}; + +struct jclass { + char *name; +}; + +struct _jmethodID { + jclass clazz; + char *name; + char *sig; +}; + +struct jvm* +jnienv_get_jvm(JNIEnv *env) +{ + return container_of(env, struct jvm, env); +} + +struct jvm* +javavm_get_jvm(JavaVM *vm) +{ + return container_of(vm, struct jvm, vm); +} + +static jint +JNIEnv_GetVersion(JNIEnv * p0) +{ + return 0; +} + +static jclass +JNIEnv_DefineClass(JNIEnv* p0, const char* p1, jobject p2, const jbyte* p3, jsize p4) +{ + return NULL; +} + +static jclass +JNIEnv_FindClass(JNIEnv* p0, const char* p1) +{ + struct jclass *class = malloc(sizeof(struct jclass)); + class->name = strdup(p1); + return class; +} + +static jmethodID +JNIEnv_FromReflectedMethod(JNIEnv* p0, jobject p1) +{ + return NULL; +} + +static jfieldID +JNIEnv_FromReflectedField(JNIEnv* p0, jobject p1) +{ + return NULL; +} + +static jobject +JNIEnv_ToReflectedMethod(JNIEnv* p0, jclass p1, jmethodID p2, jboolean p3) +{ + return NULL; +} + +static jclass +JNIEnv_GetSuperclass(JNIEnv* p0, jclass p1) +{ + return NULL; +} + +static jboolean +JNIEnv_IsAssignableFrom(JNIEnv* p0, jclass p1, jclass p2) +{ + return 0; +} + +static jobject +JNIEnv_ToReflectedField(JNIEnv* p0, jclass p1, jfieldID p2, jboolean p3) +{ + return NULL; +} + +static jint +JNIEnv_Throw(JNIEnv* p0, jthrowable p1) +{ + return 0; +} + +static jint +JNIEnv_ThrowNew(JNIEnv * p0, jclass p1, const char * p2) +{ + return 0; +} + +static jthrowable +JNIEnv_ExceptionOccurred(JNIEnv* p0) +{ + return NULL; +} + +static void +JNIEnv_ExceptionDescribe(JNIEnv* p0) +{ +} + +static void +JNIEnv_ExceptionClear(JNIEnv* p0) +{ +} + +static void +JNIEnv_FatalError(JNIEnv* p0, const char* p1) +{ +} + +static jint +JNIEnv_PushLocalFrame(JNIEnv* p0, jint p1) +{ + return 0; +} + +static jobject +JNIEnv_PopLocalFrame(JNIEnv* p0, jobject p1) +{ + return NULL; +} + +static jobject +JNIEnv_NewGlobalRef(JNIEnv* p0, jobject p1) +{ + return p1; +} + +static void +JNIEnv_DeleteGlobalRef(JNIEnv* p0, jobject p1) +{ +} + +static void +JNIEnv_DeleteLocalRef(JNIEnv* p0, jobject p1) +{ +} + +static jboolean +JNIEnv_IsSameObject(JNIEnv* p0, jobject p1, jobject p2) +{ + return 0; +} + +static jobject +JNIEnv_NewLocalRef(JNIEnv* p0, jobject p1) +{ + return NULL; +} + +static jint +JNIEnv_EnsureLocalCapacity(JNIEnv* p0, jint p1) +{ + return 0; +} + +static jobject +JNIEnv_AllocObject(JNIEnv* p0, jclass p1) +{ + return NULL; +} + +static jobject +JNIEnv_NewObject(JNIEnv* p0, jclass p1, jmethodID p2, ...) +{ + return NULL; +} + +static jobject +JNIEnv_NewObjectV(JNIEnv *env, jclass p1, jmethodID p2, va_list p3) +{ + return 0; +} + +static jobject +JNIEnv_NewObjectA(JNIEnv* p0, jclass p1, jmethodID p2, jvalue* p3) +{ + return NULL; +} + +static jclass +JNIEnv_GetObjectClass(JNIEnv* env, jobject p1) +{ + return 0; +} + +static jboolean +JNIEnv_IsInstanceOf(JNIEnv* p0, jobject p1, jclass p2) +{ + return 0; +} + +static jmethodID +jnienv_make_method(jclass clazz, const char *name, const char *sig) +{ + jmethodID id = malloc(sizeof(struct _jmethodID)); + id->clazz = clazz; + id->name = strdup(name); + id->sig = strdup(sig); + return id; +} + +static jmethodID +JNIEnv_GetMethodID(JNIEnv* p0, jclass clazz, const char* name, const char* sig) +{ + return jnienv_make_method(clazz, name, sig); +} + +static jobject +JNIEnv_CallObjectMethod(JNIEnv* p0, jobject p1, jmethodID p2, ...) +{ + return NULL; +} + +static jobject +JNIEnv_CallObjectMethodV(JNIEnv *env, jobject p1, jmethodID p2, va_list p3) +{ + return NULL; +} + +static jobject +JNIEnv_CallObjectMethodA(JNIEnv* p0, jobject p1, jmethodID p2, jvalue* p3) +{ + return NULL; +} + +static jboolean +JNIEnv_CallBooleanMethod(JNIEnv* p0, jobject p1, jmethodID p2, ...) +{ + return 0; +} + +static jboolean +JNIEnv_CallBooleanMethodV(JNIEnv* p0, jobject p1, jmethodID p2, va_list p3) +{ + return 0; +} + +static jboolean +JNIEnv_CallBooleanMethodA(JNIEnv* p0, jobject p1, jmethodID p2, jvalue* p3) +{ + return 0; +} + +static jbyte +JNIEnv_CallByteMethod(JNIEnv* p0, jobject p1, jmethodID p2, ...) +{ + return 0; +} + +static jbyte +JNIEnv_CallByteMethodV(JNIEnv* p0, jobject p1, jmethodID p2, va_list p3) +{ + return 0; +} + +static jbyte +JNIEnv_CallByteMethodA(JNIEnv* p0, jobject p1, jmethodID p2, jvalue* p3) +{ + return 0; +} + +static jchar +JNIEnv_CallCharMethod(JNIEnv* p0, jobject p1, jmethodID p2, ...) +{ + return 0; +} + +static jchar +JNIEnv_CallCharMethodV(JNIEnv* p0, jobject p1, jmethodID p2, va_list p3) +{ + return 0; +} + +static jchar +JNIEnv_CallCharMethodA(JNIEnv* p0, jobject p1, jmethodID p2, jvalue* p3) +{ + return 0; +} + +static jshort +JNIEnv_CallShortMethod(JNIEnv* p0, jobject p1, jmethodID p2, ...) +{ + return 0; +} + +static jshort +JNIEnv_CallShortMethodV(JNIEnv* p0, jobject p1, jmethodID p2, va_list p3) +{ + return 0; +} + +static jshort +JNIEnv_CallShortMethodA(JNIEnv* p0, jobject p1, jmethodID p2, jvalue* p3) +{ + return 0; +} + +static jint +JNIEnv_CallIntMethod(JNIEnv* p0, jobject p1, jmethodID p2, ...) +{ + return 0; +} + +static jint +JNIEnv_CallIntMethodV(JNIEnv* p0, jobject p1, jmethodID p2, va_list p3) +{ + return 0; +} + +static jint +JNIEnv_CallIntMethodA(JNIEnv* p0, jobject p1, jmethodID p2, jvalue* p3) +{ + return 0; +} + +static jlong +JNIEnv_CallLongMethod(JNIEnv* p0, jobject p1, jmethodID p2, ...) +{ + return 0; +} + +static jlong +JNIEnv_CallLongMethodV(JNIEnv* p0, jobject p1, jmethodID p2, va_list p3) +{ + return 0; +} + +static jlong +JNIEnv_CallLongMethodA(JNIEnv* p0, jobject p1, jmethodID p2, jvalue* p3) +{ + return 0; +} + +static jfloat +JNIEnv_CallFloatMethod(JNIEnv* p0, jobject p1, jmethodID p2, ...) +{ + return 0; +} + +static jfloat +JNIEnv_CallFloatMethodV(JNIEnv* p0, jobject p1, jmethodID p2, va_list p3) +{ + return 0; +} + +static jfloat +JNIEnv_CallFloatMethodA(JNIEnv* p0, jobject p1, jmethodID p2, jvalue* p3) +{ + return 0; +} + +static jdouble +JNIEnv_CallDoubleMethod(JNIEnv* p0, jobject p1, jmethodID p2, ...) +{ + return 0; +} + +static jdouble +JNIEnv_CallDoubleMethodV(JNIEnv* p0, jobject p1, jmethodID p2, va_list p3) +{ + return 0; +} + +static jdouble +JNIEnv_CallDoubleMethodA(JNIEnv* p0, jobject p1, jmethodID p2, jvalue* p3) +{ + return 0; +} + +static void +JNIEnv_CallVoidMethod(JNIEnv* p0, jobject p1, jmethodID p2, ...) +{ +} + +static void +JNIEnv_CallVoidMethodV(JNIEnv* p0, jobject p1, jmethodID p2, va_list p3) +{ +} + +static void +JNIEnv_CallVoidMethodA(JNIEnv* p0, jobject p1, jmethodID p2, jvalue* p3) +{ +} + +static jobject +JNIEnv_CallNonvirtualObjectMethod(JNIEnv* p0, jobject p1, jclass p2, jmethodID p3, ...) +{ + return NULL; +} + +static jobject +JNIEnv_CallNonvirtualObjectMethodV(JNIEnv* p0, jobject p1, jclass p2, jmethodID p3, va_list p4) +{ + return NULL; +} + +static jobject +JNIEnv_CallNonvirtualObjectMethodA(JNIEnv* p0, jobject p1, jclass p2, jmethodID p3, jvalue* p4) +{ + return NULL; +} + +static jboolean +JNIEnv_CallNonvirtualBooleanMethod(JNIEnv* p0, jobject p1, jclass p2, jmethodID p3, ...) +{ + return 0; +} + +static jboolean +JNIEnv_CallNonvirtualBooleanMethodV(JNIEnv* p0, jobject p1, jclass p2, jmethodID p3, va_list p4) +{ + return 0; +} + +static jboolean +JNIEnv_CallNonvirtualBooleanMethodA(JNIEnv* p0, jobject p1, jclass p2, jmethodID p3, jvalue* p4) +{ + return 0; +} + +static jbyte +JNIEnv_CallNonvirtualByteMethod(JNIEnv* p0, jobject p1, jclass p2, jmethodID p3, ...) +{ + return 0; +} + +static jbyte +JNIEnv_CallNonvirtualByteMethodV(JNIEnv* p0, jobject p1, jclass p2, jmethodID p3, va_list p4) +{ + return 0; +} + +static jbyte +JNIEnv_CallNonvirtualByteMethodA(JNIEnv* p0, jobject p1, jclass p2, jmethodID p3, jvalue* p4) +{ + return 0; +} + +static jchar +JNIEnv_CallNonvirtualCharMethod(JNIEnv* p0, jobject p1, jclass p2, jmethodID p3, ...) +{ + return 0; +} + +static jchar +JNIEnv_CallNonvirtualCharMethodV(JNIEnv* p0, jobject p1, jclass p2, jmethodID p3, va_list p4) +{ + return 0; +} + +static jchar +JNIEnv_CallNonvirtualCharMethodA(JNIEnv* p0, jobject p1, jclass p2, jmethodID p3, jvalue* p4) +{ + return 0; +} + +static jshort +JNIEnv_CallNonvirtualShortMethod(JNIEnv* p0, jobject p1, jclass p2, jmethodID p3, ...) +{ + return 0; +} + +static jshort +JNIEnv_CallNonvirtualShortMethodV(JNIEnv* p0, jobject p1, jclass p2, jmethodID p3, va_list p4) +{ + return 0; +} + +static jshort +JNIEnv_CallNonvirtualShortMethodA(JNIEnv* p0, jobject p1, jclass p2, jmethodID p3, jvalue* p4) +{ + return 0; +} + +static jint +JNIEnv_CallNonvirtualIntMethod(JNIEnv* p0, jobject p1, jclass p2, jmethodID p3, ...) +{ + return 0; +} + +static jint +JNIEnv_CallNonvirtualIntMethodV(JNIEnv* p0, jobject p1, jclass p2, jmethodID p3, va_list p4) +{ + return 0; +} + +static jint +JNIEnv_CallNonvirtualIntMethodA(JNIEnv* p0, jobject p1, jclass p2, jmethodID p3, jvalue* p4) +{ + return 0; +} + +static jlong +JNIEnv_CallNonvirtualLongMethod(JNIEnv* p0, jobject p1, jclass p2, jmethodID p3, ...) +{ + return 0; +} + +static jlong +JNIEnv_CallNonvirtualLongMethodV(JNIEnv* p0, jobject p1, jclass p2, jmethodID p3, va_list p4) +{ + return 0; +} + +static jlong +JNIEnv_CallNonvirtualLongMethodA(JNIEnv* p0, jobject p1, jclass p2, jmethodID p3, jvalue* p4) +{ + return 0; +} + +static jfloat +JNIEnv_CallNonvirtualFloatMethod(JNIEnv* p0, jobject p1, jclass p2, jmethodID p3, ...) +{ + return 0; +} + +static jfloat +JNIEnv_CallNonvirtualFloatMethodV(JNIEnv* p0, jobject p1, jclass p2, jmethodID p3, va_list p4) +{ + return 0; +} + +static jfloat +JNIEnv_CallNonvirtualFloatMethodA(JNIEnv* p0, jobject p1, jclass p2, jmethodID p3, jvalue* p4) +{ + return 0; +} + +static jdouble +JNIEnv_CallNonvirtualDoubleMethod(JNIEnv* p0, jobject p1, jclass p2, jmethodID p3, ...) +{ + return 0; +} + +static jdouble +JNIEnv_CallNonvirtualDoubleMethodV(JNIEnv* p0, jobject p1, jclass p2, jmethodID p3, va_list p4) +{ + return 0; +} + +static jdouble +JNIEnv_CallNonvirtualDoubleMethodA(JNIEnv* p0, jobject p1, jclass p2, jmethodID p3, jvalue* p4) +{ + return 0; +} + +static void +JNIEnv_CallNonvirtualVoidMethod(JNIEnv* p0, jobject p1, jclass p2, jmethodID p3, ...) +{ +} + +static void +JNIEnv_CallNonvirtualVoidMethodV(JNIEnv* p0, jobject p1, jclass p2, jmethodID p3, va_list p4) +{ +} + +static void +JNIEnv_CallNonvirtualVoidMethodA(JNIEnv* p0, jobject p1, jclass p2, jmethodID p3, jvalue* p4) +{ +} + +static jfieldID +jnienv_make_fieldid(jclass clazz, const char *name, const char *sig) +{ + return (jfieldID)jnienv_make_method(clazz,name,sig); +} + +static jfieldID +JNIEnv_GetFieldID(JNIEnv* p0, jclass clazz, const char* name, const char* sig) +{ + struct jclass *class = (struct jclass*)clazz; + return jnienv_make_fieldid(clazz, name, sig); +} + +static jobject +JNIEnv_GetObjectField(JNIEnv* p0, jobject p1, jfieldID p2) +{ + return NULL; +} + +static jboolean +JNIEnv_GetBooleanField(JNIEnv* p0, jobject p1, jfieldID p2) +{ + return 0; +} + +static jbyte +JNIEnv_GetByteField(JNIEnv* p0, jobject p1, jfieldID p2) +{ + return 0; +} + +static jchar +JNIEnv_GetCharField(JNIEnv* p0, jobject p1, jfieldID p2) +{ + return 0; +} + +static jshort +JNIEnv_GetShortField(JNIEnv* p0, jobject p1, jfieldID p2) +{ + return 0; +} + +static jint +JNIEnv_GetIntField(JNIEnv* p0, jobject p1, jfieldID p2) +{ + return 0; +} + +static jlong +JNIEnv_GetLongField(JNIEnv* p0, jobject p1, jfieldID p2) +{ + return 0; +} + +static jfloat +JNIEnv_GetFloatField(JNIEnv* p0, jobject p1, jfieldID p2) +{ + return 0; +} + +static jdouble +JNIEnv_GetDoubleField(JNIEnv* p0, jobject p1, jfieldID p2) +{ + return 0; +} + +static void +JNIEnv_SetObjectField(JNIEnv* p0, jobject p1, jfieldID p2, jobject p3) +{ +} + +static void +JNIEnv_SetBooleanField(JNIEnv* p0, jobject p1, jfieldID p2, jboolean p3) +{ +} + +static void +JNIEnv_SetByteField(JNIEnv* p0, jobject p1, jfieldID p2, jbyte p3) +{ +} + +static void +JNIEnv_SetCharField(JNIEnv* p0, jobject p1, jfieldID p2, jchar p3) +{ +} + +static void +JNIEnv_SetShortField(JNIEnv* p0, jobject p1, jfieldID p2, jshort p3) +{ +} + +static void +JNIEnv_SetIntField(JNIEnv* p0, jobject p1, jfieldID p2, jint p3) +{ +} + +static void +JNIEnv_SetLongField(JNIEnv* p0, jobject p1, jfieldID p2, jlong p3) +{ +} + +static void +JNIEnv_SetFloatField(JNIEnv* p0, jobject p1, jfieldID p2, jfloat p3) +{ +} + +static void +JNIEnv_SetDoubleField(JNIEnv* p0, jobject p1, jfieldID p2, jdouble p3) +{ +} + +static jmethodID +JNIEnv_GetStaticMethodID(JNIEnv* p0, jclass clazz, const char* name, const char* sig) +{ + return jnienv_make_method(clazz, name, sig); +} + +static jobject +JNIEnv_CallStaticObjectMethod(JNIEnv* p0, jclass p1, jmethodID p2, ...) +{ + return NULL; +} + +static jobject +JNIEnv_CallStaticObjectMethodV(JNIEnv* p0, jclass p1, jmethodID p2, va_list p3) +{ + return NULL; +} + +static jobject +JNIEnv_CallStaticObjectMethodA(JNIEnv* p0, jclass p1, jmethodID p2, jvalue* p3) +{ + return NULL; +} + +static jboolean +JNIEnv_CallStaticBooleanMethod(JNIEnv* p0, jclass p1, jmethodID p2, ...) +{ + return 0; +} + +static jboolean +JNIEnv_CallStaticBooleanMethodV(JNIEnv* p0, jclass p1, jmethodID p2, va_list p3) +{ + return 0; +} + +static jboolean +JNIEnv_CallStaticBooleanMethodA(JNIEnv* p0, jclass p1, jmethodID p2, jvalue* p3) +{ + return 0; +} + +static jbyte +JNIEnv_CallStaticByteMethod(JNIEnv* p0, jclass p1, jmethodID p2, ...) +{ + return 0; +} + +static jbyte +JNIEnv_CallStaticByteMethodV(JNIEnv* p0, jclass p1, jmethodID p2, va_list p3) +{ + return 0; +} + +static jbyte +JNIEnv_CallStaticByteMethodA(JNIEnv* p0, jclass p1, jmethodID p2, jvalue* p3) +{ + return 0; +} + +static jchar +JNIEnv_CallStaticCharMethod(JNIEnv* p0, jclass p1, jmethodID p2, ...) +{ + return 0; +} + +static jchar +JNIEnv_CallStaticCharMethodV(JNIEnv* p0, jclass p1, jmethodID p2, va_list p3) +{ + return 0; +} + +static jchar +JNIEnv_CallStaticCharMethodA(JNIEnv* p0, jclass p1, jmethodID p2, jvalue* p3) +{ + return 0; +} + +static jshort +JNIEnv_CallStaticShortMethod(JNIEnv* p0, jclass p1, jmethodID p2, ...) +{ + return 0; +} + +static jshort +JNIEnv_CallStaticShortMethodV(JNIEnv* p0, jclass p1, jmethodID p2, va_list p3) +{ + return 0; +} + +static jshort +JNIEnv_CallStaticShortMethodA(JNIEnv* p0, jclass p1, jmethodID p2, jvalue* p3) +{ + return 0; +} + +static jint +JNIEnv_CallStaticIntMethod(JNIEnv* p0, jclass p1, jmethodID p2, ...) +{ + return 0; +} + +static jint +JNIEnv_CallStaticIntMethodV(JNIEnv* p0, jclass p1, jmethodID p2, va_list p3) +{ + return 0; +} + +static jint +JNIEnv_CallStaticIntMethodA(JNIEnv* p0, jclass p1, jmethodID p2, jvalue* p3) +{ + return 0; +} + +static jlong +JNIEnv_CallStaticLongMethod(JNIEnv* p0, jclass p1, jmethodID p2, ...) +{ + return 0; +} + +static jlong +JNIEnv_CallStaticLongMethodV(JNIEnv* p0, jclass p1, jmethodID p2, va_list p3) +{ + return 0; +} + +static jlong +JNIEnv_CallStaticLongMethodA(JNIEnv* p0, jclass p1, jmethodID p2, jvalue* p3) +{ + return 0; +} + +static jfloat +JNIEnv_CallStaticFloatMethod(JNIEnv* p0, jclass p1, jmethodID p2, ...) +{ + return 0; +} + +static jfloat +JNIEnv_CallStaticFloatMethodV(JNIEnv* p0, jclass p1, jmethodID p2, va_list p3) +{ + return 0; +} + +static jfloat +JNIEnv_CallStaticFloatMethodA(JNIEnv* p0, jclass p1, jmethodID p2, jvalue* p3) +{ + return 0; +} + +static jdouble +JNIEnv_CallStaticDoubleMethod(JNIEnv* p0, jclass p1, jmethodID p2, ...) +{ + return 0; +} + +static jdouble +JNIEnv_CallStaticDoubleMethodV(JNIEnv* p0, jclass p1, jmethodID p2, va_list p3) +{ + return 0; +} + +static jdouble +JNIEnv_CallStaticDoubleMethodA(JNIEnv* p0, jclass p1, jmethodID p2, jvalue* p3) +{ + return 0; +} + +static void +JNIEnv_CallStaticVoidMethod(JNIEnv* p0, jclass p1, jmethodID p2, ...) +{ +} + +static void +JNIEnv_CallStaticVoidMethodV(JNIEnv* p0, jclass p1, jmethodID p2, va_list p3) +{ +} + +static void +JNIEnv_CallStaticVoidMethodA(JNIEnv* p0, jclass p1, jmethodID p2, jvalue* p3) +{ +} + +static jfieldID +JNIEnv_GetStaticFieldID(JNIEnv* p0, jclass clazz, const char* name, const char* sig) +{ + struct jclass *class = (struct jclass*)clazz; + return jnienv_make_fieldid(clazz, name, sig); +} + +static jobject +JNIEnv_GetStaticObjectField(JNIEnv* p0, jclass p1, jfieldID p2) +{ + return NULL; +} + +static jboolean +JNIEnv_GetStaticBooleanField(JNIEnv* p0, jclass p1, jfieldID p2) +{ + return 0; +} + +static jbyte +JNIEnv_GetStaticByteField(JNIEnv* p0, jclass p1, jfieldID p2) +{ + return 0; +} + +static jchar +JNIEnv_GetStaticCharField(JNIEnv* p0, jclass p1, jfieldID p2) +{ + return 0; +} + +static jshort +JNIEnv_GetStaticShortField(JNIEnv* p0, jclass p1, jfieldID p2) +{ + return 0; +} + +static jint +JNIEnv_GetStaticIntField(JNIEnv* p0, jclass p1, jfieldID p2) +{ + return 0; +} + +static jlong +JNIEnv_GetStaticLongField(JNIEnv* p0, jclass p1, jfieldID p2) +{ + return 0; +} + +static jfloat +JNIEnv_GetStaticFloatField(JNIEnv* p0, jclass p1, jfieldID p2) +{ + return 0; +} + +static jdouble +JNIEnv_GetStaticDoubleField(JNIEnv* p0, jclass p1, jfieldID p2) +{ + return 0; +} + +static void +JNIEnv_SetStaticObjectField(JNIEnv* p0, jclass p1, jfieldID p2, jobject p3) +{ +} + +static void +JNIEnv_SetStaticBooleanField(JNIEnv* p0, jclass p1, jfieldID p2, jboolean p3) +{ +} + +static void +JNIEnv_SetStaticByteField(JNIEnv* p0, jclass p1, jfieldID p2, jbyte p3) +{ +} + +static void +JNIEnv_SetStaticCharField(JNIEnv* p0, jclass p1, jfieldID p2, jchar p3) +{ +} + +static void +JNIEnv_SetStaticShortField(JNIEnv* p0, jclass p1, jfieldID p2, jshort p3) +{ +} + +static void +JNIEnv_SetStaticIntField(JNIEnv* p0, jclass p1, jfieldID p2, jint p3) +{ +} + +static void +JNIEnv_SetStaticLongField(JNIEnv* p0, jclass p1, jfieldID p2, jlong p3) +{ +} + +static void +JNIEnv_SetStaticFloatField(JNIEnv* p0, jclass p1, jfieldID p2, jfloat p3) +{ +} + +static void +JNIEnv_SetStaticDoubleField(JNIEnv* p0, jclass p1, jfieldID p2, jdouble p3) +{ +} + +static jstring +JNIEnv_NewString(JNIEnv* p0, const jchar* p1, jsize p2) +{ + char *str = calloc(1, p2 + 1); + memcpy(str, p1, p2); + struct jstring *result = malloc(sizeof(struct jstring)); + result->data = str; + return result; +} + +static jsize +JNIEnv_GetStringLength(JNIEnv* p0, jstring p1) +{ + return 0; +} + +const jchar* +JNIEnv_GetStringChars(JNIEnv* p0, jstring p1, jboolean* p2) +{ + return NULL; +} + +static void +JNIEnv_ReleaseStringChars(JNIEnv* p0, jstring p1, const jchar* p2) +{ +} + +static jstring +JNIEnv_NewStringUTF(JNIEnv* p0, const char* p1) +{ + struct jstring *result = malloc(sizeof(struct jstring)); + result->data = strdup((p1 ? p1 : "(null)")); + return result; +} + +static jsize +JNIEnv_GetStringUTFLength(JNIEnv* p0, jstring p1) +{ + struct jstring *result = p1; + return strlen(result->data); +} + +static jsize +JNIEnv_GetArrayLength(JNIEnv* env, jarray p1) +{ + return (p1 ? ((struct array*)p1)->length : 0); +} + +static jobjectArray +JNIEnv_NewObjectArray(JNIEnv* p0, jsize p1, jclass p2, jobject p3) +{ + return NULL; +} + +static jobject +JNIEnv_GetObjectArrayElement(JNIEnv* p0, jobjectArray p1, jsize p2) +{ + return NULL; +} + +static void +JNIEnv_SetObjectArrayElement(JNIEnv* p0, jobjectArray p1, jsize p2, jobject p3) +{ +} + +static void * +new_array(jsize len, jsize element_size) +{ + if (len <= 0) + return NULL; + + struct array *array; + if (!(array = malloc(sizeof(*array)))) + return NULL; + + if (!(array->data = calloc(len, element_size))) { + free(array); + return NULL; + } + + array->element_size = element_size; + array->length = len; + return array; +} + +static jbooleanArray +JNIEnv_NewBooleanArray(JNIEnv* p0, jsize p1) +{ + return new_array(p1, sizeof(jboolean)); +} + + +static jbyteArray +JNIEnv_NewByteArray(JNIEnv* p0, jsize p1) +{ + return new_array(p1, sizeof(jbyte)); +} + +static jcharArray +JNIEnv_NewCharArray(JNIEnv* p0, jsize p1) +{ + return new_array(p1, sizeof(jchar)); +} + +static jshortArray +JNIEnv_NewShortArray(JNIEnv* p0, jsize p1) +{ + return new_array(p1, sizeof(jshort)); +} + +static jintArray +JNIEnv_NewIntArray(JNIEnv* p0, jsize p1) +{ + return new_array(p1, sizeof(jint)); +} + +static jlongArray +JNIEnv_NewLongArray(JNIEnv* p0, jsize p1) +{ + return new_array(p1, sizeof(jlong)); +} + +static jfloatArray +JNIEnv_NewFloatArray(JNIEnv* p0, jsize p1) +{ + return new_array(p1, sizeof(jfloat)); +} + +static jdoubleArray +JNIEnv_NewDoubleArray(JNIEnv* p0, jsize p1) +{ + return new_array(p1, sizeof(jdouble)); +} + +static void* +get_array_elements(JNIEnv *env, void *array, jboolean *isCopy) +{ + if (isCopy) + *isCopy = JNI_FALSE; + + return (array ? ((struct array*)array)->data : NULL); +} + +static jboolean* +JNIEnv_GetBooleanArrayElements(JNIEnv* p0, jbooleanArray p1, jboolean* p2) +{ + return get_array_elements(p0, p1, p2); +} + +static jbyte* +JNIEnv_GetByteArrayElements(JNIEnv* p0, jbyteArray p1, jboolean* p2) +{ + return get_array_elements(p0, p1, p2); +} + +static jchar* +JNIEnv_GetCharArrayElements(JNIEnv* p0, jcharArray p1, jboolean* p2) +{ + return get_array_elements(p0, p1, p2); +} + +static jshort* +JNIEnv_GetShortArrayElements(JNIEnv* p0, jshortArray p1, jboolean* p2) +{ + return get_array_elements(p0, p1, p2); +} + +static jint* +JNIEnv_GetIntArrayElements(JNIEnv* p0, jintArray p1, jboolean* p2) +{ + return get_array_elements(p0, p1, p2); +} + +static jlong* +JNIEnv_GetLongArrayElements(JNIEnv* p0, jlongArray p1, jboolean* p2) +{ + return get_array_elements(p0, p1, p2); +} + +static jfloat* +JNIEnv_GetFloatArrayElements(JNIEnv* p0, jfloatArray p1, jboolean* p2) +{ + return get_array_elements(p0, p1, p2); +} + +static jdouble* +JNIEnv_GetDoubleArrayElements(JNIEnv* p0, jdoubleArray p1, jboolean* p2) +{ + return get_array_elements(p0, p1, p2); +} + +static void +JNIEnv_ReleaseBooleanArrayElements(JNIEnv* p0, jbooleanArray p1, jboolean* p2, jint p3) +{ +} + +static void +JNIEnv_ReleaseByteArrayElements(JNIEnv* p0, jbyteArray p1, jbyte* p2, jint p3) +{ +} + +static void +JNIEnv_ReleaseCharArrayElements(JNIEnv* p0, jcharArray p1, jchar* p2, jint p3) +{ +} + +static void +JNIEnv_ReleaseShortArrayElements(JNIEnv* p0, jshortArray p1, jshort* p2, jint p3) +{ +} + +static void +JNIEnv_ReleaseIntArrayElements(JNIEnv* p0, jintArray p1, jint* p2, jint p3) +{ +} + +static void +JNIEnv_ReleaseLongArrayElements(JNIEnv* p0, jlongArray p1, jlong* p2, jint p3) +{ +} + +static void +JNIEnv_ReleaseFloatArrayElements(JNIEnv* p0, jfloatArray p1, jfloat* p2, jint p3) +{ +} + +static void +JNIEnv_ReleaseDoubleArrayElements(JNIEnv* p0, jdoubleArray p1, jdouble* p2, jint p3) +{ +} + +static void +JNIEnv_GetBooleanArrayRegion(JNIEnv* p0, jbooleanArray p1, jsize p2, jsize p3, jboolean* p4) +{ +} + +static void +get_array_region(JNIEnv *env, const void *arrayobj, jsize start, jsize len, void *buf) +{ + if (!arrayobj) + return; + + const struct array *array = arrayobj; + assert(start + len <= array->length); + start *= array->element_size; + len *= array->element_size; + memcpy(buf, (char *)array->data + start, len); +} + +static void +JNIEnv_GetByteArrayRegion(JNIEnv *env, jbyteArray arrayobj, jsize start, jsize len, jbyte* buf) +{ + + get_array_region(env, arrayobj, start, len, buf); +} + +static void +JNIEnv_GetCharArrayRegion(JNIEnv* p0, jcharArray p1, jsize p2, jsize p3, jchar* p4) +{ + + get_array_region(p0, p1, p2, p3, p4); +} + +static void +JNIEnv_GetShortArrayRegion(JNIEnv* p0, jshortArray p1, jsize p2, jsize p3, jshort* p4) +{ + + get_array_region(p0, p1, p2, p3, p4); +} + +static void +JNIEnv_GetIntArrayRegion(JNIEnv* p0, jintArray p1, jsize p2, jsize p3, jint* p4) +{ + + get_array_region(p0, p1, p2, p3, p4); +} + +static void +JNIEnv_GetLongArrayRegion(JNIEnv* p0, jlongArray p1, jsize p2, jsize p3, jlong* p4) +{ + + get_array_region(p0, p1, p2, p3, p4); +} + +static void +JNIEnv_GetFloatArrayRegion(JNIEnv* p0, jfloatArray p1, jsize p2, jsize p3, jfloat* p4) +{ + + get_array_region(p0, p1, p2, p3, p4); +} + +static void +JNIEnv_GetDoubleArrayRegion(JNIEnv* p0, jdoubleArray p1, jsize p2, jsize p3, jdouble* p4) +{ + + get_array_region(p0, p1, p2, p3, p4); +} + +static void +set_array_region(JNIEnv *env, void *arrayobj, jsize start, jsize len, const void *buf) +{ + if (!arrayobj) + return; + + const struct array *array = arrayobj; + assert(start + len <= array->length); + start *= array->element_size; + len *= array->element_size; + memcpy((char *)array->data + start, buf, len); +} + +static void +JNIEnv_SetBooleanArrayRegion(JNIEnv* p0, jbooleanArray p1, jsize p2, jsize p3, const jboolean* p4) +{ + set_array_region(p0, p1, p2, p3, p4); +} + +static void +JNIEnv_SetByteArrayRegion(JNIEnv* p0, jbyteArray p1, jsize p2, jsize p3, const jbyte* p4) +{ + set_array_region(p0, p1, p2, p3, p4); +} + +static void +JNIEnv_SetCharArrayRegion(JNIEnv* p0, jcharArray p1, jsize p2, jsize p3, const jchar* p4) +{ + set_array_region(p0, p1, p2, p3, p4); +} + + +static void +JNIEnv_SetShortArrayRegion(JNIEnv* p0, jshortArray p1, jsize p2, jsize p3, const jshort* p4) +{ + set_array_region(p0, p1, p2, p3, p4); +} + +static void +JNIEnv_SetIntArrayRegion(JNIEnv* p0, jintArray p1, jsize p2, jsize p3, const jint* p4) +{ + set_array_region(p0, p1, p2, p3, p4); +} + + +static void +JNIEnv_SetLongArrayRegion(JNIEnv* p0, jlongArray p1, jsize p2, jsize p3, const jlong* p4) +{ + set_array_region(p0, p1, p2, p3, p4); +} + + +static void +JNIEnv_SetFloatArrayRegion(JNIEnv* p0, jfloatArray p1, jsize p2, jsize p3, const jfloat* p4) +{ + set_array_region(p0, p1, p2, p3, p4); +} + + +static void +JNIEnv_SetDoubleArrayRegion(JNIEnv* p0, jdoubleArray p1, jsize p2, jsize p3, const jdouble* p4) +{ + set_array_region(p0, p1, p2, p3, p4); +} + +static void +jnienv_register_jvm_native_method(JNIEnv *env, const char *klass, const char *method, void *function) +{ + size_t i; + struct jvm *jvm = jnienv_get_jvm(env); + for (i = 0; i < ARRAY_SIZE(jvm->methods) && jvm->methods[i].function; ++i); + assert(i < ARRAY_SIZE(jvm->methods) && "native method limit reached!"); + jvm->methods[i].klass = strdup(klass); + jvm->methods[i].method = strdup(method); + jvm->methods[i].function = function; +} + +static jint +JNIEnv_RegisterNatives(JNIEnv* p0, jclass p1, const JNINativeMethod* p2, jint p3) +{ + const struct jclass *clazz = (struct jclass*)p1; + const JNINativeMethod *method = p2; + for (int i = 0; i < p3; ++i, ++method) { + jnienv_register_jvm_native_method(p0, clazz->name, method->name, method->fnPtr); + } + return 0; +} + +static jint +JNIEnv_UnregisterNatives(JNIEnv* p0, jclass p1) +{ + const struct jclass *klass = (struct jclass*)p1; + struct jvm *jvm = jnienv_get_jvm(p0); + for (size_t i = 0; i < ARRAY_SIZE(jvm->methods) && jvm->methods[i].function; ++i) { + if (strcmp(jvm->methods[i].klass, klass->name)) + continue; + jvm->methods[i] = (struct jvm_native_method){0}; + } + return 0; +} + +static jint +JNIEnv_MonitorEnter(JNIEnv* p0, jobject p1) +{ + return 0; +} + +static jint +JNIEnv_MonitorExit(JNIEnv* p0, jobject p1) +{ + return 0; +} + +static jint +JNIEnv_GetJavaVM(JNIEnv* env, JavaVM** vm) +{ + struct jvm *jvm = jnienv_get_jvm(env); + *vm = (JavaVM*)&jvm->vm; + return 0; +} + +static void +JNIEnv_GetStringRegion(JNIEnv* p0, jstring p1, jsize p2, jsize p3, jchar* p4) +{ +} + +static void +JNIEnv_GetStringUTFRegion(JNIEnv* p0, jstring p1, jsize p2, jsize p3, char* p4) +{ +} + +static void* +JNIEnv_GetPrimitiveArrayCritical(JNIEnv *env, jarray array, jboolean *isCopy) +{ + return get_array_elements(env, array, isCopy); +} + +static void +JNIEnv_ReleasePrimitiveArrayCritical(JNIEnv *env, jarray array, void *carray, jint mode) +{ +} + +const jchar* +JNIEnv_GetStringCritical(JNIEnv* p0, jstring p1, jboolean* p2) +{ + + return NULL; +} + +static void +JNIEnv_ReleaseStringCritical(JNIEnv* p0, jstring p1, const jchar* p2) +{ + +} + +static jweak +JNIEnv_NewWeakGlobalRef(JNIEnv* p0, jobject p1) +{ + return NULL; +} + +static void +JNIEnv_DeleteWeakGlobalRef(JNIEnv* p0, jweak p1) +{ +} + +static jboolean +JNIEnv_ExceptionCheck(JNIEnv* p0) +{ + return 0; +} + +static jobject +JNIEnv_NewDirectByteBuffer(JNIEnv* p0, void* p1, jlong p2) +{ + return NULL; +} + +static void* +JNIEnv_GetDirectBufferAddress(JNIEnv* p0, jobject p1) +{ + return NULL; +} + +static jlong +JNIEnv_GetDirectBufferCapacity(JNIEnv* p0, jobject p1) +{ + return 0; +} + +const char* +JNIEnv_GetStringUTFChars(JNIEnv *env, jstring string, jboolean *isCopy) +{ + if (!string) + return strdup(""); + + struct jstring *str = (struct jstring*)string; + + return str->data; +} + +static void +JNIEnv_ReleaseStringUTFChars(JNIEnv *env, jstring string, const char *utf) +{ + free((void*)utf); +} + +static void +env_init(JNIEnv *env, struct JNINativeInterface *native) +{ + assert(env && native); + native->GetStringUTFChars = JNIEnv_GetStringUTFChars; + native->ReleaseStringUTFChars = JNIEnv_ReleaseStringUTFChars; + native->GetVersion = JNIEnv_GetVersion; + native->DefineClass = JNIEnv_DefineClass; + native->FindClass = JNIEnv_FindClass; + native->FromReflectedMethod = JNIEnv_FromReflectedMethod; + native->FromReflectedField = JNIEnv_FromReflectedField; + native->ToReflectedMethod = JNIEnv_ToReflectedMethod; + native->GetSuperclass = JNIEnv_GetSuperclass; + native->IsAssignableFrom = JNIEnv_IsAssignableFrom; + native->ToReflectedField = JNIEnv_ToReflectedField; + native->Throw = JNIEnv_Throw; + native->ThrowNew = JNIEnv_ThrowNew; + native->ExceptionOccurred = JNIEnv_ExceptionOccurred; + native->ExceptionDescribe = JNIEnv_ExceptionDescribe; + native->ExceptionClear = JNIEnv_ExceptionClear; + native->FatalError = JNIEnv_FatalError; + native->PushLocalFrame = JNIEnv_PushLocalFrame; + native->PopLocalFrame = JNIEnv_PopLocalFrame; + native->NewGlobalRef = JNIEnv_NewGlobalRef; + native->DeleteGlobalRef = JNIEnv_DeleteGlobalRef; + native->DeleteLocalRef = JNIEnv_DeleteLocalRef; + native->IsSameObject = JNIEnv_IsSameObject; + native->NewLocalRef = JNIEnv_NewLocalRef; + native->EnsureLocalCapacity = JNIEnv_EnsureLocalCapacity; + native->AllocObject = JNIEnv_AllocObject; + native->NewObject = JNIEnv_NewObject; + native->NewObjectV = JNIEnv_NewObjectV; + native->NewObjectA = JNIEnv_NewObjectA; + native->GetObjectClass = JNIEnv_GetObjectClass; + native->IsInstanceOf = JNIEnv_IsInstanceOf; + native->GetMethodID = JNIEnv_GetMethodID; + native->CallObjectMethod = JNIEnv_CallObjectMethod; + native->CallObjectMethodV = JNIEnv_CallObjectMethodV; + native->CallObjectMethodA = JNIEnv_CallObjectMethodA; + native->CallBooleanMethod = JNIEnv_CallBooleanMethod; + native->CallBooleanMethodV = JNIEnv_CallBooleanMethodV; + native->CallBooleanMethodA = JNIEnv_CallBooleanMethodA; + native->CallByteMethod = JNIEnv_CallByteMethod; + native->CallByteMethodV = JNIEnv_CallByteMethodV; + native->CallByteMethodA = JNIEnv_CallByteMethodA; + native->CallCharMethod = JNIEnv_CallCharMethod; + native->CallCharMethodV = JNIEnv_CallCharMethodV; + native->CallCharMethodA = JNIEnv_CallCharMethodA; + native->CallShortMethod = JNIEnv_CallShortMethod; + native->CallShortMethodV = JNIEnv_CallShortMethodV; + native->CallShortMethodA = JNIEnv_CallShortMethodA; + native->CallIntMethod = JNIEnv_CallIntMethod; + native->CallIntMethodV = JNIEnv_CallIntMethodV; + native->CallIntMethodA = JNIEnv_CallIntMethodA; + native->CallLongMethod = JNIEnv_CallLongMethod; + native->CallLongMethodV = JNIEnv_CallLongMethodV; + native->CallLongMethodA = JNIEnv_CallLongMethodA; + native->CallFloatMethod = JNIEnv_CallFloatMethod; + native->CallFloatMethodV = JNIEnv_CallFloatMethodV; + native->CallFloatMethodA = JNIEnv_CallFloatMethodA; + native->CallDoubleMethod = JNIEnv_CallDoubleMethod; + native->CallDoubleMethodV = JNIEnv_CallDoubleMethodV; + native->CallDoubleMethodA = JNIEnv_CallDoubleMethodA; + native->CallVoidMethod = JNIEnv_CallVoidMethod; + native->CallVoidMethodV = JNIEnv_CallVoidMethodV; + native->CallVoidMethodA = JNIEnv_CallVoidMethodA; + native->CallNonvirtualObjectMethod = JNIEnv_CallNonvirtualObjectMethod; + native->CallNonvirtualObjectMethodV = JNIEnv_CallNonvirtualObjectMethodV; + native->CallNonvirtualObjectMethodA = JNIEnv_CallNonvirtualObjectMethodA; + native->CallNonvirtualBooleanMethod = JNIEnv_CallNonvirtualBooleanMethod; + native->CallNonvirtualBooleanMethodV = JNIEnv_CallNonvirtualBooleanMethodV; + native->CallNonvirtualBooleanMethodA = JNIEnv_CallNonvirtualBooleanMethodA; + native->CallNonvirtualByteMethod = JNIEnv_CallNonvirtualByteMethod; + native->CallNonvirtualByteMethodV = JNIEnv_CallNonvirtualByteMethodV; + native->CallNonvirtualByteMethodA = JNIEnv_CallNonvirtualByteMethodA; + native->CallNonvirtualCharMethod = JNIEnv_CallNonvirtualCharMethod; + native->CallNonvirtualCharMethodV = JNIEnv_CallNonvirtualCharMethodV; + native->CallNonvirtualCharMethodA = JNIEnv_CallNonvirtualCharMethodA; + native->CallNonvirtualShortMethod = JNIEnv_CallNonvirtualShortMethod; + native->CallNonvirtualShortMethodV = JNIEnv_CallNonvirtualShortMethodV; + native->CallNonvirtualShortMethodA = JNIEnv_CallNonvirtualShortMethodA; + native->CallNonvirtualIntMethod = JNIEnv_CallNonvirtualIntMethod; + native->CallNonvirtualIntMethodV = JNIEnv_CallNonvirtualIntMethodV; + native->CallNonvirtualIntMethodA = JNIEnv_CallNonvirtualIntMethodA; + native->CallNonvirtualLongMethod = JNIEnv_CallNonvirtualLongMethod; + native->CallNonvirtualLongMethodV = JNIEnv_CallNonvirtualLongMethodV; + native->CallNonvirtualLongMethodA = JNIEnv_CallNonvirtualLongMethodA; + native->CallNonvirtualFloatMethod = JNIEnv_CallNonvirtualFloatMethod; + native->CallNonvirtualFloatMethodV = JNIEnv_CallNonvirtualFloatMethodV; + native->CallNonvirtualFloatMethodA = JNIEnv_CallNonvirtualFloatMethodA; + native->CallNonvirtualDoubleMethod = JNIEnv_CallNonvirtualDoubleMethod; + native->CallNonvirtualDoubleMethodV = JNIEnv_CallNonvirtualDoubleMethodV; + native->CallNonvirtualDoubleMethodA = JNIEnv_CallNonvirtualDoubleMethodA; + native->CallNonvirtualVoidMethod = JNIEnv_CallNonvirtualVoidMethod; + native->CallNonvirtualVoidMethodV = JNIEnv_CallNonvirtualVoidMethodV; + native->CallNonvirtualVoidMethodA = JNIEnv_CallNonvirtualVoidMethodA; + native->GetFieldID = JNIEnv_GetFieldID; + native->GetObjectField = JNIEnv_GetObjectField; + native->GetBooleanField = JNIEnv_GetBooleanField; + native->GetByteField = JNIEnv_GetByteField; + native->GetCharField = JNIEnv_GetCharField; + native->GetShortField = JNIEnv_GetShortField; + native->GetIntField = JNIEnv_GetIntField; + native->GetLongField = JNIEnv_GetLongField; + native->GetFloatField = JNIEnv_GetFloatField; + native->GetDoubleField = JNIEnv_GetDoubleField; + native->SetObjectField = JNIEnv_SetObjectField; + native->SetBooleanField = JNIEnv_SetBooleanField; + native->SetByteField = JNIEnv_SetByteField; + native->SetCharField = JNIEnv_SetCharField; + native->SetShortField = JNIEnv_SetShortField; + native->SetIntField = JNIEnv_SetIntField; + native->SetLongField = JNIEnv_SetLongField; + native->SetFloatField = JNIEnv_SetFloatField; + native->SetDoubleField = JNIEnv_SetDoubleField; + native->GetStaticMethodID = JNIEnv_GetStaticMethodID; + native->CallStaticObjectMethod = JNIEnv_CallStaticObjectMethod; + native->CallStaticObjectMethodV = JNIEnv_CallStaticObjectMethodV; + native->CallStaticObjectMethodA = JNIEnv_CallStaticObjectMethodA; + native->CallStaticBooleanMethod = JNIEnv_CallStaticBooleanMethod; + native->CallStaticBooleanMethodV = JNIEnv_CallStaticBooleanMethodV; + native->CallStaticBooleanMethodA = JNIEnv_CallStaticBooleanMethodA; + native->CallStaticByteMethod = JNIEnv_CallStaticByteMethod; + native->CallStaticByteMethodV = JNIEnv_CallStaticByteMethodV; + native->CallStaticByteMethodA = JNIEnv_CallStaticByteMethodA; + native->CallStaticCharMethod = JNIEnv_CallStaticCharMethod; + native->CallStaticCharMethodV = JNIEnv_CallStaticCharMethodV; + native->CallStaticCharMethodA = JNIEnv_CallStaticCharMethodA; + native->CallStaticShortMethod = JNIEnv_CallStaticShortMethod; + native->CallStaticShortMethodV = JNIEnv_CallStaticShortMethodV; + native->CallStaticShortMethodA = JNIEnv_CallStaticShortMethodA; + native->CallStaticIntMethod = JNIEnv_CallStaticIntMethod; + native->CallStaticIntMethodV = JNIEnv_CallStaticIntMethodV; + native->CallStaticIntMethodA = JNIEnv_CallStaticIntMethodA; + native->CallStaticLongMethod = JNIEnv_CallStaticLongMethod; + native->CallStaticLongMethodV = JNIEnv_CallStaticLongMethodV; + native->CallStaticLongMethodA = JNIEnv_CallStaticLongMethodA; + native->CallStaticFloatMethod = JNIEnv_CallStaticFloatMethod; + native->CallStaticFloatMethodV = JNIEnv_CallStaticFloatMethodV; + native->CallStaticFloatMethodA = JNIEnv_CallStaticFloatMethodA; + native->CallStaticDoubleMethod = JNIEnv_CallStaticDoubleMethod; + native->CallStaticDoubleMethodV = JNIEnv_CallStaticDoubleMethodV; + native->CallStaticDoubleMethodA = JNIEnv_CallStaticDoubleMethodA; + native->CallStaticVoidMethod = JNIEnv_CallStaticVoidMethod; + native->CallStaticVoidMethodV = JNIEnv_CallStaticVoidMethodV; + native->CallStaticVoidMethodA = JNIEnv_CallStaticVoidMethodA; + native->GetStaticFieldID = JNIEnv_GetStaticFieldID; + native->GetStaticObjectField = JNIEnv_GetStaticObjectField; + native->GetStaticBooleanField = JNIEnv_GetStaticBooleanField; + native->GetStaticByteField = JNIEnv_GetStaticByteField; + native->GetStaticCharField = JNIEnv_GetStaticCharField; + native->GetStaticShortField = JNIEnv_GetStaticShortField; + native->GetStaticIntField = JNIEnv_GetStaticIntField; + native->GetStaticLongField = JNIEnv_GetStaticLongField; + native->GetStaticFloatField = JNIEnv_GetStaticFloatField; + native->GetStaticDoubleField = JNIEnv_GetStaticDoubleField; + native->SetStaticObjectField = JNIEnv_SetStaticObjectField; + native->SetStaticBooleanField = JNIEnv_SetStaticBooleanField; + native->SetStaticByteField = JNIEnv_SetStaticByteField; + native->SetStaticCharField = JNIEnv_SetStaticCharField; + native->SetStaticShortField = JNIEnv_SetStaticShortField; + native->SetStaticIntField = JNIEnv_SetStaticIntField; + native->SetStaticLongField = JNIEnv_SetStaticLongField; + native->SetStaticFloatField = JNIEnv_SetStaticFloatField; + native->SetStaticDoubleField = JNIEnv_SetStaticDoubleField; + native->NewString = JNIEnv_NewString; + native->GetStringLength = JNIEnv_GetStringLength; + native->GetStringChars = JNIEnv_GetStringChars; + native->ReleaseStringChars = JNIEnv_ReleaseStringChars; + native->NewStringUTF = JNIEnv_NewStringUTF; + native->GetStringUTFLength = JNIEnv_GetStringUTFLength; + native->GetArrayLength = JNIEnv_GetArrayLength; + native->NewObjectArray = JNIEnv_NewObjectArray; + native->GetObjectArrayElement = JNIEnv_GetObjectArrayElement; + native->SetObjectArrayElement = JNIEnv_SetObjectArrayElement; + native->NewBooleanArray = JNIEnv_NewBooleanArray; + native->NewByteArray = JNIEnv_NewByteArray; + native->NewCharArray = JNIEnv_NewCharArray; + native->NewShortArray = JNIEnv_NewShortArray; + native->NewIntArray = JNIEnv_NewIntArray; + native->NewLongArray = JNIEnv_NewLongArray; + native->NewFloatArray = JNIEnv_NewFloatArray; + native->NewDoubleArray = JNIEnv_NewDoubleArray; + native->GetBooleanArrayElements = JNIEnv_GetBooleanArrayElements; + native->GetByteArrayElements = JNIEnv_GetByteArrayElements; + native->GetCharArrayElements = JNIEnv_GetCharArrayElements; + native->GetShortArrayElements = JNIEnv_GetShortArrayElements; + native->GetIntArrayElements = JNIEnv_GetIntArrayElements; + native->GetLongArrayElements = JNIEnv_GetLongArrayElements; + native->GetFloatArrayElements = JNIEnv_GetFloatArrayElements; + native->GetDoubleArrayElements = JNIEnv_GetDoubleArrayElements; + native->ReleaseBooleanArrayElements = JNIEnv_ReleaseBooleanArrayElements; + native->ReleaseByteArrayElements = JNIEnv_ReleaseByteArrayElements; + native->ReleaseCharArrayElements = JNIEnv_ReleaseCharArrayElements; + native->ReleaseShortArrayElements = JNIEnv_ReleaseShortArrayElements; + native->ReleaseIntArrayElements = JNIEnv_ReleaseIntArrayElements; + native->ReleaseLongArrayElements = JNIEnv_ReleaseLongArrayElements; + native->ReleaseFloatArrayElements = JNIEnv_ReleaseFloatArrayElements; + native->ReleaseDoubleArrayElements = JNIEnv_ReleaseDoubleArrayElements; + native->GetBooleanArrayRegion = JNIEnv_GetBooleanArrayRegion; + native->GetByteArrayRegion = JNIEnv_GetByteArrayRegion; + native->GetCharArrayRegion = JNIEnv_GetCharArrayRegion; + native->GetShortArrayRegion = JNIEnv_GetShortArrayRegion; + native->GetIntArrayRegion = JNIEnv_GetIntArrayRegion; + native->GetLongArrayRegion = JNIEnv_GetLongArrayRegion; + native->GetFloatArrayRegion = JNIEnv_GetFloatArrayRegion; + native->GetDoubleArrayRegion = JNIEnv_GetDoubleArrayRegion; + native->SetBooleanArrayRegion = JNIEnv_SetBooleanArrayRegion; + native->SetByteArrayRegion = JNIEnv_SetByteArrayRegion; + native->SetCharArrayRegion = JNIEnv_SetCharArrayRegion; + native->SetShortArrayRegion = JNIEnv_SetShortArrayRegion; + native->SetIntArrayRegion = JNIEnv_SetIntArrayRegion; + native->SetLongArrayRegion = JNIEnv_SetLongArrayRegion; + native->SetFloatArrayRegion = JNIEnv_SetFloatArrayRegion; + native->SetDoubleArrayRegion = JNIEnv_SetDoubleArrayRegion; + native->RegisterNatives = JNIEnv_RegisterNatives; + native->UnregisterNatives = JNIEnv_UnregisterNatives; + native->MonitorEnter = JNIEnv_MonitorEnter; + native->MonitorExit = JNIEnv_MonitorExit; + native->GetJavaVM = JNIEnv_GetJavaVM; + native->GetStringRegion = JNIEnv_GetStringRegion; + native->GetStringUTFRegion = JNIEnv_GetStringUTFRegion; + native->GetPrimitiveArrayCritical = JNIEnv_GetPrimitiveArrayCritical; + native->ReleasePrimitiveArrayCritical = JNIEnv_ReleasePrimitiveArrayCritical; + native->GetStringCritical = JNIEnv_GetStringCritical; + native->ReleaseStringCritical = JNIEnv_ReleaseStringCritical; + native->NewWeakGlobalRef = JNIEnv_NewWeakGlobalRef; + native->DeleteWeakGlobalRef = JNIEnv_DeleteWeakGlobalRef; + native->ExceptionCheck = JNIEnv_ExceptionCheck; + native->NewDirectByteBuffer = JNIEnv_NewDirectByteBuffer; + native->GetDirectBufferAddress = JNIEnv_GetDirectBufferAddress; + native->GetDirectBufferCapacity = JNIEnv_GetDirectBufferCapacity; + *env = native; +} + +static jint +JavaVM_DestroyJavaVM(JavaVM *vm) +{ + return 0; +} + +static jint +JavaVM_AttachCurrentThread(JavaVM *vm, JNIEnv **env, void *args) +{ + *env = &javavm_get_jvm(vm)->env; + return 0; +} + +static jint +JavaVM_DetachCurrentThread(JavaVM *vm) +{ + return 0; +} + +static jint +JavaVM_GetEnv(JavaVM *vm, void **env, jint version) +{ + *env = &javavm_get_jvm(vm)->env; + return 0; +} + +static jint +JavaVM_AttachCurrentThreadAsDaemon(JavaVM *vm, JNIEnv **env, void *args) +{ + *env = &javavm_get_jvm(vm)->env; + return 0; +} + +static void +vm_init(JavaVM *vm, struct JNIInvokeInterface *invoke) +{ + assert(vm && invoke); + invoke->DestroyJavaVM = JavaVM_DestroyJavaVM; + invoke->AttachCurrentThread = JavaVM_AttachCurrentThread; + invoke->DetachCurrentThread = JavaVM_DetachCurrentThread; + invoke->GetEnv = JavaVM_GetEnv; + invoke->AttachCurrentThreadAsDaemon = JavaVM_AttachCurrentThreadAsDaemon; + *vm = invoke; +} + +void +jvm_init(struct jvm *jvm) +{ + assert(jvm); + *jvm = (struct jvm){0}; + vm_init(&jvm->vm, &jvm->invoke); + env_init(&jvm->env, &jvm->native); +} diff --git a/src/fakejvm/jvm.h b/src/fakejvm/jvm.h new file mode 100644 index 0000000..f7e9d6c --- /dev/null +++ b/src/fakejvm/jvm.h @@ -0,0 +1,25 @@ +#pragma once + +#include "jni.h" + +struct jvm_native_method { + char *klass; + char *method; + void *function; +}; + +struct jvm { + struct jvm_native_method methods[255]; + + struct JNINativeInterface native; + struct JNIInvokeInterface invoke; + + // JNI's api is weird.. pointer to a reference of a struct, OK! + // Developers have to dereference these pointers to call methods from an ... reference. + // NOTE: These are pointers, and JNI interface passes pointers to these pointers! + JNIEnv env; // points to native + JavaVM vm; // points to invoke +}; + +void +jvm_init(struct jvm *jvm); -- cgit v1.2.3-70-g09d2