/* Copyright (c) 2008-2014, Avian Contributors Permission to use, copy, modify, and/or distribute this software for any purpose with or without fee is hereby granted, provided that the above copyright notice and this permission notice appear in all copies. There is NO WARRANTY for this software. See license.txt for details. */ #include "avian/jnienv.h" #include "avian/machine.h" #include "avian/util.h" #include "avian/processor.h" #include "avian/constants.h" #include using namespace vm; namespace { namespace local { jint JNICALL AttachCurrentThread(Machine* m, Thread** t, void*) { *t = static_cast(m->localThread->get()); if (*t == 0) { *t = attachThread(m, false); } return 0; } jint JNICALL AttachCurrentThreadAsDaemon(Machine* m, Thread** t, void*) { *t = static_cast(m->localThread->get()); if (*t == 0) { *t = attachThread(m, true); } return 0; } jint JNICALL DetachCurrentThread(Machine* m) { Thread* t = static_cast(m->localThread->get()); if (t) { // todo: detaching the root thread seems to cause stability // problems which I haven't yet had a chance to investigate // thoroughly. Meanwhile, we just ignore requests to detach it, // which leaks a bit of memory but should be harmless otherwise. if (m->rootThread != t) { m->localThread->set(0); ACQUIRE_RAW(t, t->m->stateLock); enter(t, Thread::ActiveState); t->javaThread->peer() = 0; enter(t, Thread::ZombieState); t->state = Thread::JoinedState; } return 0; } else { return -1; } } uint64_t destroyJavaVM(Thread* t, uintptr_t*) { // wait for other non-daemon threads to exit { ACQUIRE(t, t->m->stateLock); while (t->m->liveCount - t->m->daemonCount > 1) { t->m->stateLock->wait(t->systemThread, 0); } } { ENTER(t, Thread::ActiveState); t->m->classpath->shutDown(t); } // wait again in case the Classpath::shutDown process started new // threads: { ACQUIRE(t, t->m->stateLock); while (t->m->liveCount - t->m->daemonCount > 1) { t->m->stateLock->wait(t->systemThread, 0); } enter(t, Thread::ExclusiveState); } shutDown(t); return 1; } jint JNICALL DestroyJavaVM(Machine* m) { Thread* t; AttachCurrentThread(m, &t, 0); if (runRaw(t, destroyJavaVM, 0)) { t->exit(); return 0; } else { return -1; } } jint JNICALL GetEnv(Machine* m, Thread** t, jint version) { *t = static_cast(m->localThread->get()); if (*t) { if (version <= JNI_VERSION_1_6) { return AVIAN_JNI_OK; } else { return AVIAN_JNI_EVERSION; } } else { return AVIAN_JNI_EDETACHED; } } jint JNICALL GetVersion(Thread* t) { ENTER(t, Thread::ActiveState); return JNI_VERSION_1_6; } jsize JNICALL GetStringLength(Thread* t, jstring s) { ENTER(t, Thread::ActiveState); return (*s)->length(t); } const jchar* JNICALL GetStringChars(Thread* t, jstring s, jboolean* isCopy) { ENTER(t, Thread::ActiveState); jchar* chars = static_cast( t->m->heap->allocate(((*s)->length(t) + 1) * sizeof(jchar))); stringChars(t, *s, chars); if (isCopy) *isCopy = true; return chars; } void JNICALL ReleaseStringChars(Thread* t, jstring s, const jchar* chars) { ENTER(t, Thread::ActiveState); t->m->heap->free(chars, ((*s)->length(t) + 1) * sizeof(jchar)); } void JNICALL GetStringRegion(Thread* t, jstring s, jsize start, jsize length, jchar* dst) { ENTER(t, Thread::ActiveState); stringChars(t, *s, start, length, dst); } const jchar* JNICALL GetStringCritical(Thread* t, jstring s, jboolean* isCopy) { if (t->criticalLevel == 0) { enter(t, Thread::ActiveState); } ++t->criticalLevel; if (isCopy) { *isCopy = true; } object data = (*s)->data(); if (objectClass(t, data) == type(t, GcByteArray::Type)) { return GetStringChars(t, s, isCopy); } else { return &cast(t, data)->body()[(*s)->offset(t)]; } } void JNICALL ReleaseStringCritical(Thread* t, jstring s, const jchar* chars) { if (objectClass(t, (*s)->data()) == type(t, GcByteArray::Type)) { ReleaseStringChars(t, s, chars); } if ((--t->criticalLevel) == 0) { enter(t, Thread::IdleState); } } jsize JNICALL GetStringUTFLength(Thread* t, jstring s) { ENTER(t, Thread::ActiveState); return stringUTFLength(t, *s); } const char* JNICALL GetStringUTFChars(Thread* t, jstring s, jboolean* isCopy) { ENTER(t, Thread::ActiveState); int length = stringUTFLength(t, *s); char* chars = static_cast(t->m->heap->allocate(length + 1)); stringUTFChars(t, *s, chars, length); if (isCopy) *isCopy = true; return chars; } void JNICALL ReleaseStringUTFChars(Thread* t, jstring s, const char* chars) { ENTER(t, Thread::ActiveState); t->m->heap->free(chars, stringUTFLength(t, *s) + 1); } void JNICALL GetStringUTFRegion(Thread* t, jstring s, jsize start, jsize length, char* dst) { ENTER(t, Thread::ActiveState); stringUTFChars( t, *s, start, length, dst, stringUTFLength(t, *s, start, length)); } jsize JNICALL GetArrayLength(Thread* t, jarray array) { ENTER(t, Thread::ActiveState); return fieldAtOffset(*array, BytesPerWord); } uint64_t newString(Thread* t, uintptr_t* arguments) { const jchar* chars = reinterpret_cast(arguments[0]); jsize size = arguments[1]; GcCharArray* a = makeCharArray(t, size); if (size) { memcpy(a->body().begin(), chars, size * sizeof(jchar)); } return reinterpret_cast( makeLocalReference(t, t->m->classpath->makeString(t, a, 0, size))); } jstring JNICALL NewString(Thread* t, const jchar* chars, jsize size) { if (chars == 0) return 0; uintptr_t arguments[] = {reinterpret_cast(chars), static_cast(size)}; return reinterpret_cast(run(t, newString, arguments)); } uint64_t newStringUTF(Thread* t, uintptr_t* arguments) { const char* chars = reinterpret_cast(arguments[0]); object array = parseUtf8(t, chars, strlen(chars)); return reinterpret_cast(makeLocalReference( t, t->m->classpath->makeString( t, array, 0, fieldAtOffset(array, BytesPerWord) - 1))); } jstring JNICALL NewStringUTF(Thread* t, const char* chars) { if (chars == 0) return 0; uintptr_t arguments[] = {reinterpret_cast(chars)}; return reinterpret_cast(run(t, newStringUTF, arguments)); } void replace(int a, int b, const char* in, int8_t* out) { while (*in) { *out = (*in == a ? b : *in); ++in; ++out; } *out = 0; } uint64_t defineClass(Thread* t, uintptr_t* arguments) { jobject loader = reinterpret_cast(arguments[0]); const uint8_t* buffer = reinterpret_cast(arguments[1]); jsize length = arguments[2]; return reinterpret_cast(makeLocalReference( t, getJClass( t, cast(t, defineClass(t, loader ? cast(t, *loader) : roots(t)->bootLoader(), buffer, length))))); } jclass JNICALL DefineClass(Thread* t, const char*, jobject loader, const jbyte* buffer, jsize length) { uintptr_t arguments[] = {reinterpret_cast(loader), reinterpret_cast(buffer), static_cast(length)}; return reinterpret_cast(run(t, defineClass, arguments)); } uint64_t findClass(Thread* t, uintptr_t* arguments) { const char* name = reinterpret_cast(arguments[0]); GcByteArray* n = makeByteArray(t, strlen(name) + 1); replace('.', '/', name, n->body().begin()); GcMethod* caller = getCaller(t, 0); GcClass* c = resolveClass(t, caller ? t->m->classpath->libraryClassLoader(t, caller) : roots(t)->appLoader(), n); if (t->m->classpath->mayInitClasses()) { PROTECT(t, c); initClass(t, c); } return reinterpret_cast(makeLocalReference(t, getJClass(t, c))); } jclass JNICALL FindClass(Thread* t, const char* name) { uintptr_t arguments[] = {reinterpret_cast(name)}; return reinterpret_cast(run(t, findClass, arguments)); } uint64_t throwNew(Thread* t, uintptr_t* arguments) { jclass c = reinterpret_cast(arguments[0]); const char* message = reinterpret_cast(arguments[1]); GcString* m = 0; PROTECT(t, m); if (message) { m = makeString(t, "%s", message); } object trace = makeTrace(t); PROTECT(t, trace); t->exception = cast(t, make(t, (*c)->vmClass())); t->exception->setMessage(t, m); t->exception->setTrace(t, trace); return 1; } jint JNICALL ThrowNew(Thread* t, jclass c, const char* message) { if (t->exception) { return -1; } uintptr_t arguments[] = {reinterpret_cast(c), reinterpret_cast(message)}; return run(t, throwNew, arguments) ? 0 : -1; } jint JNICALL Throw(Thread* t, jthrowable throwable) { if (t->exception) { return -1; } ENTER(t, Thread::ActiveState); t->exception = *throwable; return 0; } jobject JNICALL NewLocalRef(Thread* t, jobject o) { ENTER(t, Thread::ActiveState); return makeLocalReference(t, *o); } void JNICALL DeleteLocalRef(Thread* t, jobject r) { ENTER(t, Thread::ActiveState); disposeLocalReference(t, r); } jboolean JNICALL ExceptionCheck(Thread* t) { return t->exception != 0; } uint64_t getObjectClass(Thread* t, uintptr_t* arguments) { jobject o = reinterpret_cast(arguments[0]); return reinterpret_cast( makeLocalReference(t, getJClass(t, objectClass(t, *o)))); } jclass JNICALL GetObjectClass(Thread* t, jobject o) { uintptr_t arguments[] = {reinterpret_cast(o)}; return reinterpret_cast(run(t, getObjectClass, arguments)); } uint64_t getSuperclass(Thread* t, uintptr_t* arguments) { GcClass* class_ = (*reinterpret_cast(arguments[0]))->vmClass(); if (class_->flags() & ACC_INTERFACE) { return 0; } else { GcClass* super = class_->super(); return super ? reinterpret_cast( makeLocalReference(t, getJClass(t, super))) : 0; } } jclass JNICALL GetSuperclass(Thread* t, jclass c) { uintptr_t arguments[] = {reinterpret_cast(c)}; return reinterpret_cast(run(t, getSuperclass, arguments)); } uint64_t isInstanceOf(Thread* t, uintptr_t* arguments) { jobject o = reinterpret_cast(arguments[0]); jclass c = reinterpret_cast(arguments[1]); return instanceOf(t, (*c)->vmClass(), *o); } jboolean JNICALL IsInstanceOf(Thread* t, jobject o, jclass c) { uintptr_t arguments[] = {reinterpret_cast(o), reinterpret_cast(c)}; return run(t, isInstanceOf, arguments); } uint64_t isAssignableFrom(Thread* t, uintptr_t* arguments) { jclass b = reinterpret_cast(arguments[0]); jclass a = reinterpret_cast(arguments[1]); return isAssignableFrom(t, (*a)->vmClass(), (*b)->vmClass()); } jboolean JNICALL IsAssignableFrom(Thread* t, jclass b, jclass a) { uintptr_t arguments[] = {reinterpret_cast(b), reinterpret_cast(a)}; return run(t, isAssignableFrom, arguments); } GcMethod* findMethod(Thread* t, jclass c, const char* name, const char* spec) { GcByteArray* n = makeByteArray(t, "%s", name); PROTECT(t, n); GcByteArray* s = makeByteArray(t, "%s", spec); return vm::findMethod(t, (*c)->vmClass(), n, s); } jint methodID(Thread* t, GcMethod* method) { int id = method->nativeID(); loadMemoryBarrier(); if (id == 0) { PROTECT(t, method); ACQUIRE(t, t->m->referenceLock); if (method->nativeID() == 0) { GcVector* v = vectorAppend(t, roots(t)->jNIMethodTable(), method); // sequence point, for gc (don't recombine statements) roots(t)->setJNIMethodTable(t, v); storeStoreMemoryBarrier(); method->nativeID() = roots(t)->jNIMethodTable()->size(); } } return method->nativeID(); } uint64_t getMethodID(Thread* t, uintptr_t* arguments) { jclass c = reinterpret_cast(arguments[0]); const char* name = reinterpret_cast(arguments[1]); const char* spec = reinterpret_cast(arguments[2]); GcMethod* method = findMethod(t, c, name, spec); assertT(t, (method->flags() & ACC_STATIC) == 0); return methodID(t, method); } jmethodID JNICALL GetMethodID(Thread* t, jclass c, const char* name, const char* spec) { uintptr_t arguments[] = {reinterpret_cast(c), reinterpret_cast(name), reinterpret_cast(spec)}; return run(t, getMethodID, arguments); } uint64_t getStaticMethodID(Thread* t, uintptr_t* arguments) { jclass c = reinterpret_cast(arguments[0]); const char* name = reinterpret_cast(arguments[1]); const char* spec = reinterpret_cast(arguments[2]); GcMethod* method = findMethod(t, c, name, spec); assertT(t, method->flags() & ACC_STATIC); return methodID(t, method); } jmethodID JNICALL GetStaticMethodID(Thread* t, jclass c, const char* name, const char* spec) { uintptr_t arguments[] = {reinterpret_cast(c), reinterpret_cast(name), reinterpret_cast(spec)}; return run(t, getStaticMethodID, arguments); } GcMethod* getMethod(Thread* t, jmethodID m) { assertT(t, m); GcMethod* method = cast(t, roots(t)->jNIMethodTable()->body()[m - 1]); assertT(t, (method->flags() & ACC_STATIC) == 0); return method; } uint64_t newObjectV(Thread* t, uintptr_t* arguments) { jclass c = reinterpret_cast(arguments[0]); jmethodID m = arguments[1]; va_list* a = reinterpret_cast(arguments[2]); object o = make(t, (*c)->vmClass()); PROTECT(t, o); t->m->processor->invokeList(t, getMethod(t, m), o, true, *a); return reinterpret_cast(makeLocalReference(t, o)); } jobject JNICALL NewObjectV(Thread* t, jclass c, jmethodID m, va_list a) { uintptr_t arguments[] = {reinterpret_cast(c), m, reinterpret_cast(VA_LIST(a))}; return reinterpret_cast(run(t, newObjectV, arguments)); } jobject JNICALL NewObject(Thread* t, jclass c, jmethodID m, ...) { va_list a; va_start(a, m); jobject r = NewObjectV(t, c, m, a); va_end(a); return r; } uint64_t newObjectA(Thread* t, uintptr_t* arguments) { jclass c = reinterpret_cast(arguments[0]); jmethodID m = arguments[1]; const jvalue* a = reinterpret_cast(arguments[2]); object o = make(t, (*c)->vmClass()); PROTECT(t, o); t->m->processor->invokeArray(t, getMethod(t, m), o, a); return reinterpret_cast(makeLocalReference(t, o)); } jobject JNICALL NewObjectA(Thread* t, jclass c, jmethodID m, const jvalue* a) { uintptr_t arguments[] = {reinterpret_cast(c), m, reinterpret_cast(a)}; return reinterpret_cast(run(t, newObjectA, arguments)); } uint64_t callObjectMethodV(Thread* t, uintptr_t* arguments) { jobject o = reinterpret_cast(arguments[0]); jmethodID m = arguments[1]; va_list* a = reinterpret_cast(arguments[2]); return reinterpret_cast(makeLocalReference( t, t->m->processor->invokeList(t, getMethod(t, m), *o, true, *a))); } jobject JNICALL CallObjectMethodV(Thread* t, jobject o, jmethodID m, va_list a) { uintptr_t arguments[] = {reinterpret_cast(o), m, reinterpret_cast(VA_LIST(a))}; return reinterpret_cast(run(t, callObjectMethodV, arguments)); } jobject JNICALL CallObjectMethod(Thread* t, jobject o, jmethodID m, ...) { va_list a; va_start(a, m); jobject r = CallObjectMethodV(t, o, m, a); va_end(a); return r; } uint64_t callObjectMethodA(Thread* t, uintptr_t* arguments) { jobject o = reinterpret_cast(arguments[0]); jmethodID m = arguments[1]; const jvalue* a = reinterpret_cast(arguments[2]); return reinterpret_cast(makeLocalReference( t, t->m->processor->invokeArray(t, getMethod(t, m), *o, a))); } jobject JNICALL CallObjectMethodA(Thread* t, jobject o, jmethodID m, const jvalue* a) { uintptr_t arguments[] = {reinterpret_cast(o), m, reinterpret_cast(a)}; return reinterpret_cast(run(t, callObjectMethodA, arguments)); } uint64_t callIntMethodV(Thread* t, uintptr_t* arguments) { jobject o = reinterpret_cast(arguments[0]); jmethodID m = arguments[1]; va_list* a = reinterpret_cast(arguments[2]); return cast( t, t->m->processor->invokeList(t, getMethod(t, m), *o, true, *a)) ->value(); } jboolean JNICALL CallBooleanMethodV(Thread* t, jobject o, jmethodID m, va_list a) { uintptr_t arguments[] = {reinterpret_cast(o), m, reinterpret_cast(VA_LIST(a))}; return run(t, callIntMethodV, arguments) != 0; } jboolean JNICALL CallBooleanMethod(Thread* t, jobject o, jmethodID m, ...) { va_list a; va_start(a, m); jboolean r = CallBooleanMethodV(t, o, m, a); va_end(a); return r; } uint64_t callIntMethodA(Thread* t, uintptr_t* arguments) { jobject o = reinterpret_cast(arguments[0]); jmethodID m = arguments[1]; const jvalue* a = reinterpret_cast(arguments[2]); return cast(t, t->m->processor->invokeArray(t, getMethod(t, m), *o, a)) ->value(); } jboolean JNICALL CallBooleanMethodA(Thread* t, jobject o, jmethodID m, const jvalue* a) { uintptr_t arguments[] = {reinterpret_cast(o), m, reinterpret_cast(a)}; return run(t, callIntMethodA, arguments) != 0; } jbyte JNICALL CallByteMethodV(Thread* t, jobject o, jmethodID m, va_list a) { uintptr_t arguments[] = {reinterpret_cast(o), m, reinterpret_cast(VA_LIST(a))}; return run(t, callIntMethodV, arguments); } jbyte JNICALL CallByteMethod(Thread* t, jobject o, jmethodID m, ...) { va_list a; va_start(a, m); jbyte r = CallByteMethodV(t, o, m, a); va_end(a); return r; } jbyte JNICALL CallByteMethodA(Thread* t, jobject o, jmethodID m, const jvalue* a) { uintptr_t arguments[] = {reinterpret_cast(o), m, reinterpret_cast(a)}; return run(t, callIntMethodA, arguments); } jchar JNICALL CallCharMethodV(Thread* t, jobject o, jmethodID m, va_list a) { uintptr_t arguments[] = {reinterpret_cast(o), m, reinterpret_cast(VA_LIST(a))}; return run(t, callIntMethodV, arguments); } jchar JNICALL CallCharMethod(Thread* t, jobject o, jmethodID m, ...) { va_list a; va_start(a, m); jchar r = CallCharMethodV(t, o, m, a); va_end(a); return r; } jchar JNICALL CallCharMethodA(Thread* t, jobject o, jmethodID m, const jvalue* a) { uintptr_t arguments[] = {reinterpret_cast(o), m, reinterpret_cast(a)}; return run(t, callIntMethodA, arguments); } jshort JNICALL CallShortMethodV(Thread* t, jobject o, jmethodID m, va_list a) { uintptr_t arguments[] = {reinterpret_cast(o), m, reinterpret_cast(VA_LIST(a))}; return run(t, callIntMethodV, arguments); } jshort JNICALL CallShortMethod(Thread* t, jobject o, jmethodID m, ...) { va_list a; va_start(a, m); jshort r = CallShortMethodV(t, o, m, a); va_end(a); return r; } jshort JNICALL CallShortMethodA(Thread* t, jobject o, jmethodID m, const jvalue* a) { uintptr_t arguments[] = {reinterpret_cast(o), m, reinterpret_cast(a)}; return run(t, callIntMethodA, arguments); } jint JNICALL CallIntMethodV(Thread* t, jobject o, jmethodID m, va_list a) { uintptr_t arguments[] = {reinterpret_cast(o), m, reinterpret_cast(VA_LIST(a))}; return run(t, callIntMethodV, arguments); } jint JNICALL CallIntMethod(Thread* t, jobject o, jmethodID m, ...) { va_list a; va_start(a, m); jint r = CallIntMethodV(t, o, m, a); va_end(a); return r; } jint JNICALL CallIntMethodA(Thread* t, jobject o, jmethodID m, const jvalue* a) { uintptr_t arguments[] = {reinterpret_cast(o), m, reinterpret_cast(a)}; return run(t, callIntMethodA, arguments); } uint64_t callLongMethodV(Thread* t, uintptr_t* arguments) { jobject o = reinterpret_cast(arguments[0]); jmethodID m = arguments[1]; va_list* a = reinterpret_cast(arguments[2]); return cast( t, t->m->processor->invokeList(t, getMethod(t, m), *o, true, *a)) ->value(); } jlong JNICALL CallLongMethodV(Thread* t, jobject o, jmethodID m, va_list a) { uintptr_t arguments[] = {reinterpret_cast(o), m, reinterpret_cast(VA_LIST(a))}; return run(t, callLongMethodV, arguments); } jlong JNICALL CallLongMethod(Thread* t, jobject o, jmethodID m, ...) { va_list a; va_start(a, m); jlong r = CallLongMethodV(t, o, m, a); va_end(a); return r; } uint64_t callLongMethodA(Thread* t, uintptr_t* arguments) { jobject o = reinterpret_cast(arguments[0]); jmethodID m = arguments[1]; const jvalue* a = reinterpret_cast(arguments[2]); return cast(t, t->m->processor->invokeArray(t, getMethod(t, m), *o, a)) ->value(); } jlong JNICALL CallLongMethodA(Thread* t, jobject o, jmethodID m, const jvalue* a) { uintptr_t arguments[] = {reinterpret_cast(o), m, reinterpret_cast(a)}; return run(t, callLongMethodA, arguments); } jfloat JNICALL CallFloatMethodV(Thread* t, jobject o, jmethodID m, va_list a) { uintptr_t arguments[] = {reinterpret_cast(o), m, reinterpret_cast(VA_LIST(a))}; return bitsToFloat(run(t, callIntMethodV, arguments)); } jfloat JNICALL CallFloatMethod(Thread* t, jobject o, jmethodID m, ...) { va_list a; va_start(a, m); jfloat r = CallFloatMethodV(t, o, m, a); va_end(a); return r; } jfloat JNICALL CallFloatMethodA(Thread* t, jobject o, jmethodID m, const jvalue* a) { uintptr_t arguments[] = {reinterpret_cast(o), m, reinterpret_cast(a)}; return bitsToFloat(run(t, callIntMethodA, arguments)); } jdouble JNICALL CallDoubleMethodV(Thread* t, jobject o, jmethodID m, va_list a) { uintptr_t arguments[] = {reinterpret_cast(o), m, reinterpret_cast(VA_LIST(a))}; return bitsToDouble(run(t, callLongMethodV, arguments)); } jdouble JNICALL CallDoubleMethod(Thread* t, jobject o, jmethodID m, ...) { va_list a; va_start(a, m); jdouble r = CallDoubleMethodV(t, o, m, a); va_end(a); return r; } jdouble JNICALL CallDoubleMethodA(Thread* t, jobject o, jmethodID m, const jvalue* a) { uintptr_t arguments[] = {reinterpret_cast(o), m, reinterpret_cast(a)}; return bitsToDouble(run(t, callLongMethodA, arguments)); } uint64_t callVoidMethodV(Thread* t, uintptr_t* arguments) { jobject o = reinterpret_cast(arguments[0]); jmethodID m = arguments[1]; va_list* a = reinterpret_cast(arguments[2]); t->m->processor->invokeList(t, getMethod(t, m), *o, true, *a); return 0; } void JNICALL CallVoidMethodV(Thread* t, jobject o, jmethodID m, va_list a) { uintptr_t arguments[] = {reinterpret_cast(o), m, reinterpret_cast(VA_LIST(a))}; run(t, callVoidMethodV, arguments); } void JNICALL CallVoidMethod(Thread* t, jobject o, jmethodID m, ...) { va_list a; va_start(a, m); CallVoidMethodV(t, o, m, a); va_end(a); } uint64_t callVoidMethodA(Thread* t, uintptr_t* arguments) { jobject o = reinterpret_cast(arguments[0]); jmethodID m = arguments[1]; const jvalue* a = reinterpret_cast(arguments[2]); t->m->processor->invokeArray(t, getMethod(t, m), *o, a); return 0; } void JNICALL CallVoidMethodA(Thread* t, jobject o, jmethodID m, const jvalue* a) { uintptr_t arguments[] = {reinterpret_cast(o), m, reinterpret_cast(a)}; run(t, callVoidMethodA, arguments); } GcMethod* getStaticMethod(Thread* t, jmethodID m) { assertT(t, m); GcMethod* method = cast(t, roots(t)->jNIMethodTable()->body()[m - 1]); assertT(t, method->flags() & ACC_STATIC); return method; } uint64_t callStaticObjectMethodV(Thread* t, uintptr_t* arguments) { jmethodID m = arguments[0]; va_list* a = reinterpret_cast(arguments[1]); return reinterpret_cast(makeLocalReference( t, t->m->processor->invokeList(t, getStaticMethod(t, m), 0, true, *a))); } jobject JNICALL CallStaticObjectMethodV(Thread* t, jclass, jmethodID m, va_list a) { uintptr_t arguments[] = {m, reinterpret_cast(VA_LIST(a))}; return reinterpret_cast(run(t, callStaticObjectMethodV, arguments)); } jobject JNICALL CallStaticObjectMethod(Thread* t, jclass c, jmethodID m, ...) { va_list a; va_start(a, m); jobject r = CallStaticObjectMethodV(t, c, m, a); va_end(a); return r; } uint64_t callStaticObjectMethodA(Thread* t, uintptr_t* arguments) { jmethodID m = arguments[0]; const jvalue* a = reinterpret_cast(arguments[1]); return reinterpret_cast(makeLocalReference( t, t->m->processor->invokeArray(t, getStaticMethod(t, m), 0, a))); } jobject JNICALL CallStaticObjectMethodA(Thread* t, jclass, jmethodID m, const jvalue* a) { uintptr_t arguments[] = {m, reinterpret_cast(a)}; return reinterpret_cast(run(t, callStaticObjectMethodA, arguments)); } uint64_t callStaticIntMethodV(Thread* t, uintptr_t* arguments) { jmethodID m = arguments[0]; va_list* a = reinterpret_cast(arguments[1]); return cast(t, t->m->processor->invokeList( t, getStaticMethod(t, m), 0, true, *a))->value(); } jboolean JNICALL CallStaticBooleanMethodV(Thread* t, jclass, jmethodID m, va_list a) { uintptr_t arguments[] = {m, reinterpret_cast(VA_LIST(a))}; return run(t, callStaticIntMethodV, arguments) != 0; } jboolean JNICALL CallStaticBooleanMethod(Thread* t, jclass c, jmethodID m, ...) { va_list a; va_start(a, m); jboolean r = CallStaticBooleanMethodV(t, c, m, a); va_end(a); return r; } uint64_t callStaticIntMethodA(Thread* t, uintptr_t* arguments) { jmethodID m = arguments[0]; const jvalue* a = reinterpret_cast(arguments[1]); return cast( t, t->m->processor->invokeArray(t, getStaticMethod(t, m), 0, a)) ->value(); } jboolean JNICALL CallStaticBooleanMethodA(Thread* t, jclass, jmethodID m, const jvalue* a) { uintptr_t arguments[] = {m, reinterpret_cast(a)}; return run(t, callStaticIntMethodA, arguments) != 0; } jbyte JNICALL CallStaticByteMethodV(Thread* t, jclass, jmethodID m, va_list a) { uintptr_t arguments[] = {m, reinterpret_cast(VA_LIST(a))}; return run(t, callStaticIntMethodV, arguments); } jbyte JNICALL CallStaticByteMethod(Thread* t, jclass c, jmethodID m, ...) { va_list a; va_start(a, m); jbyte r = CallStaticByteMethodV(t, c, m, a); va_end(a); return r; } jbyte JNICALL CallStaticByteMethodA(Thread* t, jclass, jmethodID m, const jvalue* a) { uintptr_t arguments[] = {m, reinterpret_cast(a)}; return run(t, callStaticIntMethodA, arguments); } jchar JNICALL CallStaticCharMethodV(Thread* t, jclass, jmethodID m, va_list a) { uintptr_t arguments[] = {m, reinterpret_cast(VA_LIST(a))}; return run(t, callStaticIntMethodV, arguments); } jchar JNICALL CallStaticCharMethod(Thread* t, jclass c, jmethodID m, ...) { va_list a; va_start(a, m); jchar r = CallStaticCharMethodV(t, c, m, a); va_end(a); return r; } jchar JNICALL CallStaticCharMethodA(Thread* t, jclass, jmethodID m, const jvalue* a) { uintptr_t arguments[] = {m, reinterpret_cast(a)}; return run(t, callStaticIntMethodA, arguments); } jshort JNICALL CallStaticShortMethodV(Thread* t, jclass, jmethodID m, va_list a) { uintptr_t arguments[] = {m, reinterpret_cast(VA_LIST(a))}; return run(t, callStaticIntMethodV, arguments); } jshort JNICALL CallStaticShortMethod(Thread* t, jclass c, jmethodID m, ...) { va_list a; va_start(a, m); jshort r = CallStaticShortMethodV(t, c, m, a); va_end(a); return r; } jshort JNICALL CallStaticShortMethodA(Thread* t, jclass, jmethodID m, const jvalue* a) { uintptr_t arguments[] = {m, reinterpret_cast(a)}; return run(t, callStaticIntMethodA, arguments); } jint JNICALL CallStaticIntMethodV(Thread* t, jclass, jmethodID m, va_list a) { uintptr_t arguments[] = {m, reinterpret_cast(VA_LIST(a))}; return run(t, callStaticIntMethodV, arguments); } jint JNICALL CallStaticIntMethod(Thread* t, jclass c, jmethodID m, ...) { va_list a; va_start(a, m); jint r = CallStaticIntMethodV(t, c, m, a); va_end(a); return r; } jint JNICALL CallStaticIntMethodA(Thread* t, jclass, jmethodID m, const jvalue* a) { uintptr_t arguments[] = {m, reinterpret_cast(a)}; return run(t, callStaticIntMethodA, arguments); } uint64_t callStaticLongMethodV(Thread* t, uintptr_t* arguments) { jmethodID m = arguments[0]; va_list* a = reinterpret_cast(arguments[1]); return cast(t, t->m->processor->invokeList( t, getStaticMethod(t, m), 0, true, *a))->value(); } jlong JNICALL CallStaticLongMethodV(Thread* t, jclass, jmethodID m, va_list a) { uintptr_t arguments[] = {m, reinterpret_cast(VA_LIST(a))}; return run(t, callStaticLongMethodV, arguments); } jlong JNICALL CallStaticLongMethod(Thread* t, jclass c, jmethodID m, ...) { va_list a; va_start(a, m); jlong r = CallStaticLongMethodV(t, c, m, a); va_end(a); return r; } uint64_t callStaticLongMethodA(Thread* t, uintptr_t* arguments) { jmethodID m = arguments[0]; const jvalue* a = reinterpret_cast(arguments[1]); return cast( t, t->m->processor->invokeArray(t, getStaticMethod(t, m), 0, a)) ->value(); } jlong JNICALL CallStaticLongMethodA(Thread* t, jclass, jmethodID m, const jvalue* a) { uintptr_t arguments[] = {m, reinterpret_cast(a)}; return run(t, callStaticLongMethodA, arguments); } jfloat JNICALL CallStaticFloatMethodV(Thread* t, jclass, jmethodID m, va_list a) { uintptr_t arguments[] = {m, reinterpret_cast(VA_LIST(a))}; return bitsToFloat(run(t, callStaticIntMethodV, arguments)); } jfloat JNICALL CallStaticFloatMethod(Thread* t, jclass c, jmethodID m, ...) { va_list a; va_start(a, m); jfloat r = CallStaticFloatMethodV(t, c, m, a); va_end(a); return r; } jfloat JNICALL CallStaticFloatMethodA(Thread* t, jclass, jmethodID m, const jvalue* a) { uintptr_t arguments[] = {m, reinterpret_cast(a)}; return bitsToFloat(run(t, callStaticIntMethodA, arguments)); } jdouble JNICALL CallStaticDoubleMethodV(Thread* t, jclass, jmethodID m, va_list a) { uintptr_t arguments[] = {m, reinterpret_cast(VA_LIST(a))}; return bitsToDouble(run(t, callStaticLongMethodV, arguments)); } jdouble JNICALL CallStaticDoubleMethod(Thread* t, jclass c, jmethodID m, ...) { va_list a; va_start(a, m); jdouble r = CallStaticDoubleMethodV(t, c, m, a); va_end(a); return r; } jdouble JNICALL CallStaticDoubleMethodA(Thread* t, jclass, jmethodID m, const jvalue* a) { uintptr_t arguments[] = {m, reinterpret_cast(a)}; return bitsToDouble(run(t, callStaticLongMethodA, arguments)); } uint64_t callStaticVoidMethodV(Thread* t, uintptr_t* arguments) { jmethodID m = arguments[0]; va_list* a = reinterpret_cast(arguments[1]); t->m->processor->invokeList(t, getStaticMethod(t, m), 0, true, *a); return 0; } void JNICALL CallStaticVoidMethodV(Thread* t, jclass, jmethodID m, va_list a) { uintptr_t arguments[] = {m, reinterpret_cast(VA_LIST(a))}; run(t, callStaticVoidMethodV, arguments); } void JNICALL CallStaticVoidMethod(Thread* t, jclass c, jmethodID m, ...) { va_list a; va_start(a, m); CallStaticVoidMethodV(t, c, m, a); va_end(a); } uint64_t callStaticVoidMethodA(Thread* t, uintptr_t* arguments) { jmethodID m = arguments[0]; const jvalue* a = reinterpret_cast(arguments[1]); t->m->processor->invokeArray(t, getStaticMethod(t, m), 0, a); return 0; } void JNICALL CallStaticVoidMethodA(Thread* t, jclass, jmethodID m, const jvalue* a) { uintptr_t arguments[] = {m, reinterpret_cast(a)}; run(t, callStaticVoidMethodA, arguments); } jint fieldID(Thread* t, GcField* field) { int id = field->nativeID(); loadMemoryBarrier(); if (id == 0) { PROTECT(t, field); ACQUIRE(t, t->m->referenceLock); if (field->nativeID() == 0) { GcVector* v = vectorAppend(t, roots(t)->jNIFieldTable(), field); // sequence point, for gc (don't recombine statements) roots(t)->setJNIFieldTable(t, v); storeStoreMemoryBarrier(); field->nativeID() = roots(t)->jNIFieldTable()->size(); } } return field->nativeID(); } uint64_t getFieldID(Thread* t, uintptr_t* arguments) { jclass c = reinterpret_cast(arguments[0]); const char* name = reinterpret_cast(arguments[1]); const char* spec = reinterpret_cast(arguments[2]); return fieldID(t, resolveField(t, (*c)->vmClass(), name, spec)); } jfieldID JNICALL GetFieldID(Thread* t, jclass c, const char* name, const char* spec) { uintptr_t arguments[] = {reinterpret_cast(c), reinterpret_cast(name), reinterpret_cast(spec)}; return run(t, getFieldID, arguments); } jfieldID JNICALL GetStaticFieldID(Thread* t, jclass c, const char* name, const char* spec) { uintptr_t arguments[] = {reinterpret_cast(c), reinterpret_cast(name), reinterpret_cast(spec)}; return run(t, getFieldID, arguments); } GcField* getField(Thread* t, jfieldID f) { assertT(t, f); GcField* field = cast(t, roots(t)->jNIFieldTable()->body()[f - 1]); assertT(t, (field->flags() & ACC_STATIC) == 0); return field; } uint64_t getObjectField(Thread* t, uintptr_t* arguments) { jobject o = reinterpret_cast(arguments[0]); GcField* field = getField(t, arguments[1]); PROTECT(t, field); ACQUIRE_FIELD_FOR_READ(t, field); return reinterpret_cast( makeLocalReference(t, fieldAtOffset(*o, field->offset()))); } jobject JNICALL GetObjectField(Thread* t, jobject o, jfieldID field) { uintptr_t arguments[] = {reinterpret_cast(o), field}; return reinterpret_cast(run(t, getObjectField, arguments)); } uint64_t getBooleanField(Thread* t, uintptr_t* arguments) { jobject o = reinterpret_cast(arguments[0]); GcField* field = getField(t, arguments[1]); PROTECT(t, field); ACQUIRE_FIELD_FOR_READ(t, field); return fieldAtOffset(*o, field->offset()); } jboolean JNICALL GetBooleanField(Thread* t, jobject o, jfieldID field) { uintptr_t arguments[] = {reinterpret_cast(o), field}; return run(t, getBooleanField, arguments); } uint64_t getByteField(Thread* t, uintptr_t* arguments) { jobject o = reinterpret_cast(arguments[0]); GcField* field = getField(t, arguments[1]); PROTECT(t, field); ACQUIRE_FIELD_FOR_READ(t, field); return fieldAtOffset(*o, field->offset()); } jbyte JNICALL GetByteField(Thread* t, jobject o, jfieldID field) { uintptr_t arguments[] = {reinterpret_cast(o), field}; return run(t, getByteField, arguments); } uint64_t getCharField(Thread* t, uintptr_t* arguments) { jobject o = reinterpret_cast(arguments[0]); GcField* field = getField(t, arguments[1]); PROTECT(t, field); ACQUIRE_FIELD_FOR_READ(t, field); return fieldAtOffset(*o, field->offset()); } jchar JNICALL GetCharField(Thread* t, jobject o, jfieldID field) { uintptr_t arguments[] = {reinterpret_cast(o), field}; return run(t, getCharField, arguments); } uint64_t getShortField(Thread* t, uintptr_t* arguments) { jobject o = reinterpret_cast(arguments[0]); GcField* field = getField(t, arguments[1]); PROTECT(t, field); ACQUIRE_FIELD_FOR_READ(t, field); return fieldAtOffset(*o, field->offset()); } jshort JNICALL GetShortField(Thread* t, jobject o, jfieldID field) { uintptr_t arguments[] = {reinterpret_cast(o), field}; return run(t, getShortField, arguments); } uint64_t getIntField(Thread* t, uintptr_t* arguments) { jobject o = reinterpret_cast(arguments[0]); GcField* field = getField(t, arguments[1]); PROTECT(t, field); ACQUIRE_FIELD_FOR_READ(t, field); return fieldAtOffset(*o, field->offset()); } jint JNICALL GetIntField(Thread* t, jobject o, jfieldID field) { uintptr_t arguments[] = {reinterpret_cast(o), field}; return run(t, getIntField, arguments); } uint64_t getLongField(Thread* t, uintptr_t* arguments) { jobject o = reinterpret_cast(arguments[0]); GcField* field = getField(t, arguments[1]); PROTECT(t, field); ACQUIRE_FIELD_FOR_READ(t, field); return fieldAtOffset(*o, field->offset()); } jlong JNICALL GetLongField(Thread* t, jobject o, jfieldID field) { uintptr_t arguments[] = {reinterpret_cast(o), field}; return run(t, getLongField, arguments); } uint64_t getFloatField(Thread* t, uintptr_t* arguments) { jobject o = reinterpret_cast(arguments[0]); GcField* field = getField(t, arguments[1]); PROTECT(t, field); ACQUIRE_FIELD_FOR_READ(t, field); return floatToBits(fieldAtOffset(*o, field->offset())); } jfloat JNICALL GetFloatField(Thread* t, jobject o, jfieldID field) { uintptr_t arguments[] = {reinterpret_cast(o), field}; return bitsToFloat(run(t, getFloatField, arguments)); } uint64_t getDoubleField(Thread* t, uintptr_t* arguments) { jobject o = reinterpret_cast(arguments[0]); GcField* field = getField(t, arguments[1]); PROTECT(t, field); ACQUIRE_FIELD_FOR_READ(t, field); return doubleToBits(fieldAtOffset(*o, field->offset())); } jdouble JNICALL GetDoubleField(Thread* t, jobject o, jfieldID field) { uintptr_t arguments[] = {reinterpret_cast(o), field}; return bitsToDouble(run(t, getDoubleField, arguments)); } uint64_t setObjectField(Thread* t, uintptr_t* arguments) { jobject o = reinterpret_cast(arguments[0]); GcField* field = getField(t, arguments[1]); jobject v = reinterpret_cast(arguments[2]); PROTECT(t, field); ACQUIRE_FIELD_FOR_WRITE(t, field); setField(t, *o, field->offset(), (v ? *v : 0)); return 1; } void JNICALL SetObjectField(Thread* t, jobject o, jfieldID field, jobject v) { uintptr_t arguments[] = {reinterpret_cast(o), field, reinterpret_cast(v)}; run(t, setObjectField, arguments); } uint64_t setBooleanField(Thread* t, uintptr_t* arguments) { jobject o = reinterpret_cast(arguments[0]); GcField* field = getField(t, arguments[1]); jboolean v = arguments[2]; PROTECT(t, field); ACQUIRE_FIELD_FOR_WRITE(t, field); fieldAtOffset(*o, field->offset()) = v; return 1; } void JNICALL SetBooleanField(Thread* t, jobject o, jfieldID field, jboolean v) { uintptr_t arguments[] = {reinterpret_cast(o), field, v}; run(t, setBooleanField, arguments); } uint64_t setByteField(Thread* t, uintptr_t* arguments) { jobject o = reinterpret_cast(arguments[0]); GcField* field = getField(t, arguments[1]); jbyte v = arguments[2]; PROTECT(t, field); ACQUIRE_FIELD_FOR_WRITE(t, field); fieldAtOffset(*o, field->offset()) = v; return 1; } void JNICALL SetByteField(Thread* t, jobject o, jfieldID field, jbyte v) { uintptr_t arguments[] = {reinterpret_cast(o), field, static_cast(v)}; run(t, setByteField, arguments); } uint64_t setCharField(Thread* t, uintptr_t* arguments) { jobject o = reinterpret_cast(arguments[0]); GcField* field = getField(t, arguments[1]); jchar v = arguments[2]; PROTECT(t, field); ACQUIRE_FIELD_FOR_WRITE(t, field); fieldAtOffset(*o, field->offset()) = v; return 1; } void JNICALL SetCharField(Thread* t, jobject o, jfieldID field, jchar v) { uintptr_t arguments[] = {reinterpret_cast(o), field, v}; run(t, setCharField, arguments); } uint64_t setShortField(Thread* t, uintptr_t* arguments) { jobject o = reinterpret_cast(arguments[0]); GcField* field = getField(t, arguments[1]); jshort v = arguments[2]; PROTECT(t, field); ACQUIRE_FIELD_FOR_WRITE(t, field); fieldAtOffset(*o, field->offset()) = v; return 1; } void JNICALL SetShortField(Thread* t, jobject o, jfieldID field, jshort v) { uintptr_t arguments[] = {reinterpret_cast(o), field, static_cast(v)}; run(t, setShortField, arguments); } uint64_t setIntField(Thread* t, uintptr_t* arguments) { jobject o = reinterpret_cast(arguments[0]); GcField* field = getField(t, arguments[1]); jint v = arguments[2]; PROTECT(t, field); ACQUIRE_FIELD_FOR_WRITE(t, field); fieldAtOffset(*o, field->offset()) = v; return 1; } void JNICALL SetIntField(Thread* t, jobject o, jfieldID field, jint v) { uintptr_t arguments[] = {reinterpret_cast(o), field, static_cast(v)}; run(t, setIntField, arguments); } uint64_t setLongField(Thread* t, uintptr_t* arguments) { jobject o = reinterpret_cast(arguments[0]); GcField* field = getField(t, arguments[1]); jlong v; memcpy(&v, arguments + 2, sizeof(jlong)); PROTECT(t, field); ACQUIRE_FIELD_FOR_WRITE(t, field); fieldAtOffset(*o, field->offset()) = v; return 1; } void JNICALL SetLongField(Thread* t, jobject o, jfieldID field, jlong v) { uintptr_t arguments[2 + (sizeof(jlong) / BytesPerWord)]; arguments[0] = reinterpret_cast(o); arguments[1] = field; memcpy(arguments + 2, &v, sizeof(jlong)); run(t, setLongField, arguments); } uint64_t setFloatField(Thread* t, uintptr_t* arguments) { jobject o = reinterpret_cast(arguments[0]); GcField* field = getField(t, arguments[1]); jfloat v = bitsToFloat(arguments[2]); PROTECT(t, field); ACQUIRE_FIELD_FOR_WRITE(t, field); fieldAtOffset(*o, field->offset()) = v; return 1; } void JNICALL SetFloatField(Thread* t, jobject o, jfieldID field, jfloat v) { uintptr_t arguments[] = {reinterpret_cast(o), field, floatToBits(v)}; run(t, setFloatField, arguments); } uint64_t setDoubleField(Thread* t, uintptr_t* arguments) { jobject o = reinterpret_cast(arguments[0]); GcField* field = getField(t, arguments[1]); jdouble v; memcpy(&v, arguments + 2, sizeof(jdouble)); PROTECT(t, field); ACQUIRE_FIELD_FOR_WRITE(t, field); fieldAtOffset(*o, field->offset()) = v; return 1; } void JNICALL SetDoubleField(Thread* t, jobject o, jfieldID field, jdouble v) { uintptr_t arguments[2 + (sizeof(jdouble) / BytesPerWord)]; arguments[0] = reinterpret_cast(o); arguments[1] = field; memcpy(arguments + 2, &v, sizeof(jdouble)); run(t, setDoubleField, arguments); } GcField* getStaticField(Thread* t, jfieldID f) { assertT(t, f); GcField* field = cast(t, roots(t)->jNIFieldTable()->body()[f - 1]); assertT(t, field->flags() & ACC_STATIC); return field; } uint64_t getStaticObjectField(Thread* t, uintptr_t* arguments) { GcJclass* c = cast(t, *reinterpret_cast(arguments[0])); initClass(t, c->vmClass()); GcField* field = getStaticField(t, arguments[1]); PROTECT(t, field); ACQUIRE_FIELD_FOR_READ(t, field); return reinterpret_cast(makeLocalReference( t, fieldAtOffset(c->vmClass()->staticTable(), field->offset()))); } jobject JNICALL GetStaticObjectField(Thread* t, jclass c, jfieldID field) { uintptr_t arguments[] = {reinterpret_cast(c), field}; return reinterpret_cast(run(t, getStaticObjectField, arguments)); } uint64_t getStaticBooleanField(Thread* t, uintptr_t* arguments) { GcJclass* c = cast(t, *reinterpret_cast(arguments[0])); initClass(t, c->vmClass()); GcField* field = getStaticField(t, arguments[1]); PROTECT(t, field); ACQUIRE_FIELD_FOR_READ(t, field); return fieldAtOffset(c->vmClass()->staticTable(), field->offset()); } jboolean JNICALL GetStaticBooleanField(Thread* t, jclass c, jfieldID field) { uintptr_t arguments[] = {reinterpret_cast(c), field}; return run(t, getStaticBooleanField, arguments); } uint64_t getStaticByteField(Thread* t, uintptr_t* arguments) { GcJclass* c = cast(t, *reinterpret_cast(arguments[0])); initClass(t, c->vmClass()); GcField* field = getStaticField(t, arguments[1]); PROTECT(t, field); ACQUIRE_FIELD_FOR_READ(t, field); return fieldAtOffset(c->vmClass()->staticTable(), field->offset()); } jbyte JNICALL GetStaticByteField(Thread* t, jclass c, jfieldID field) { uintptr_t arguments[] = {reinterpret_cast(c), field}; return run(t, getStaticByteField, arguments); } uint64_t getStaticCharField(Thread* t, uintptr_t* arguments) { GcJclass* c = cast(t, *reinterpret_cast(arguments[0])); initClass(t, c->vmClass()); GcField* field = getStaticField(t, arguments[1]); PROTECT(t, field); ACQUIRE_FIELD_FOR_READ(t, field); return fieldAtOffset(c->vmClass()->staticTable(), field->offset()); } jchar JNICALL GetStaticCharField(Thread* t, jclass c, jfieldID field) { uintptr_t arguments[] = {reinterpret_cast(c), field}; return run(t, getStaticCharField, arguments); } uint64_t getStaticShortField(Thread* t, uintptr_t* arguments) { GcJclass* c = cast(t, *reinterpret_cast(arguments[0])); initClass(t, c->vmClass()); GcField* field = getStaticField(t, arguments[1]); PROTECT(t, field); ACQUIRE_FIELD_FOR_READ(t, field); return fieldAtOffset(c->vmClass()->staticTable(), field->offset()); } jshort JNICALL GetStaticShortField(Thread* t, jclass c, jfieldID field) { uintptr_t arguments[] = {reinterpret_cast(c), field}; return run(t, getStaticShortField, arguments); } uint64_t getStaticIntField(Thread* t, uintptr_t* arguments) { GcJclass* c = cast(t, *reinterpret_cast(arguments[0])); initClass(t, c->vmClass()); GcField* field = getStaticField(t, arguments[1]); PROTECT(t, field); ACQUIRE_FIELD_FOR_READ(t, field); return fieldAtOffset(c->vmClass()->staticTable(), field->offset()); } jint JNICALL GetStaticIntField(Thread* t, jclass c, jfieldID field) { uintptr_t arguments[] = {reinterpret_cast(c), field}; return run(t, getStaticIntField, arguments); } uint64_t getStaticLongField(Thread* t, uintptr_t* arguments) { GcJclass* c = cast(t, *reinterpret_cast(arguments[0])); initClass(t, c->vmClass()); GcField* field = getStaticField(t, arguments[1]); PROTECT(t, field); ACQUIRE_FIELD_FOR_READ(t, field); return fieldAtOffset(c->vmClass()->staticTable(), field->offset()); } jlong JNICALL GetStaticLongField(Thread* t, jclass c, jfieldID field) { uintptr_t arguments[] = {reinterpret_cast(c), field}; return run(t, getStaticLongField, arguments); } uint64_t getStaticFloatField(Thread* t, uintptr_t* arguments) { GcJclass* c = cast(t, *reinterpret_cast(arguments[0])); initClass(t, c->vmClass()); GcField* field = getStaticField(t, arguments[1]); PROTECT(t, field); ACQUIRE_FIELD_FOR_READ(t, field); return floatToBits( fieldAtOffset(c->vmClass()->staticTable(), field->offset())); } jfloat JNICALL GetStaticFloatField(Thread* t, jclass c, jfieldID field) { uintptr_t arguments[] = {reinterpret_cast(c), field}; return bitsToFloat(run(t, getStaticFloatField, arguments)); } uint64_t getStaticDoubleField(Thread* t, uintptr_t* arguments) { GcJclass* c = cast(t, *reinterpret_cast(arguments[0])); initClass(t, c->vmClass()); GcField* field = getStaticField(t, arguments[1]); PROTECT(t, field); ACQUIRE_FIELD_FOR_READ(t, field); return doubleToBits( fieldAtOffset(c->vmClass()->staticTable(), field->offset())); } jdouble JNICALL GetStaticDoubleField(Thread* t, jclass c, jfieldID field) { uintptr_t arguments[] = {reinterpret_cast(c), field}; return bitsToDouble(run(t, getStaticDoubleField, arguments)); } uint64_t setStaticObjectField(Thread* t, uintptr_t* arguments) { GcJclass* c = cast(t, *reinterpret_cast(arguments[0])); initClass(t, c->vmClass()); GcField* field = getStaticField(t, arguments[1]); jobject v = reinterpret_cast(arguments[2]); PROTECT(t, field); ACQUIRE_FIELD_FOR_WRITE(t, field); setField(t, c->vmClass()->staticTable(), field->offset(), (v ? *v : 0)); return 1; } void JNICALL SetStaticObjectField(Thread* t, jclass c, jfieldID field, jobject v) { uintptr_t arguments[] = {reinterpret_cast(c), field, reinterpret_cast(v)}; run(t, setStaticObjectField, arguments); } uint64_t setStaticBooleanField(Thread* t, uintptr_t* arguments) { GcJclass* c = cast(t, *reinterpret_cast(arguments[0])); initClass(t, c->vmClass()); GcField* field = getStaticField(t, arguments[1]); jboolean v = arguments[2]; PROTECT(t, field); ACQUIRE_FIELD_FOR_WRITE(t, field); fieldAtOffset(c->vmClass()->staticTable(), field->offset()) = v; return 1; } void JNICALL SetStaticBooleanField(Thread* t, jclass c, jfieldID field, jboolean v) { uintptr_t arguments[] = {reinterpret_cast(c), field, v}; run(t, setStaticBooleanField, arguments); } uint64_t setStaticByteField(Thread* t, uintptr_t* arguments) { GcJclass* c = cast(t, *reinterpret_cast(arguments[0])); initClass(t, c->vmClass()); GcField* field = getStaticField(t, arguments[1]); jbyte v = arguments[2]; PROTECT(t, field); ACQUIRE_FIELD_FOR_WRITE(t, field); fieldAtOffset(c->vmClass()->staticTable(), field->offset()) = v; return 1; } void JNICALL SetStaticByteField(Thread* t, jclass c, jfieldID field, jbyte v) { uintptr_t arguments[] = {reinterpret_cast(c), field, static_cast(v)}; run(t, setStaticByteField, arguments); } uint64_t setStaticCharField(Thread* t, uintptr_t* arguments) { GcJclass* c = cast(t, *reinterpret_cast(arguments[0])); initClass(t, c->vmClass()); GcField* field = getStaticField(t, arguments[1]); jchar v = arguments[2]; PROTECT(t, field); ACQUIRE_FIELD_FOR_WRITE(t, field); fieldAtOffset(c->vmClass()->staticTable(), field->offset()) = v; return 1; } void JNICALL SetStaticCharField(Thread* t, jclass c, jfieldID field, jchar v) { uintptr_t arguments[] = {reinterpret_cast(c), field, v}; run(t, setStaticCharField, arguments); } uint64_t setStaticShortField(Thread* t, uintptr_t* arguments) { GcJclass* c = cast(t, *reinterpret_cast(arguments[0])); initClass(t, c->vmClass()); GcField* field = getStaticField(t, arguments[1]); jshort v = arguments[2]; PROTECT(t, field); ACQUIRE_FIELD_FOR_WRITE(t, field); fieldAtOffset(c->vmClass()->staticTable(), field->offset()) = v; return 1; } void JNICALL SetStaticShortField(Thread* t, jclass c, jfieldID field, jshort v) { uintptr_t arguments[] = {reinterpret_cast(c), field, static_cast(v)}; run(t, setStaticShortField, arguments); } uint64_t setStaticIntField(Thread* t, uintptr_t* arguments) { GcJclass* c = cast(t, *reinterpret_cast(arguments[0])); initClass(t, c->vmClass()); GcField* field = getStaticField(t, arguments[1]); jint v = arguments[2]; PROTECT(t, field); ACQUIRE_FIELD_FOR_WRITE(t, field); fieldAtOffset(c->vmClass()->staticTable(), field->offset()) = v; return 1; } void JNICALL SetStaticIntField(Thread* t, jclass c, jfieldID field, jint v) { uintptr_t arguments[] = {reinterpret_cast(c), field, static_cast(v)}; run(t, setStaticIntField, arguments); } uint64_t setStaticLongField(Thread* t, uintptr_t* arguments) { GcJclass* c = cast(t, *reinterpret_cast(arguments[0])); initClass(t, c->vmClass()); GcField* field = getStaticField(t, arguments[1]); jlong v; memcpy(&v, arguments + 2, sizeof(jlong)); PROTECT(t, field); ACQUIRE_FIELD_FOR_WRITE(t, field); fieldAtOffset(c->vmClass()->staticTable(), field->offset()) = v; return 1; } void JNICALL SetStaticLongField(Thread* t, jclass c, jfieldID field, jlong v) { uintptr_t arguments[2 + (sizeof(jlong) / BytesPerWord)]; arguments[0] = reinterpret_cast(c); arguments[1] = field; memcpy(arguments + 2, &v, sizeof(jlong)); run(t, setStaticLongField, arguments); } uint64_t setStaticFloatField(Thread* t, uintptr_t* arguments) { GcJclass* c = cast(t, *reinterpret_cast(arguments[0])); initClass(t, c->vmClass()); GcField* field = getStaticField(t, arguments[1]); jfloat v = bitsToFloat(arguments[2]); PROTECT(t, field); ACQUIRE_FIELD_FOR_WRITE(t, field); fieldAtOffset(c->vmClass()->staticTable(), field->offset()) = v; return 1; } void JNICALL SetStaticFloatField(Thread* t, jclass c, jfieldID field, jfloat v) { uintptr_t arguments[] = {reinterpret_cast(c), field, floatToBits(v)}; run(t, setStaticFloatField, arguments); } uint64_t setStaticDoubleField(Thread* t, uintptr_t* arguments) { GcJclass* c = cast(t, *reinterpret_cast(arguments[0])); initClass(t, c->vmClass()); GcField* field = getStaticField(t, arguments[1]); jdouble v; memcpy(&v, arguments + 2, sizeof(jdouble)); PROTECT(t, field); ACQUIRE_FIELD_FOR_WRITE(t, field); fieldAtOffset(c->vmClass()->staticTable(), field->offset()) = v; return 1; } void JNICALL SetStaticDoubleField(Thread* t, jclass c, jfieldID field, jdouble v) { uintptr_t arguments[2 + (sizeof(jdouble) / BytesPerWord)]; arguments[0] = reinterpret_cast(c); arguments[1] = field; memcpy(arguments + 2, &v, sizeof(jdouble)); run(t, setStaticDoubleField, arguments); } jobject JNICALL newGlobalRef(Thread* t, jobject o, bool weak) { ENTER(t, Thread::ActiveState); ACQUIRE(t, t->m->referenceLock); if (o) { for (Reference* r = t->m->jniReferences; r; r = r->next) { if (r->target == *o and r->weak == weak) { acquire(t, r); return &(r->target); } } Reference* r = new (t->m->heap->allocate(sizeof(Reference))) Reference(*o, &(t->m->jniReferences), weak); acquire(t, r); return &(r->target); } else { return 0; } } jobject JNICALL NewGlobalRef(Thread* t, jobject o) { return newGlobalRef(t, o, false); } void JNICALL DeleteGlobalRef(Thread* t, jobject r) { ENTER(t, Thread::ActiveState); ACQUIRE(t, t->m->referenceLock); if (r) { release(t, reinterpret_cast(r)); } } jobject JNICALL NewWeakGlobalRef(Thread* t, jobject o) { return newGlobalRef(t, o, true); } void JNICALL DeleteWeakGlobalRef(Thread* t, jobject r) { DeleteGlobalRef(t, r); } jint JNICALL EnsureLocalCapacity(Thread*, jint) { return 0; } jthrowable JNICALL ExceptionOccurred(Thread* t) { ENTER(t, Thread::ActiveState); return reinterpret_cast(makeLocalReference(t, t->exception)); } void JNICALL ExceptionDescribe(Thread* t) { ENTER(t, Thread::ActiveState); return printTrace(t, t->exception); } void JNICALL ExceptionClear(Thread* t) { ENTER(t, Thread::ActiveState); t->exception = 0; } uint64_t newObjectArray(Thread* t, uintptr_t* arguments) { jsize length = arguments[0]; jclass class_ = reinterpret_cast(arguments[1]); jobject init = reinterpret_cast(arguments[2]); object a = makeObjectArray(t, (*class_)->vmClass(), length); object value = (init ? *init : 0); for (jsize i = 0; i < length; ++i) { reinterpret_cast(a)->setBodyElement(t, i, value); } return reinterpret_cast(makeLocalReference(t, a)); } jobjectArray JNICALL NewObjectArray(Thread* t, jsize length, jclass class_, jobject init) { uintptr_t arguments[] = {static_cast(length), reinterpret_cast(class_), reinterpret_cast(init)}; return reinterpret_cast(run(t, newObjectArray, arguments)); } jobject JNICALL GetObjectArrayElement(Thread* t, jobjectArray array, jsize index) { ENTER(t, Thread::ActiveState); return makeLocalReference( t, objectArrayBody(t, reinterpret_cast(*array), index)); } void JNICALL SetObjectArrayElement(Thread* t, jobjectArray array, jsize index, jobject value) { ENTER(t, Thread::ActiveState); setField(t, reinterpret_cast(*array), ArrayBody + (index * BytesPerWord), (value ? *value : 0)); } uint64_t newArray(Thread* t, uintptr_t* arguments) { object (*constructor)(Thread*, unsigned) = reinterpret_cast(arguments[0]); jsize length = arguments[1]; return reinterpret_cast( makeLocalReference(t, constructor(t, length))); } jbooleanArray JNICALL NewBooleanArray(Thread* t, jsize length) { uintptr_t arguments[] = {reinterpret_cast(voidPointer(makeBooleanArray)), static_cast(length)}; return reinterpret_cast(run(t, newArray, arguments)); } object makeByteArray0(Thread* t, unsigned length) { return makeByteArray(t, length); } jbyteArray JNICALL NewByteArray(Thread* t, jsize length) { uintptr_t arguments[] = {reinterpret_cast(voidPointer(makeByteArray0)), static_cast(length)}; return reinterpret_cast(run(t, newArray, arguments)); } jcharArray JNICALL NewCharArray(Thread* t, jsize length) { uintptr_t arguments[] = {reinterpret_cast(voidPointer(makeCharArray)), static_cast(length)}; return reinterpret_cast(run(t, newArray, arguments)); } jshortArray JNICALL NewShortArray(Thread* t, jsize length) { uintptr_t arguments[] = {reinterpret_cast(voidPointer(makeShortArray)), static_cast(length)}; return reinterpret_cast(run(t, newArray, arguments)); } jintArray JNICALL NewIntArray(Thread* t, jsize length) { uintptr_t arguments[] = {reinterpret_cast(voidPointer(makeIntArray)), static_cast(length)}; return reinterpret_cast(run(t, newArray, arguments)); } jlongArray JNICALL NewLongArray(Thread* t, jsize length) { uintptr_t arguments[] = {reinterpret_cast(voidPointer(makeLongArray)), static_cast(length)}; return reinterpret_cast(run(t, newArray, arguments)); } jfloatArray JNICALL NewFloatArray(Thread* t, jsize length) { uintptr_t arguments[] = {reinterpret_cast(voidPointer(makeFloatArray)), static_cast(length)}; return reinterpret_cast(run(t, newArray, arguments)); } jdoubleArray JNICALL NewDoubleArray(Thread* t, jsize length) { uintptr_t arguments[] = {reinterpret_cast(voidPointer(makeDoubleArray)), static_cast(length)}; return reinterpret_cast(run(t, newArray, arguments)); } jboolean* JNICALL GetBooleanArrayElements(Thread* t, jbooleanArray array, jboolean* isCopy) { ENTER(t, Thread::ActiveState); unsigned size = (*array)->length() * sizeof(jboolean); jboolean* p = static_cast(t->m->heap->allocate(size)); if (size) { memcpy(p, (*array)->body().begin(), size); } if (isCopy) { *isCopy = true; } return p; } jbyte* JNICALL GetByteArrayElements(Thread* t, jbyteArray array, jboolean* isCopy) { ENTER(t, Thread::ActiveState); unsigned size = (*array)->length() * sizeof(jbyte); jbyte* p = static_cast(t->m->heap->allocate(size)); if (size) { memcpy(p, (*array)->body().begin(), size); } if (isCopy) { *isCopy = true; } return p; } jchar* JNICALL GetCharArrayElements(Thread* t, jcharArray array, jboolean* isCopy) { ENTER(t, Thread::ActiveState); unsigned size = (*array)->length() * sizeof(jchar); jchar* p = static_cast(t->m->heap->allocate(size)); if (size) { memcpy(p, (*array)->body().begin(), size); } if (isCopy) { *isCopy = true; } return p; } jshort* JNICALL GetShortArrayElements(Thread* t, jshortArray array, jboolean* isCopy) { ENTER(t, Thread::ActiveState); unsigned size = (*array)->length() * sizeof(jshort); jshort* p = static_cast(t->m->heap->allocate(size)); if (size) { memcpy(p, (*array)->body().begin(), size); } if (isCopy) { *isCopy = true; } return p; } jint* JNICALL GetIntArrayElements(Thread* t, jintArray array, jboolean* isCopy) { ENTER(t, Thread::ActiveState); unsigned size = (*array)->length() * sizeof(jint); jint* p = static_cast(t->m->heap->allocate(size)); if (size) { memcpy(p, (*array)->body().begin(), size); } if (isCopy) { *isCopy = true; } return p; } jlong* JNICALL GetLongArrayElements(Thread* t, jlongArray array, jboolean* isCopy) { ENTER(t, Thread::ActiveState); unsigned size = (*array)->length() * sizeof(jlong); jlong* p = static_cast(t->m->heap->allocate(size)); if (size) { memcpy(p, (*array)->body().begin(), size); } if (isCopy) { *isCopy = true; } return p; } jfloat* JNICALL GetFloatArrayElements(Thread* t, jfloatArray array, jboolean* isCopy) { ENTER(t, Thread::ActiveState); unsigned size = (*array)->length() * sizeof(jfloat); jfloat* p = static_cast(t->m->heap->allocate(size)); if (size) { memcpy(p, (*array)->body().begin(), size); } if (isCopy) { *isCopy = true; } return p; } jdouble* JNICALL GetDoubleArrayElements(Thread* t, jdoubleArray array, jboolean* isCopy) { ENTER(t, Thread::ActiveState); unsigned size = (*array)->length() * sizeof(jdouble); jdouble* p = static_cast(t->m->heap->allocate(size)); if (size) { memcpy(p, (*array)->body().begin(), size); } if (isCopy) { *isCopy = true; } return p; } void JNICALL ReleaseBooleanArrayElements(Thread* t, jbooleanArray array, jboolean* p, jint mode) { ENTER(t, Thread::ActiveState); unsigned size = (*array)->length() * sizeof(jboolean); if (mode == 0 or mode == AVIAN_JNI_COMMIT) { if (size) { memcpy((*array)->body().begin(), p, size); } } if (mode == 0 or mode == AVIAN_JNI_ABORT) { t->m->heap->free(p, size); } } void JNICALL ReleaseByteArrayElements(Thread* t, jbyteArray array, jbyte* p, jint mode) { ENTER(t, Thread::ActiveState); unsigned size = (*array)->length() * sizeof(jbyte); if (mode == 0 or mode == AVIAN_JNI_COMMIT) { if (size) { memcpy((*array)->body().begin(), p, size); } } if (mode == 0 or mode == AVIAN_JNI_ABORT) { t->m->heap->free(p, size); } } void JNICALL ReleaseCharArrayElements(Thread* t, jcharArray array, jchar* p, jint mode) { ENTER(t, Thread::ActiveState); unsigned size = (*array)->length() * sizeof(jchar); if (mode == 0 or mode == AVIAN_JNI_COMMIT) { if (size) { memcpy((*array)->body().begin(), p, size); } } if (mode == 0 or mode == AVIAN_JNI_ABORT) { t->m->heap->free(p, size); } } void JNICALL ReleaseShortArrayElements(Thread* t, jshortArray array, jshort* p, jint mode) { ENTER(t, Thread::ActiveState); unsigned size = (*array)->length() * sizeof(jshort); if (mode == 0 or mode == AVIAN_JNI_COMMIT) { if (size) { memcpy((*array)->body().begin(), p, size); } } if (mode == 0 or mode == AVIAN_JNI_ABORT) { t->m->heap->free(p, size); } } void JNICALL ReleaseIntArrayElements(Thread* t, jintArray array, jint* p, jint mode) { ENTER(t, Thread::ActiveState); unsigned size = (*array)->length() * sizeof(jint); if (mode == 0 or mode == AVIAN_JNI_COMMIT) { if (size) { memcpy((*array)->body().begin(), p, size); } } if (mode == 0 or mode == AVIAN_JNI_ABORT) { t->m->heap->free(p, size); } } void JNICALL ReleaseLongArrayElements(Thread* t, jlongArray array, jlong* p, jint mode) { ENTER(t, Thread::ActiveState); unsigned size = (*array)->length() * sizeof(jlong); if (mode == 0 or mode == AVIAN_JNI_COMMIT) { if (size) { memcpy((*array)->body().begin(), p, size); } } if (mode == 0 or mode == AVIAN_JNI_ABORT) { t->m->heap->free(p, size); } } void JNICALL ReleaseFloatArrayElements(Thread* t, jfloatArray array, jfloat* p, jint mode) { ENTER(t, Thread::ActiveState); unsigned size = (*array)->length() * sizeof(jfloat); if (mode == 0 or mode == AVIAN_JNI_COMMIT) { if (size) { memcpy((*array)->body().begin(), p, size); } } if (mode == 0 or mode == AVIAN_JNI_ABORT) { t->m->heap->free(p, size); } } void JNICALL ReleaseDoubleArrayElements(Thread* t, jdoubleArray array, jdouble* p, jint mode) { ENTER(t, Thread::ActiveState); unsigned size = (*array)->length() * sizeof(jdouble); if (mode == 0 or mode == AVIAN_JNI_COMMIT) { if (size) { memcpy((*array)->body().begin(), p, size); } } if (mode == 0 or mode == AVIAN_JNI_ABORT) { t->m->heap->free(p, size); } } void JNICALL GetBooleanArrayRegion(Thread* t, jbooleanArray array, jint offset, jint length, jboolean* dst) { ENTER(t, Thread::ActiveState); if (length) { memcpy(dst, &(*array)->body()[offset], length * sizeof(jboolean)); } } void JNICALL GetByteArrayRegion(Thread* t, jbyteArray array, jint offset, jint length, jbyte* dst) { ENTER(t, Thread::ActiveState); if (length) { memcpy(dst, &(*array)->body()[offset], length * sizeof(jbyte)); } } void JNICALL GetCharArrayRegion(Thread* t, jcharArray array, jint offset, jint length, jchar* dst) { ENTER(t, Thread::ActiveState); if (length) { memcpy(dst, &(*array)->body()[offset], length * sizeof(jchar)); } } void JNICALL GetShortArrayRegion(Thread* t, jshortArray array, jint offset, jint length, jshort* dst) { ENTER(t, Thread::ActiveState); if (length) { memcpy(dst, &(*array)->body()[offset], length * sizeof(jshort)); } } void JNICALL GetIntArrayRegion(Thread* t, jintArray array, jint offset, jint length, jint* dst) { ENTER(t, Thread::ActiveState); if (length) { memcpy(dst, &(*array)->body()[offset], length * sizeof(jint)); } } void JNICALL GetLongArrayRegion(Thread* t, jlongArray array, jint offset, jint length, jlong* dst) { ENTER(t, Thread::ActiveState); if (length) { memcpy(dst, &(*array)->body()[offset], length * sizeof(jlong)); } } void JNICALL GetFloatArrayRegion(Thread* t, jfloatArray array, jint offset, jint length, jfloat* dst) { ENTER(t, Thread::ActiveState); if (length) { memcpy(dst, &(*array)->body()[offset], length * sizeof(jfloat)); } } void JNICALL GetDoubleArrayRegion(Thread* t, jdoubleArray array, jint offset, jint length, jdouble* dst) { ENTER(t, Thread::ActiveState); if (length) { memcpy(dst, &(*array)->body()[offset], length * sizeof(jdouble)); } } void JNICALL SetBooleanArrayRegion(Thread* t, jbooleanArray array, jint offset, jint length, const jboolean* src) { ENTER(t, Thread::ActiveState); if (length) { memcpy(&(*array)->body()[offset], src, length * sizeof(jboolean)); } } void JNICALL SetByteArrayRegion(Thread* t, jbyteArray array, jint offset, jint length, const jbyte* src) { ENTER(t, Thread::ActiveState); if (length) { memcpy(&(*array)->body()[offset], src, length * sizeof(jbyte)); } } void JNICALL SetCharArrayRegion(Thread* t, jcharArray array, jint offset, jint length, const jchar* src) { ENTER(t, Thread::ActiveState); if (length) { memcpy(&(*array)->body()[offset], src, length * sizeof(jchar)); } } void JNICALL SetShortArrayRegion(Thread* t, jshortArray array, jint offset, jint length, const jshort* src) { ENTER(t, Thread::ActiveState); if (length) { memcpy(&(*array)->body()[offset], src, length * sizeof(jshort)); } } void JNICALL SetIntArrayRegion(Thread* t, jintArray array, jint offset, jint length, const jint* src) { ENTER(t, Thread::ActiveState); if (length) { memcpy(&(*array)->body()[offset], src, length * sizeof(jint)); } } void JNICALL SetLongArrayRegion(Thread* t, jlongArray array, jint offset, jint length, const jlong* src) { ENTER(t, Thread::ActiveState); if (length) { memcpy(&(*array)->body()[offset], src, length * sizeof(jlong)); } } void JNICALL SetFloatArrayRegion(Thread* t, jfloatArray array, jint offset, jint length, const jfloat* src) { ENTER(t, Thread::ActiveState); if (length) { memcpy(&(*array)->body()[offset], src, length * sizeof(jfloat)); } } void JNICALL SetDoubleArrayRegion(Thread* t, jdoubleArray array, jint offset, jint length, const jdouble* src) { ENTER(t, Thread::ActiveState); if (length) { memcpy(&(*array)->body()[offset], src, length * sizeof(jdouble)); } } void* JNICALL GetPrimitiveArrayCritical(Thread* t, jarray array, jboolean* isCopy) { if (t->criticalLevel == 0) { enter(t, Thread::ActiveState); } ++t->criticalLevel; if (isCopy) { *isCopy = true; } expect(t, *array); return reinterpret_cast(*array) + 2; } void JNICALL ReleasePrimitiveArrayCritical(Thread* t, jarray, void*, jint) { if ((--t->criticalLevel) == 0) { enter(t, Thread::IdleState); } } uint64_t fromReflectedMethod(Thread* t, uintptr_t* arguments) { jobject m = reinterpret_cast(arguments[0]); return methodID(t, t->m->classpath->getVMMethod(t, *m)); } jmethodID JNICALL FromReflectedMethod(Thread* t, jobject method) { uintptr_t arguments[] = {reinterpret_cast(method)}; return static_cast(run(t, fromReflectedMethod, arguments)); } uint64_t toReflectedMethod(Thread* t, uintptr_t* arguments) { jmethodID m = arguments[1]; jboolean isStatic = arguments[2]; return reinterpret_cast(makeLocalReference( t, t->m->classpath->makeJMethod( t, isStatic ? getStaticMethod(t, m) : getMethod(t, m)))); } jobject JNICALL ToReflectedMethod(Thread* t, jclass c, jmethodID method, jboolean isStatic) { uintptr_t arguments[] = {reinterpret_cast(c), static_cast(method), static_cast(isStatic)}; return reinterpret_cast(run(t, toReflectedMethod, arguments)); } uint64_t fromReflectedField(Thread* t, uintptr_t* arguments) { jobject f = reinterpret_cast(arguments[0]); return fieldID(t, t->m->classpath->getVMField(t, cast(t, *f))); } jfieldID JNICALL FromReflectedField(Thread* t, jobject field) { uintptr_t arguments[] = {reinterpret_cast(field)}; return static_cast(run(t, fromReflectedField, arguments)); } uint64_t toReflectedField(Thread* t, uintptr_t* arguments) { jfieldID f = arguments[1]; jboolean isStatic = arguments[2]; return reinterpret_cast(makeLocalReference( t, t->m->classpath->makeJField( t, isStatic ? getStaticField(t, f) : getField(t, f)))); } jobject JNICALL ToReflectedField(Thread* t, jclass c, jfieldID field, jboolean isStatic) { uintptr_t arguments[] = {reinterpret_cast(c), static_cast(field), static_cast(isStatic)}; return reinterpret_cast(run(t, toReflectedField, arguments)); } uint64_t registerNatives(Thread* t, uintptr_t* arguments) { jclass c = reinterpret_cast(arguments[0]); const JNINativeMethod* methods = reinterpret_cast(arguments[1]); jint methodCount = arguments[2]; for (int i = 0; i < methodCount; ++i) { if (methods[i].function) { // Android's class library sometimes prepends a mysterious "!" // to the method signature, which we happily ignore: const char* sig = methods[i].signature; if (*sig == '!') ++sig; GcMethod* method = findMethodOrNull(t, (*c)->vmClass(), methods[i].name, sig); if (method == 0 or (method->flags() & ACC_NATIVE) == 0) { // The JNI spec says we must throw a NoSuchMethodError in this // case, but that would prevent using a code shrinker like // ProGuard effectively. Instead, we just ignore it. if (false) { fprintf(stderr, "not found: %s.%s%s\n", (*c)->vmClass()->name()->body().begin(), methods[i].name, sig); abort(t); } } else { registerNative(t, method, methods[i].function); } } } return 1; } jint JNICALL RegisterNatives(Thread* t, jclass c, const JNINativeMethod* methods, jint methodCount) { uintptr_t arguments[] = {reinterpret_cast(c), reinterpret_cast(methods), static_cast(methodCount)}; return run(t, registerNatives, arguments) ? 0 : -1; } jint JNICALL UnregisterNatives(Thread* t, jclass c) { ENTER(t, Thread::ActiveState); unregisterNatives(t, (*c)->vmClass()); return 0; } uint64_t monitorOp(Thread* t, uintptr_t* arguments) { void (*op)(Thread*, object) = reinterpret_cast(arguments[0]); jobject o = reinterpret_cast(arguments[1]); op(t, *o); return 1; } void acquire0(Thread* t, object o) { return acquire(t, o); } jint JNICALL MonitorEnter(Thread* t, jobject o) { uintptr_t arguments[] = {reinterpret_cast(voidPointer(acquire0)), reinterpret_cast(o)}; return run(t, monitorOp, arguments) ? 0 : -1; } void release0(Thread* t, object o) { return release(t, o); } jint JNICALL MonitorExit(Thread* t, jobject o) { uintptr_t arguments[] = {reinterpret_cast(voidPointer(release0)), reinterpret_cast(o)}; return run(t, monitorOp, arguments) ? 0 : -1; } jint JNICALL GetJavaVM(Thread* t, Machine** m) { *m = t->m; return 0; } jboolean JNICALL IsSameObject(Thread* t, jobject a, jobject b) { if (a and b) { ENTER(t, Thread::ActiveState); return *a == *b; } else { return a == b; } } uint64_t pushLocalFrame(Thread* t, uintptr_t* arguments) { if (t->m->processor->pushLocalFrame(t, arguments[0])) { return 1; } else { throw_(t, roots(t)->outOfMemoryError()); } } jint JNICALL PushLocalFrame(Thread* t, jint capacity) { uintptr_t arguments[] = {static_cast(capacity)}; return run(t, pushLocalFrame, arguments) ? 0 : -1; } uint64_t popLocalFrame(Thread* t, uintptr_t* arguments) { uint64_t r; jobject presult = reinterpret_cast(arguments[0]); if (presult != NULL) { object result = *presult; PROTECT(t, result); t->m->processor->popLocalFrame(t); r = reinterpret_cast(makeLocalReference(t, result)); } else { t->m->processor->popLocalFrame(t); r = 0; } return r; } jobject JNICALL PopLocalFrame(Thread* t, jobject result) { uintptr_t arguments[] = {reinterpret_cast(result)}; return reinterpret_cast(run(t, popLocalFrame, arguments)); } uint64_t newDirectByteBuffer(Thread* t, uintptr_t* arguments) { jlong capacity; memcpy(&capacity, arguments + 1, sizeof(jlong)); return reinterpret_cast(makeLocalReference( t, t->m->classpath->makeDirectByteBuffer( t, reinterpret_cast(arguments[0]), capacity))); } jobject JNICALL NewDirectByteBuffer(Thread* t, void* p, jlong capacity) { uintptr_t arguments[1 + (sizeof(jlong) / BytesPerWord)]; arguments[0] = reinterpret_cast(p); memcpy(arguments + 1, &capacity, sizeof(jlong)); return reinterpret_cast(run(t, newDirectByteBuffer, arguments)); } uint64_t getDirectBufferAddress(Thread* t, uintptr_t* arguments) { return reinterpret_cast(t->m->classpath->getDirectBufferAddress( t, *reinterpret_cast(arguments[0]))); } void* JNICALL GetDirectBufferAddress(Thread* t, jobject b) { uintptr_t arguments[] = {reinterpret_cast(b)}; return reinterpret_cast(run(t, getDirectBufferAddress, arguments)); } uint64_t getDirectBufferCapacity(Thread* t, uintptr_t* arguments) { return t->m->classpath->getDirectBufferCapacity( t, *reinterpret_cast(arguments[0])); } jlong JNICALL GetDirectBufferCapacity(Thread* t, jobject b) { uintptr_t arguments[] = {reinterpret_cast(b)}; return run(t, getDirectBufferCapacity, arguments); } struct JavaVMOption { char* optionString; void* extraInfo; }; struct JavaVMInitArgs { jint version; jint nOptions; JavaVMOption* options; jboolean ignoreUnrecognized; }; int parseSize(const char* s) { unsigned length = strlen(s); RUNTIME_ARRAY(char, buffer, length + 1); if (length == 0) return 0; char suffix = s[length - 1]; if (suffix== 'k' or suffix == 'K') { memcpy(RUNTIME_ARRAY_BODY(buffer), s, length - 1); RUNTIME_ARRAY_BODY(buffer)[length - 1] = 0; return atoi(RUNTIME_ARRAY_BODY(buffer)) * 1024; } if (suffix == 'm' or suffix == 'M') { memcpy(RUNTIME_ARRAY_BODY(buffer), s, length - 1); RUNTIME_ARRAY_BODY(buffer)[length - 1] = 0; return atoi(RUNTIME_ARRAY_BODY(buffer)) * 1024 * 1024; } if (suffix == 'g' or suffix == 'G') { memcpy(RUNTIME_ARRAY_BODY(buffer), s, length - 1); RUNTIME_ARRAY_BODY(buffer)[length - 1] = 0; return atoi(RUNTIME_ARRAY_BODY(buffer)) * 1024 * 1024 * 1024; } return atoi(s); } void append(char** p, const char* value, unsigned length, char tail) { if (length) { memcpy(*p, value, length); *p += length; *((*p)++) = tail; } } uint64_t boot(Thread* t, uintptr_t*) { GcThrowable* throwable = makeThrowable(t, GcNullPointerException::Type); // sequence point, for gc (don't recombine statements) roots(t)->setNullPointerException(t, throwable); throwable = makeThrowable(t, GcArithmeticException::Type); // sequence point, for gc (don't recombine statements) roots(t)->setArithmeticException(t, throwable); throwable = makeThrowable(t, GcArrayIndexOutOfBoundsException::Type); // sequence point, for gc (don't recombine statements) roots(t)->setArrayIndexOutOfBoundsException(t, throwable); throwable = makeThrowable(t, GcOutOfMemoryError::Type); // sequence point, for gc (don't recombine statements) roots(t)->setOutOfMemoryError(t, throwable); throwable = makeThrowable(t, GcThrowable::Type); // sequence point, for gc (don't recombine statements) roots(t)->setShutdownInProgress(t, throwable); t->m->classpath->preBoot(t); t->javaThread = t->m->classpath->makeThread(t, 0); t->javaThread->peer() = reinterpret_cast(t); GcThread* jthread = t->m->classpath->makeThread(t, t); // sequence point, for gc (don't recombine statements) roots(t)->setFinalizerThread(t, jthread); roots(t)->finalizerThread()->daemon() = true; t->m->classpath->boot(t); const char* port = findProperty(t, "avian.trace.port"); if (port) { GcString* host = makeString(t, "0.0.0.0"); PROTECT(t, host); GcMethod* method = resolveMethod(t, roots(t)->bootLoader(), "avian/Traces", "startTraceListener", "(Ljava/lang/String;I)V"); t->m->processor->invoke(t, method, 0, host, atoi(port)); } enter(t, Thread::IdleState); return 1; } } // namespace local } // namespace namespace vm { void populateJNITables(JavaVMVTable* vmTable, JNIEnvVTable* envTable) { memset(vmTable, 0, sizeof(JavaVMVTable)); vmTable->DestroyJavaVM = local::DestroyJavaVM; vmTable->AttachCurrentThread = local::AttachCurrentThread; vmTable->AttachCurrentThreadAsDaemon = local::AttachCurrentThreadAsDaemon; vmTable->DetachCurrentThread = local::DetachCurrentThread; vmTable->GetEnv = local::GetEnv; memset(envTable, 0, sizeof(JNIEnvVTable)); envTable->GetVersion = local::GetVersion; envTable->GetStringLength = local::GetStringLength; envTable->GetStringChars = local::GetStringChars; envTable->ReleaseStringChars = local::ReleaseStringChars; envTable->GetStringRegion = local::GetStringRegion; envTable->GetStringCritical = local::GetStringCritical; envTable->ReleaseStringCritical = local::ReleaseStringCritical; envTable->GetStringUTFLength = local::GetStringUTFLength; envTable->GetStringUTFChars = local::GetStringUTFChars; envTable->ReleaseStringUTFChars = local::ReleaseStringUTFChars; envTable->GetStringUTFRegion = local::GetStringUTFRegion; envTable->GetArrayLength = local::GetArrayLength; envTable->NewString = local::NewString; envTable->NewStringUTF = local::NewStringUTF; envTable->DefineClass = local::DefineClass; envTable->FindClass = local::FindClass; envTable->ThrowNew = local::ThrowNew; envTable->Throw = local::Throw; envTable->ExceptionCheck = local::ExceptionCheck; envTable->NewDirectByteBuffer = local::NewDirectByteBuffer; envTable->GetDirectBufferAddress = local::GetDirectBufferAddress; envTable->GetDirectBufferCapacity = local::GetDirectBufferCapacity; envTable->NewLocalRef = local::NewLocalRef; envTable->DeleteLocalRef = local::DeleteLocalRef; envTable->GetObjectClass = local::GetObjectClass; envTable->GetSuperclass = local::GetSuperclass; envTable->IsInstanceOf = local::IsInstanceOf; envTable->IsAssignableFrom = local::IsAssignableFrom; envTable->GetFieldID = local::GetFieldID; envTable->GetMethodID = local::GetMethodID; envTable->GetStaticMethodID = local::GetStaticMethodID; envTable->NewObjectV = local::NewObjectV; envTable->NewObjectA = local::NewObjectA; envTable->NewObject = local::NewObject; envTable->CallObjectMethodV = local::CallObjectMethodV; envTable->CallObjectMethodA = local::CallObjectMethodA; envTable->CallObjectMethod = local::CallObjectMethod; envTable->CallBooleanMethodV = local::CallBooleanMethodV; envTable->CallBooleanMethodA = local::CallBooleanMethodA; envTable->CallBooleanMethod = local::CallBooleanMethod; envTable->CallByteMethodV = local::CallByteMethodV; envTable->CallByteMethodA = local::CallByteMethodA; envTable->CallByteMethod = local::CallByteMethod; envTable->CallCharMethodV = local::CallCharMethodV; envTable->CallCharMethodA = local::CallCharMethodA; envTable->CallCharMethod = local::CallCharMethod; envTable->CallShortMethodV = local::CallShortMethodV; envTable->CallShortMethodA = local::CallShortMethodA; envTable->CallShortMethod = local::CallShortMethod; envTable->CallIntMethodV = local::CallIntMethodV; envTable->CallIntMethodA = local::CallIntMethodA; envTable->CallIntMethod = local::CallIntMethod; envTable->CallLongMethodV = local::CallLongMethodV; envTable->CallLongMethodA = local::CallLongMethodA; envTable->CallLongMethod = local::CallLongMethod; envTable->CallFloatMethodV = local::CallFloatMethodV; envTable->CallFloatMethodA = local::CallFloatMethodA; envTable->CallFloatMethod = local::CallFloatMethod; envTable->CallDoubleMethodV = local::CallDoubleMethodV; envTable->CallDoubleMethodA = local::CallDoubleMethodA; envTable->CallDoubleMethod = local::CallDoubleMethod; envTable->CallVoidMethodV = local::CallVoidMethodV; envTable->CallVoidMethodA = local::CallVoidMethodA; envTable->CallVoidMethod = local::CallVoidMethod; envTable->CallStaticObjectMethodV = local::CallStaticObjectMethodV; envTable->CallStaticObjectMethodA = local::CallStaticObjectMethodA; envTable->CallStaticObjectMethod = local::CallStaticObjectMethod; envTable->CallStaticBooleanMethodV = local::CallStaticBooleanMethodV; envTable->CallStaticBooleanMethodA = local::CallStaticBooleanMethodA; envTable->CallStaticBooleanMethod = local::CallStaticBooleanMethod; envTable->CallStaticByteMethodV = local::CallStaticByteMethodV; envTable->CallStaticByteMethodA = local::CallStaticByteMethodA; envTable->CallStaticByteMethod = local::CallStaticByteMethod; envTable->CallStaticCharMethodV = local::CallStaticCharMethodV; envTable->CallStaticCharMethodA = local::CallStaticCharMethodA; envTable->CallStaticCharMethod = local::CallStaticCharMethod; envTable->CallStaticShortMethodV = local::CallStaticShortMethodV; envTable->CallStaticShortMethodA = local::CallStaticShortMethodA; envTable->CallStaticShortMethod = local::CallStaticShortMethod; envTable->CallStaticIntMethodV = local::CallStaticIntMethodV; envTable->CallStaticIntMethodA = local::CallStaticIntMethodA; envTable->CallStaticIntMethod = local::CallStaticIntMethod; envTable->CallStaticLongMethodV = local::CallStaticLongMethodV; envTable->CallStaticLongMethodA = local::CallStaticLongMethodA; envTable->CallStaticLongMethod = local::CallStaticLongMethod; envTable->CallStaticFloatMethodV = local::CallStaticFloatMethodV; envTable->CallStaticFloatMethodA = local::CallStaticFloatMethodA; envTable->CallStaticFloatMethod = local::CallStaticFloatMethod; envTable->CallStaticDoubleMethodV = local::CallStaticDoubleMethodV; envTable->CallStaticDoubleMethodA = local::CallStaticDoubleMethodA; envTable->CallStaticDoubleMethod = local::CallStaticDoubleMethod; envTable->CallStaticVoidMethodV = local::CallStaticVoidMethodV; envTable->CallStaticVoidMethodA = local::CallStaticVoidMethodA; envTable->CallStaticVoidMethod = local::CallStaticVoidMethod; envTable->GetStaticFieldID = local::GetStaticFieldID; envTable->GetObjectField = local::GetObjectField; envTable->GetBooleanField = local::GetBooleanField; envTable->GetByteField = local::GetByteField; envTable->GetCharField = local::GetCharField; envTable->GetShortField = local::GetShortField; envTable->GetIntField = local::GetIntField; envTable->GetLongField = local::GetLongField; envTable->GetFloatField = local::GetFloatField; envTable->GetDoubleField = local::GetDoubleField; envTable->SetObjectField = local::SetObjectField; envTable->SetBooleanField = local::SetBooleanField; envTable->SetByteField = local::SetByteField; envTable->SetCharField = local::SetCharField; envTable->SetShortField = local::SetShortField; envTable->SetIntField = local::SetIntField; envTable->SetLongField = local::SetLongField; envTable->SetFloatField = local::SetFloatField; envTable->SetDoubleField = local::SetDoubleField; envTable->GetStaticObjectField = local::GetStaticObjectField; envTable->GetStaticBooleanField = local::GetStaticBooleanField; envTable->GetStaticByteField = local::GetStaticByteField; envTable->GetStaticCharField = local::GetStaticCharField; envTable->GetStaticShortField = local::GetStaticShortField; envTable->GetStaticIntField = local::GetStaticIntField; envTable->GetStaticLongField = local::GetStaticLongField; envTable->GetStaticFloatField = local::GetStaticFloatField; envTable->GetStaticDoubleField = local::GetStaticDoubleField; envTable->SetStaticObjectField = local::SetStaticObjectField; envTable->SetStaticBooleanField = local::SetStaticBooleanField; envTable->SetStaticByteField = local::SetStaticByteField; envTable->SetStaticCharField = local::SetStaticCharField; envTable->SetStaticShortField = local::SetStaticShortField; envTable->SetStaticIntField = local::SetStaticIntField; envTable->SetStaticLongField = local::SetStaticLongField; envTable->SetStaticFloatField = local::SetStaticFloatField; envTable->SetStaticDoubleField = local::SetStaticDoubleField; envTable->NewGlobalRef = local::NewGlobalRef; envTable->NewWeakGlobalRef = local::NewWeakGlobalRef; envTable->DeleteGlobalRef = local::DeleteGlobalRef; envTable->DeleteWeakGlobalRef = local::DeleteWeakGlobalRef; envTable->EnsureLocalCapacity = local::EnsureLocalCapacity; envTable->ExceptionOccurred = local::ExceptionOccurred; envTable->ExceptionDescribe = local::ExceptionDescribe; envTable->ExceptionClear = local::ExceptionClear; envTable->NewObjectArray = local::NewObjectArray; envTable->GetObjectArrayElement = local::GetObjectArrayElement; envTable->SetObjectArrayElement = local::SetObjectArrayElement; envTable->NewBooleanArray = local::NewBooleanArray; envTable->NewByteArray = local::NewByteArray; envTable->NewCharArray = local::NewCharArray; envTable->NewShortArray = local::NewShortArray; envTable->NewIntArray = local::NewIntArray; envTable->NewLongArray = local::NewLongArray; envTable->NewFloatArray = local::NewFloatArray; envTable->NewDoubleArray = local::NewDoubleArray; envTable->GetBooleanArrayElements = local::GetBooleanArrayElements; envTable->GetByteArrayElements = local::GetByteArrayElements; envTable->GetCharArrayElements = local::GetCharArrayElements; envTable->GetShortArrayElements = local::GetShortArrayElements; envTable->GetIntArrayElements = local::GetIntArrayElements; envTable->GetLongArrayElements = local::GetLongArrayElements; envTable->GetFloatArrayElements = local::GetFloatArrayElements; envTable->GetDoubleArrayElements = local::GetDoubleArrayElements; envTable->ReleaseBooleanArrayElements = local::ReleaseBooleanArrayElements; envTable->ReleaseByteArrayElements = local::ReleaseByteArrayElements; envTable->ReleaseCharArrayElements = local::ReleaseCharArrayElements; envTable->ReleaseShortArrayElements = local::ReleaseShortArrayElements; envTable->ReleaseIntArrayElements = local::ReleaseIntArrayElements; envTable->ReleaseLongArrayElements = local::ReleaseLongArrayElements; envTable->ReleaseFloatArrayElements = local::ReleaseFloatArrayElements; envTable->ReleaseDoubleArrayElements = local::ReleaseDoubleArrayElements; envTable->GetBooleanArrayRegion = local::GetBooleanArrayRegion; envTable->GetByteArrayRegion = local::GetByteArrayRegion; envTable->GetCharArrayRegion = local::GetCharArrayRegion; envTable->GetShortArrayRegion = local::GetShortArrayRegion; envTable->GetIntArrayRegion = local::GetIntArrayRegion; envTable->GetLongArrayRegion = local::GetLongArrayRegion; envTable->GetFloatArrayRegion = local::GetFloatArrayRegion; envTable->GetDoubleArrayRegion = local::GetDoubleArrayRegion; envTable->SetBooleanArrayRegion = local::SetBooleanArrayRegion; envTable->SetByteArrayRegion = local::SetByteArrayRegion; envTable->SetCharArrayRegion = local::SetCharArrayRegion; envTable->SetShortArrayRegion = local::SetShortArrayRegion; envTable->SetIntArrayRegion = local::SetIntArrayRegion; envTable->SetLongArrayRegion = local::SetLongArrayRegion; envTable->SetFloatArrayRegion = local::SetFloatArrayRegion; envTable->SetDoubleArrayRegion = local::SetDoubleArrayRegion; envTable->GetPrimitiveArrayCritical = local::GetPrimitiveArrayCritical; envTable->ReleasePrimitiveArrayCritical = local::ReleasePrimitiveArrayCritical; envTable->RegisterNatives = local::RegisterNatives; envTable->UnregisterNatives = local::UnregisterNatives; envTable->MonitorEnter = local::MonitorEnter; envTable->MonitorExit = local::MonitorExit; envTable->GetJavaVM = local::GetJavaVM; envTable->IsSameObject = local::IsSameObject; envTable->PushLocalFrame = local::PushLocalFrame; envTable->PopLocalFrame = local::PopLocalFrame; envTable->FromReflectedMethod = local::FromReflectedMethod; envTable->ToReflectedMethod = local::ToReflectedMethod; envTable->FromReflectedField = local::FromReflectedField; envTable->ToReflectedField = local::ToReflectedField; } } // namespace vm extern "C" AVIAN_EXPORT jint JNICALL JNI_GetDefaultJavaVMInitArgs(void*) { return 0; } extern "C" AVIAN_EXPORT jint JNICALL JNI_GetCreatedJavaVMs(Machine**, jsize, jsize*) { // todo return -1; } extern "C" AVIAN_EXPORT jint JNICALL JNI_CreateJavaVM(Machine** m, Thread** t, void* args) { local::JavaVMInitArgs* a = static_cast(args); unsigned heapLimit = 0; unsigned stackLimit = 0; const char* bootLibraries = 0; const char* classpath = 0; const char* javaHome = AVIAN_JAVA_HOME; const char* embedPrefix = AVIAN_EMBED_PREFIX; const char* bootClasspathPrepend = ""; const char* bootClasspath = 0; const char* bootClasspathAppend = ""; const char* crashDumpDirectory = 0; unsigned propertyCount = 0; for (int i = 0; i < a->nOptions; ++i) { if (strncmp(a->options[i].optionString, "-X", 2) == 0) { const char* p = a->options[i].optionString + 2; if (strncmp(p, "mx", 2) == 0) { heapLimit = local::parseSize(p + 2); } else if (strncmp(p, "ss", 2) == 0) { stackLimit = local::parseSize(p + 2); } else if (strncmp(p, BOOTCLASSPATH_PREPEND_OPTION ":", sizeof(BOOTCLASSPATH_PREPEND_OPTION)) == 0) { bootClasspathPrepend = p + sizeof(BOOTCLASSPATH_PREPEND_OPTION); } else if (strncmp( p, BOOTCLASSPATH_OPTION ":", sizeof(BOOTCLASSPATH_OPTION)) == 0) { bootClasspath = p + sizeof(BOOTCLASSPATH_OPTION); } else if (strncmp(p, BOOTCLASSPATH_APPEND_OPTION ":", sizeof(BOOTCLASSPATH_APPEND_OPTION)) == 0) { bootClasspathAppend = p + sizeof(BOOTCLASSPATH_APPEND_OPTION); } } else if (strncmp(a->options[i].optionString, "-D", 2) == 0) { const char* p = a->options[i].optionString + 2; if (strncmp(p, BOOTSTRAP_PROPERTY "=", sizeof(BOOTSTRAP_PROPERTY)) == 0) { bootLibraries = p + sizeof(BOOTSTRAP_PROPERTY); } else if (strncmp(p, JAVA_COMMAND_PROPERTY "=", sizeof(JAVA_COMMAND_PROPERTY)) == 0 or strncmp(p, JAVA_LAUNCHER_PROPERTY "=", sizeof(JAVA_LAUNCHER_PROPERTY)) == 0) { // this means we're being invoked via the javac or java // command, so the bootstrap library should be e.g. libjvm.so bootLibraries = SO_PREFIX "jvm" SO_SUFFIX; } else if (strncmp(p, CRASHDIR_PROPERTY "=", sizeof(CRASHDIR_PROPERTY)) == 0) { crashDumpDirectory = p + sizeof(CRASHDIR_PROPERTY); } else if (strncmp(p, CLASSPATH_PROPERTY "=", sizeof(CLASSPATH_PROPERTY)) == 0) { classpath = p + sizeof(CLASSPATH_PROPERTY); } else if (strncmp(p, JAVA_HOME_PROPERTY "=", sizeof(JAVA_HOME_PROPERTY)) == 0) { javaHome = p + sizeof(JAVA_HOME_PROPERTY); } else if (strncmp(p, EMBED_PREFIX_PROPERTY "=", sizeof(EMBED_PREFIX_PROPERTY)) == 0) { embedPrefix = p + sizeof(EMBED_PREFIX_PROPERTY); } ++propertyCount; } } if (heapLimit == 0) heapLimit = 128 * 1024 * 1024; if (stackLimit == 0) stackLimit = 128 * 1024; bool addClasspathProperty = classpath == 0; if (addClasspathProperty) { classpath = "."; ++propertyCount; } System* s = makeSystem(); Heap* h = makeHeap(s, heapLimit); Classpath* c = makeClasspath(s, h, javaHome, embedPrefix); if (bootClasspath == 0) { bootClasspath = c->bootClasspath(); } unsigned bcppl = strlen(bootClasspathPrepend); unsigned bcpl = strlen(bootClasspath); unsigned bcpal = strlen(bootClasspathAppend); unsigned bootClasspathBufferSize = bcppl + bcpl + bcpal + 3; RUNTIME_ARRAY(char, bootClasspathBuffer, bootClasspathBufferSize); char* bootClasspathPointer = RUNTIME_ARRAY_BODY(bootClasspathBuffer); if (bootClasspathBufferSize > 3) { local::append(&bootClasspathPointer, bootClasspathPrepend, bcppl, bcpl + bcpal ? PATH_SEPARATOR : 0); local::append( &bootClasspathPointer, bootClasspath, bcpl, bcpal ? PATH_SEPARATOR : 0); local::append(&bootClasspathPointer, bootClasspathAppend, bcpal, 0); } else { *RUNTIME_ARRAY_BODY(bootClasspathBuffer) = 0; } char* bootLibrary = bootLibraries ? strdup(bootLibraries) : 0; char* bootLibraryEnd = bootLibrary ? strchr(bootLibrary, PATH_SEPARATOR) : 0; if (bootLibraryEnd) *bootLibraryEnd = 0; Finder* bf = makeFinder(s, h, RUNTIME_ARRAY_BODY(bootClasspathBuffer), bootLibrary); Finder* af = makeFinder(s, h, classpath, bootLibrary); if (bootLibrary) free(bootLibrary); Processor* p = makeProcessor(s, h, crashDumpDirectory, true); // reserve space for avian.version and file.encoding: propertyCount += 2; const char** properties = static_cast( h->allocate(sizeof(const char*) * propertyCount)); const char** propertyPointer = properties; const char** arguments = static_cast( h->allocate(sizeof(const char*) * a->nOptions)); const char** argumentPointer = arguments; for (int i = 0; i < a->nOptions; ++i) { if (strncmp(a->options[i].optionString, "-D", 2) == 0) { *(propertyPointer++) = a->options[i].optionString + 2; } *(argumentPointer++) = a->options[i].optionString; } unsigned cpl = strlen(classpath); RUNTIME_ARRAY(char, classpathProperty, cpl + strlen(CLASSPATH_PROPERTY) + 2); if (addClasspathProperty) { char* p = RUNTIME_ARRAY_BODY(classpathProperty); local::append(&p, CLASSPATH_PROPERTY, strlen(CLASSPATH_PROPERTY), '='); local::append(&p, classpath, cpl, 0); *(propertyPointer++) = RUNTIME_ARRAY_BODY(classpathProperty); } *(propertyPointer++) = "avian.version=" AVIAN_VERSION; // todo: should this be derived from the OS locale? Should it be // overrideable via JavaVMInitArgs? *(propertyPointer++) = "file.encoding=UTF-8"; *m = new (h->allocate(sizeof(Machine))) Machine(s, h, bf, af, p, c, properties, propertyCount, arguments, a->nOptions, stackLimit); h->free(properties, sizeof(const char*) * propertyCount); *t = p->makeThread(*m, 0, 0); enter(*t, Thread::ActiveState); enter(*t, Thread::IdleState); return run(*t, local::boot, 0) ? 0 : -1; }