#include "builtin.h" #include "machine.h" #include "constants.h" #include "run.h" using namespace vm; namespace { object doInvoke(Thread* t, object this_, object instance, object arguments) { object v = pushReference(t, run2(t, this_, instance, arguments)); if (t->exception) { t->exception = makeInvocationTargetException(t, t->exception); } return v; } inline void replace(char a, char b, char* c) { for (; *c; ++c) if (*c == a) *c = b; } jstring Object_toString(Thread* t, jobject this_) { object s = makeString (t, "%s@%p", &byteArrayBody(t, className(t, objectClass(t, *this_)), 0), *this_); return pushReference(t, s); } jclass Object_getClass(Thread* t, jobject this_) { return pushReference(t, objectClass(t, *this_)); } void Object_wait(Thread* t, jobject this_, jlong milliseconds) { vm::wait(t, *this_, milliseconds); } void Object_notify(Thread* t, jobject this_) { vm::notify(t, *this_); } void Object_notifyAll(Thread* t, jobject this_) { vm::notifyAll(t, *this_); } jclass Class_forName(Thread* t, jclass, jstring name) { if (LIKELY(name)) { object n = makeByteArray(t, stringLength(t, *name) + 1, false); char* s = reinterpret_cast(&byteArrayBody(t, n, 0)); stringChars(t, *name, s); replace('.', '/', s); object c = resolveClass(t, n); if (t->exception) { return 0; } object clinit = classInitializer(t, c); if (clinit) { PROTECT(t, c); set(t, classInitializer(t, c), 0); run(t, clinit, 0); } return pushReference(t, c); } else { t->exception = makeNullPointerException(t); return 0; } } jboolean Class_isAssignableFrom(Thread* t, jobject this_, jclass that) { if (LIKELY(that)) { return vm::isAssignableFrom(t, *this_, *that); } else { t->exception = makeNullPointerException(t); return 0; } } jobject Field_get(Thread* t, jobject this_, jobject instancep) { object field = *this_; if (fieldFlags(t, field) & ACC_STATIC) { return pushReference (t, arrayBody(t, classStaticTable(t, fieldClass(t, field)), fieldOffset(t, field))); } else if (instancep) { object instance = *instancep; if (instanceOf(t, fieldClass(t, this_), instance)) { switch (fieldCode(t, field)) { case ByteField: return pushReference (t, makeByte(t, cast(instance, fieldOffset(t, field)))); case BooleanField: return pushReference (t, makeBoolean(t, cast(instance, fieldOffset(t, field)))); case CharField: return pushReference (t, makeChar(t, cast(instance, fieldOffset(t, field)))); case ShortField: return pushReference (t, makeShort(t, cast(instance, fieldOffset(t, field)))); case FloatField: return pushReference (t, makeFloat(t, cast(instance, fieldOffset(t, field)))); case IntField: return pushReference (t, makeInt(t, cast(instance, fieldOffset(t, field)))); case DoubleField: return pushReference (t, makeDouble(t, cast(instance, fieldOffset(t, field)))); case LongField: return pushReference (t, makeLong(t, cast(instance, fieldOffset(t, field)))); case ObjectField: return pushReference (t, cast(instance, fieldOffset(t, field))); default: abort(t); } } else { t->exception = makeIllegalArgumentException(t); } } else { t->exception = makeNullPointerException(t); } return 0; } void Field_set(Thread* t, jobject this_, jobject instancep, jobject value) { object field = *this_; object v = (value ? *value : 0); if (fieldFlags(t, field) & ACC_STATIC) { if (fieldCode(t, field) == ObjectField or v) { set(t, arrayBody(t, classStaticTable(t, fieldClass(t, field)), fieldOffset(t, field)), v); } else { t->exception = makeNullPointerException(t); } } else if (instancep) { object instance = *instancep; if (instanceOf(t, fieldClass(t, this_), instance)) { switch (fieldCode(t, field)) { case ObjectField: set(t, cast(instance, fieldOffset(t, field)), v); break; default: { uint8_t* body = &cast(instance, fieldOffset(t, field)); if (v) { memcpy(body, &cast(v, BytesPerWord), primitiveSize(t, fieldCode(t, field))); } else { t->exception = makeNullPointerException(t); } } break; } } else { t->exception = makeIllegalArgumentException(t); } } else { t->exception = makeNullPointerException(t); } } jobject Constructor_make(Thread* t, jclass, jclass c) { return pushReference(t, make(t, c)); } jobject Method_invoke(Thread* t, jobject this_, jobject instancep, jobjectArray argumentsp) { object method = *this_; if (argumentsp) { object arguments = *argumentsp; if (methodFlags(t, method) & ACC_STATIC) { if (objectArrayLength(t, arguments) == methodParameterCount(t, method)) { return pushReference(t, doInvoke(t, method, 0, arguments)); } else { t->exception = makeArrayIndexOutOfBoundsException(t, 0); } } else if (instancep) { object instance = *instancep; if (instanceOf(t, methodClass(t, method), instance)) { if (objectArrayLength(t, arguments) == static_cast(methodParameterCount(t, method) - 1)) { return pushReference(t, doInvoke(t, method, instance, arguments)); } else { t->exception = makeArrayIndexOutOfBoundsException(t, 0); } } } else { t->exception = makeNullPointerException(t); } } else { t->exception = makeNullPointerException(t); } return 0; } jobject Array_get(Thread* t, jobject array, int index) { if (LIKELY(array)) { object a = *array; unsigned elementSize = classArrayElementSize(t, objectClass(t, a)); if (LIKELY(elementSize)) { intptr_t length = cast(a, BytesPerWord); if (LIKELY(index >= 0 and index < length)) { switch (byteArrayBody(t, className(t, objectClass(t, a)), 1)) { case 'B': return pushReference(t, makeByte(t, byteArrayBody(t, a, index))); case 'C': return pushReference(t, makeChar(t, charArrayBody(t, a, index))); case 'D': return pushReference(t, makeDouble(t, doubleArrayBody(t, a, index))); case 'F': return pushReference(t, makeFloat(t, floatArrayBody(t, a, index))); case 'I': return pushReference(t, makeInt(t, intArrayBody(t, a, index))); case 'J': return pushReference(t, makeLong(t, longArrayBody(t, a, index))); case 'S': return pushReference(t, makeShort(t, shortArrayBody(t, a, index))); case 'Z': return pushReference (t, makeBoolean(t, booleanArrayBody(t, a, index))); case 'L': case '[': return pushReference(t, objectArrayBody(t, a, index)); default: abort(t); } } else { t->exception = makeArrayIndexOutOfBoundsException(t, 0); } } else { t->exception = makeIllegalArgumentException(t); } } else { t->exception = makeNullPointerException(t); } return 0; } void Array_set(Thread* t, jobject array, int index, jobject value) { if (LIKELY(array)) { object a = *array; object v = (value ? *value : 0); unsigned elementSize = classArrayElementSize(t, objectClass(t, a)); if (LIKELY(elementSize)) { intptr_t length = cast(a, BytesPerWord); if (LIKELY(index >= 0 and index < length)) { switch (byteArrayBody(t, className(t, objectClass(t, a)), 1)) { case 'L': case '[': set(t, objectArrayBody(t, a, index), v); break; default: { uint8_t* p = &cast (a, (2 * BytesPerWord) + (index * elementSize)); if (v) { memcpy(p, &cast(v, BytesPerWord), elementSize); } else { t->exception = makeNullPointerException(t); } } break; } } else { t->exception = makeArrayIndexOutOfBoundsException(t, 0); } } else { t->exception = makeIllegalArgumentException(t); } } else { t->exception = makeNullPointerException(t); } } jint Array_getLength(Thread* t, jobject array) { if (LIKELY(array)) { object a = *array; unsigned elementSize = classArrayElementSize(t, objectClass(t, a)); if (LIKELY(elementSize)) { return cast(a, BytesPerWord); } else { t->exception = makeIllegalArgumentException(t); } } else { t->exception = makeNullPointerException(t); } return 0; } jobject Array_makeObjectArray(Thread* t, jclass, jclass elementType, jint length) { return pushReference(t, makeObjectArray(t, *elementType, length, true)); } void System_arraycopy(Thread* t, jclass, jobject src, jint srcOffset, jobject dst, jint dstOffset, jint length) { if (LIKELY(src and dst)) { object s = *src; object d = *dst; if (LIKELY(objectClass(t, s) == objectClass(t, d))) { unsigned elementSize = classArrayElementSize(t, objectClass(t, s)); if (LIKELY(elementSize)) { intptr_t sl = cast(s, BytesPerWord); intptr_t dl = cast(d, BytesPerWord); if (LIKELY(srcOffset >= 0 and srcOffset + length <= sl and dstOffset >= 0 and dstOffset + length <= dl)) { uint8_t* sbody = &cast(s, 2 * BytesPerWord); uint8_t* dbody = &cast(d, 2 * BytesPerWord); if (src == dst) { memmove(dbody + (dstOffset * elementSize), sbody + (srcOffset * elementSize), length * elementSize); } else { memcpy(dbody + (dstOffset * elementSize), sbody + (srcOffset * elementSize), length * elementSize); } return; } } } } else { t->exception = makeNullPointerException(t); return; } t->exception = makeArrayStoreException(t); } jlong System_currentTimeMillis(Thread* t, jclass) { return t->vm->system->now(); } void Runtime_loadLibrary(Thread* t, jobject, jstring name) { if (LIKELY(name)) { char n[stringLength(t, *name) + 1]; stringChars(t, *name, n); for (System::Library* lib = t->vm->libraries; lib; lib = lib->next()) { if (::strcmp(lib->name(), n) == 0) { // already loaded return; } } System::Library* lib; if (LIKELY(t->vm->system->success (t->vm->system->load(&lib, n, t->vm->libraries)))) { t->vm->libraries = lib; } else { object message = makeString(t, "library not found: %s", n); t->exception = makeRuntimeException(t, message); } } else { t->exception = makeNullPointerException(t); } } void Runtime_gc(Thread* t, jobject) { ENTER(t, Thread::ExclusiveState); collect(t, Heap::MajorCollection); } void Runtime_exit(Thread* t, jobject, jint code) { t->vm->system->exit(code); } jlong Runtime_freeMemory(Thread*, jobject) { // todo return 0; } jobject Throwable_trace(Thread* t, jclass, jint skipCount) { int frame = t->frame; while (skipCount-- and frame >= 0) { frame = frameNext(t, frame); } if (methodClass(t, frameMethod(t, frame)) == arrayBody(t, t->vm->types, Machine::ThrowableType)) { // skip Throwable constructors while (strcmp(reinterpret_cast(""), &byteArrayBody(t, methodName(t, frameMethod(t, frame)), 0)) == 0) { frame = frameNext(t, frame); } } return pushReference(t, makeTrace(t, frame)); } jarray Throwable_resolveTrace(Thread* t, jclass, jobject trace) { unsigned length = arrayLength(t, *trace); object array = makeObjectArray (t, arrayBody(t, t->vm->types, Machine::StackTraceElementType), length, true); PROTECT(t, array); object e = 0; PROTECT(t, e); object class_ = 0; PROTECT(t, class_); for (unsigned i = 0; i < length; ++i) { e = arrayBody(t, *trace, i); class_ = className(t, methodClass(t, traceElementMethod(t, e))); class_ = makeString(t, class_, 0, byteArrayLength(t, class_) - 1, 0); object method = methodName(t, traceElementMethod(t, e)); method = makeString(t, method, 0, byteArrayLength(t, method) - 1, 0); unsigned line = lineNumber (t, traceElementMethod(t, e), traceElementIp(t, e)); object ste = makeStackTraceElement(t, class_, method, 0, line); set(t, objectArrayBody(t, array, i), ste); } return pushReference(t, array); } jobject Thread_currentThread(Thread* t, jclass) { return pushReference(t, t->javaThread); } void Thread_sleep(Thread* t, jclass, jlong milliseconds) { if (milliseconds == 0) milliseconds = INT64_MAX; ENTER(t, Thread::IdleState); t->vm->system->sleep(milliseconds); } void Thread_start(Thread* t, jobject this_) { Thread* p = reinterpret_cast(threadPeer(t, *this_)); if (p) { object message = makeString(t, "thread already started"); t->exception = makeIllegalStateException(t, message); } else { p = new (t->vm->system->allocate(sizeof(Thread))) Thread(t->vm, *this_, t); enter(p, Thread::ActiveState); class Runnable: public System::Runnable { public: Runnable(System* s, Thread* t): s(s), t(t) { } virtual void run(System::Thread* st) { t->systemThread = st; vm::run(t, "java/lang/Thread", "run", "()V", t->javaThread); if (t->exception) { printTrace(t, t->exception); } t->exit(); } virtual void dispose() { s->free(this); } System* s; Thread* t; }* r = new (t->vm->system->allocate(sizeof(Runnable))) Runnable(t->vm->system, p); if (not t->vm->system->success(t->vm->system->start(r))) { p->exit(); object message = makeString(t, "unable to start native thread"); t->exception = makeRuntimeException(t, message); } } } } // namespace namespace vm { void populateBuiltinMap(Thread* t, object map) { struct { const char* key; void* value; } builtins[] = { { "Java_java_lang_Class_forName", reinterpret_cast(::Class_forName) }, { "Java_java_lang_Class_isAssignableFrom", reinterpret_cast(::Class_isAssignableFrom) }, { "Java_java_lang_System_arraycopy", reinterpret_cast(::System_arraycopy) }, { "Java_java_lang_System_currentTimeMillis", reinterpret_cast(::System_currentTimeMillis) }, { "Java_java_lang_Runtime_loadLibrary", reinterpret_cast(::Runtime_loadLibrary) }, { "Java_java_lang_Runtime_gc", reinterpret_cast(::Runtime_gc) }, { "Java_java_lang_Runtiime_exit", reinterpret_cast(::Runtime_exit) }, { "Java_java_lang_Thread_doStart", reinterpret_cast(::Thread_start) }, { "Java_java_lang_Thread_currentThread", reinterpret_cast(::Thread_currentThread) }, { "Java_java_lang_Thread_sleep", reinterpret_cast(::Thread_sleep) }, { "Java_java_lang_Throwable_resolveTrace", reinterpret_cast(::Throwable_resolveTrace) }, { "Java_java_lang_Throwable_trace", reinterpret_cast(::Throwable_trace) }, { "Java_java_lang_Object_getClass", reinterpret_cast(::Object_getClass) }, { "Java_java_lang_Object_notify", reinterpret_cast(::Object_notify) }, { "Java_java_lang_Object_notifyAll", reinterpret_cast(::Object_notifyAll) }, { "Java_java_lang_Object_toString", reinterpret_cast(::Object_toString) }, { "Java_java_lang_Object_wait", reinterpret_cast(::Object_wait) }, { "Java_java_lang_reflect_Array_get", reinterpret_cast(::Array_get) }, { "Java_java_lang_reflect_Array_set", reinterpret_cast(::Array_set) }, { "Java_java_lang_reflect_Array_getLength", reinterpret_cast(::Array_getLength) }, { "Java_java_lang_reflect_Array_makeObjectArray", reinterpret_cast(::Array_makeObjectArray) }, { "Java_java_lang_reflect_Constructor_make", reinterpret_cast(::Constructor_make) }, { "Java_java_lang_reflect_Field_get", reinterpret_cast(::Field_get) }, { "Java_java_lang_reflect_Field_set", reinterpret_cast(::Field_set) }, { "Java_java_lang_reflect_Method_invoke", reinterpret_cast(::Method_invoke) }, { 0, 0 } }; for (unsigned i = 0; builtins[i].key; ++i) { object key = makeByteArray(t, builtins[i].key); PROTECT(t, key); object value = makePointer(t, builtins[i].value); hashMapInsert(t, map, key, value, byteArrayHash); } } } // namespace vm