#include "common.h" #include "system.h" #include "heap.h" #include "finder.h" #include "constants.h" #include "run.h" #include "jnienv.h" #include "builtin.h" #include "machine.h" using namespace vm; namespace { void pushFrame(Thread* t, object method) { if (t->frame >= 0) { pokeInt(t, t->frame + FrameIpOffset, t->ip); } t->ip = 0; unsigned parameterFootprint = methodParameterFootprint(t, method); unsigned base = t->sp - parameterFootprint; unsigned locals = parameterFootprint; if ((methodFlags(t, method) & ACC_NATIVE) == 0) { t->code = methodCode(t, method); locals = codeMaxLocals(t, t->code); memset(t->stack + ((base + parameterFootprint) * 2), 0, (locals - parameterFootprint) * BytesPerWord * 2); } unsigned frame = base + locals; pokeInt(t, frame + FrameNextOffset, t->frame); t->frame = frame; t->sp = frame + FrameFootprint; pokeInt(t, frame + FrameBaseOffset, base); pokeObject(t, frame + FrameMethodOffset, method); pokeInt(t, t->frame + FrameIpOffset, 0); if (methodFlags(t, method) & ACC_SYNCHRONIZED) { if (methodFlags(t, method) & ACC_STATIC) { acquire(t, methodClass(t, method)); } else { acquire(t, peekObject(t, base)); } } } void popFrame(Thread* t) { object method = frameMethod(t, t->frame); if (methodFlags(t, method) & ACC_SYNCHRONIZED) { if (methodFlags(t, method) & ACC_STATIC) { release(t, methodClass(t, method)); } else { release(t, peekObject(t, frameBase(t, t->frame))); } } if (UNLIKELY(methodVmFlags(t, method) & ClassInitFlag)) { if (t->exception) { t->exception = makeExceptionInInitializerError(t, t->exception); } classVmFlags(t, methodClass(t, method)) &= ~(NeedInitFlag | InitFlag); release(t, t->vm->classLock); } t->sp = frameBase(t, t->frame); t->frame = frameNext(t, t->frame); if (t->frame >= 0) { t->code = methodCode(t, frameMethod(t, t->frame)); t->ip = frameIp(t, t->frame); } else { t->code = 0; t->ip = 0; } } object findInterfaceMethod(Thread* t, object method, object class_) { object interface = methodClass(t, method); object itable = classInterfaceTable(t, class_); for (unsigned i = 0; i < arrayLength(t, itable); i += 2) { if (arrayBody(t, itable, i) == interface) { return arrayBody(t, arrayBody(t, itable, i + 1), methodOffset(t, method)); } } abort(t); } inline object findMethod(Thread* t, object method, object class_) { return arrayBody(t, classVirtualTable(t, class_), methodOffset(t, method)); } bool isSuperclass(Thread* t, object class_, object base) { for (object oc = classSuper(t, base); oc; oc = classSuper(t, oc)) { if (oc == class_) { return true; } } return false; } inline bool isSpecialMethod(Thread* t, object method, object class_) { return (classFlags(t, class_) & ACC_SUPER) and strcmp(reinterpret_cast(""), &byteArrayBody(t, methodName(t, method), 0)) != 0 and isSuperclass(t, methodClass(t, method), class_); } object find(Thread* t, object table, object reference, object& (*name)(Thread*, object), object& (*spec)(Thread*, object)) { if (table) { object n = referenceName(t, reference); object s = referenceSpec(t, reference); for (unsigned i = 0; i < arrayLength(t, table); ++i) { object o = arrayBody(t, table, i); if (strcmp(&byteArrayBody(t, name(t, o), 0), &byteArrayBody(t, n, 0)) == 0 and strcmp(&byteArrayBody(t, spec(t, o), 0), &byteArrayBody(t, s, 0)) == 0) { return o; } } } return 0; } inline object findFieldInClass(Thread* t, object class_, object reference) { return find(t, classFieldTable(t, class_), reference, fieldName, fieldSpec); } inline object findMethodInClass(Thread* t, object class_, object reference) { return find(t, classMethodTable(t, class_), reference, methodName, methodSpec); } inline object resolveClass(Thread* t, object pool, unsigned index) { object o = arrayBody(t, pool, index); if (objectClass(t, o) == arrayBody(t, t->vm->types, Machine::ByteArrayType)) { PROTECT(t, pool); o = resolveClass(t, o); if (UNLIKELY(t->exception)) return 0; set(t, arrayBody(t, pool, index), o); } return o; } inline object resolveClass(Thread* t, object container, object& (*class_)(Thread*, object)) { object o = class_(t, container); if (objectClass(t, o) == arrayBody(t, t->vm->types, Machine::ByteArrayType)) { PROTECT(t, container); o = resolveClass(t, o); if (UNLIKELY(t->exception)) return 0; set(t, class_(t, container), o); } return o; } inline object resolve(Thread* t, object pool, unsigned index, object (*find)(Thread*, object, object), object (*makeError)(Thread*, object)) { object o = arrayBody(t, pool, index); if (objectClass(t, o) == arrayBody(t, t->vm->types, Machine::ReferenceType)) { PROTECT(t, pool); object reference = o; PROTECT(t, reference); object class_ = resolveClass(t, o, referenceClass); if (UNLIKELY(t->exception)) return 0; o = 0; if (classFlags(t, class_) & ACC_INTERFACE) { if (classVirtualTable(t, class_)) { o = ::find(t, classVirtualTable(t, class_), arrayBody(t, pool, index), methodName, methodSpec); } } else { for (; o == 0 and class_; class_ = classSuper(t, class_)) { o = find(t, class_, arrayBody(t, pool, index)); } } if (o == 0) { object message = makeString (t, "%s %s not found in %s", &byteArrayBody(t, referenceName(t, reference), 0), &byteArrayBody(t, referenceSpec(t, reference), 0), &byteArrayBody(t, className(t, referenceClass(t, reference)), 0)); t->exception = makeError(t, message); } set(t, arrayBody(t, pool, index), o); } return o; } inline object resolveField(Thread* t, object pool, unsigned index) { return resolve(t, pool, index, findFieldInClass, makeNoSuchFieldError); } inline object resolveMethod(Thread* t, object pool, unsigned index) { return resolve(t, pool, index, findMethodInClass, makeNoSuchMethodError); } object makeNativeMethodData(Thread* t, object method, void* function, bool builtin) { PROTECT(t, method); unsigned count = methodParameterCount(t, method) + 1; if (methodFlags(t, method) & ACC_STATIC) { ++ count; } object data = makeNativeMethodData(t, function, 0, // argument table size 0, // return code, builtin, count, false); unsigned argumentTableSize = BytesPerWord * 2; unsigned index = 0; nativeMethodDataParameterTypes(t, data, index++) = POINTER_TYPE; nativeMethodDataParameterTypes(t, data, index++) = POINTER_TYPE; const char* s = reinterpret_cast (&byteArrayBody(t, methodSpec(t, method), 0)); ++ s; // skip '(' while (*s and *s != ')') { unsigned code = fieldCode(t, *s); nativeMethodDataParameterTypes(t, data, index++) = fieldType(t, code); switch (*s) { case 'L': argumentTableSize += BytesPerWord; while (*s and *s != ';') ++ s; ++ s; break; case '[': argumentTableSize += BytesPerWord; while (*s == '[') ++ s; switch (*s) { case 'L': while (*s and *s != ';') ++ s; ++ s; break; default: ++ s; break; } break; default: argumentTableSize += pad(primitiveSize(t, code)); ++ s; break; } } nativeMethodDataArgumentTableSize(t, data) = argumentTableSize; nativeMethodDataReturnCode(t, data) = fieldCode(t, s[1]); return data; } inline object resolveNativeMethodData(Thread* t, object method) { if (objectClass(t, methodCode(t, method)) == arrayBody(t, t->vm->types, Machine::ByteArrayType)) { object data = 0; for (System::Library* lib = t->vm->libraries; lib; lib = lib->next()) { void* p = lib->resolve(reinterpret_cast (&byteArrayBody(t, methodCode(t, method), 0))); if (p) { PROTECT(t, method); data = makeNativeMethodData(t, method, p, false); break; } } if (data == 0) { object p = hashMapFind(t, t->vm->builtinMap, methodCode(t, method), byteArrayHash, byteArrayEqual); if (p) { PROTECT(t, method); data = makeNativeMethodData(t, method, pointerValue(t, p), true); } } if (LIKELY(data)) { set(t, methodCode(t, method), data); } else { object message = makeString (t, "%s", &byteArrayBody(t, methodCode(t, method), 0)); t->exception = makeUnsatisfiedLinkError(t, message); } return data; } else { return methodCode(t, method); } } inline void checkStack(Thread* t, object method) { if (UNLIKELY(t->sp + methodParameterFootprint(t, method) + codeMaxLocals(t, methodCode(t, method)) + FrameFootprint + codeMaxStack(t, methodCode(t, method)) > Thread::StackSizeInWords / 2)) { t->exception = makeStackOverflowError(t); } } unsigned invokeNative(Thread* t, object method) { PROTECT(t, method); object data = resolveNativeMethodData(t, method); if (UNLIKELY(t->exception)) { return VoidField; } PROTECT(t, data); pushFrame(t, method); unsigned count = methodParameterCount(t, method); if (methodFlags(t, method) & ACC_STATIC) { ++ count; } unsigned size = nativeMethodDataArgumentTableSize(t, data); uintptr_t args[size / BytesPerWord]; unsigned offset = 0; args[offset++] = reinterpret_cast(t); unsigned i = 0; if (methodFlags(t, method) & ACC_STATIC) { ++ i; args[offset++] = reinterpret_cast (pushReference(t, methodClass(t, method))); } unsigned sp = frameBase(t, t->frame); for (; i < count; ++i) { unsigned type = nativeMethodDataParameterTypes(t, data, i + 1); switch (type) { case INT8_TYPE: case INT16_TYPE: case INT32_TYPE: case FLOAT_TYPE: args[offset++] = peekInt(t, sp++); break; case INT64_TYPE: case DOUBLE_TYPE: { uint64_t v = peekLong(t, sp); memcpy(args + offset, &v, 8); offset += (8 / BytesPerWord); sp += 2; } break; case POINTER_TYPE: args[offset++] = reinterpret_cast (t->stack + ((sp++) * 2) + 1); break; default: abort(t); } } unsigned returnCode = nativeMethodDataReturnCode(t, data); unsigned returnType = fieldType(t, returnCode); void* function = nativeMethodDataFunction(t, data); bool builtin = nativeMethodDataBuiltin(t, data); Thread::State oldState = t->state; if (not builtin) { enter(t, Thread::IdleState); } if (DebugRun) { fprintf(stderr, "invoke native method %s.%s\n", &byteArrayBody(t, className(t, methodClass(t, method)), 0), &byteArrayBody(t, methodName(t, method), 0)); } uint64_t result = t->vm->system->call (function, args, &nativeMethodDataParameterTypes(t, data, 0), count + 1, size, returnType); if (DebugRun) { fprintf(stderr, "return from native method %s.%s\n", &byteArrayBody (t, className(t, methodClass(t, frameMethod(t, t->frame))), 0), &byteArrayBody (t, methodName(t, frameMethod(t, t->frame)), 0)); } if (not builtin) { enter(t, oldState); } popFrame(t); if (UNLIKELY(t->exception)) { return VoidField; } switch (returnCode) { case ByteField: case BooleanField: case CharField: case ShortField: case FloatField: case IntField: if (DebugRun) { fprintf(stderr, "result: %"LLD"\n", result); } pushInt(t, result); break; case LongField: case DoubleField: if (DebugRun) { fprintf(stderr, "result: %"LLD"\n", result); } pushLong(t, result); break; case ObjectField: if (DebugRun) { fprintf(stderr, "result: %p at %p\n", result == 0 ? 0 : *reinterpret_cast(static_cast(result)), reinterpret_cast(static_cast(result))); } pushObject(t, result == 0 ? 0 : *reinterpret_cast(static_cast(result))); break; case VoidField: break; default: abort(t); }; return returnCode; } bool classInit2(Thread* t, object class_, unsigned ipOffset) { PROTECT(t, class_); acquire(t, t->vm->classLock); if (classVmFlags(t, class_) & NeedInitFlag and (classVmFlags(t, class_) & InitFlag) == 0) { classVmFlags(t, class_) |= InitFlag; t->code = classInitializer(t, class_); t->ip -= ipOffset; return true; } else { release(t, t->vm->classLock); return false; } } inline bool classInit(Thread* t, object class_, unsigned ipOffset) { if (UNLIKELY(classVmFlags(t, class_) & NeedInitFlag)) { return classInit2(t, class_, ipOffset); } else { return false; } } inline int16_t codeReadInt16(Thread* t, unsigned& i) { uint8_t v1 = codeBody(t, t->code, i++); uint8_t v2 = codeBody(t, t->code, i++); return ((v1 << 8) | v2); } inline int32_t codeReadInt32(Thread* t, unsigned& i) { uint8_t v1 = codeBody(t, t->code, i++); uint8_t v2 = codeBody(t, t->code, i++); uint8_t v3 = codeBody(t, t->code, i++); uint8_t v4 = codeBody(t, t->code, i++); return ((v1 << 24) | (v2 << 16) | (v3 << 8) | v4); } object run(Thread* t) { const int base = t->frame; unsigned instruction = nop; unsigned& ip = t->ip; unsigned& sp = t->sp; int& frame = t->frame; object& code = t->code; object& exception = t->exception; uintptr_t* stack = t->stack; if (UNLIKELY(exception)) { goto throw_; } if (UNLIKELY(classInit(t, methodClass(t, frameMethod(t, frame)), 0))) { goto invoke; } loop: instruction = codeBody(t, code, ip++); if (DebugRun) { fprintf(stderr, "ip: %d; instruction: 0x%x in %s.%s ", ip - 1, instruction, &byteArrayBody (t, className(t, methodClass(t, frameMethod(t, frame))), 0), &byteArrayBody (t, methodName(t, frameMethod(t, frame)), 0)); int line = lineNumber(t, frameMethod(t, frame), ip); switch (line) { case NativeLine: fprintf(stderr, "(native)\n"); break; case UnknownLine: fprintf(stderr, "(unknown line)\n"); break; default: fprintf(stderr, "(line %d)\n", line); } } switch (instruction) { case aaload: { int32_t index = popInt(t); object array = popObject(t); if (LIKELY(array)) { if (LIKELY(index >= 0 and static_cast(index) < objectArrayLength(t, array))) { pushObject(t, objectArrayBody(t, array, index)); } else { object message = makeString(t, "%d not in [0,%d)", index, objectArrayLength(t, array)); exception = makeArrayIndexOutOfBoundsException(t, message); goto throw_; } } else { exception = makeNullPointerException(t); goto throw_; } } goto loop; case aastore: { object value = popObject(t); int32_t index = popInt(t); object array = popObject(t); if (LIKELY(array)) { if (LIKELY(index >= 0 and static_cast(index) < objectArrayLength(t, array))) { set(t, objectArrayBody(t, array, index), value); } else { object message = makeString(t, "%d not in [0,%d)", index, objectArrayLength(t, array)); exception = makeArrayIndexOutOfBoundsException(t, message); goto throw_; } } else { exception = makeNullPointerException(t); goto throw_; } } goto loop; case aconst_null: { pushObject(t, 0); } goto loop; case aload: { pushObject(t, localObject(t, codeBody(t, code, ip++))); } goto loop; case aload_0: { pushObject(t, localObject(t, 0)); } goto loop; case aload_1: { pushObject(t, localObject(t, 1)); } goto loop; case aload_2: { pushObject(t, localObject(t, 2)); } goto loop; case aload_3: { pushObject(t, localObject(t, 3)); } goto loop; case anewarray: { int32_t count = popInt(t); if (LIKELY(count >= 0)) { uint16_t index = codeReadInt16(t, ip); object class_ = resolveClass(t, codePool(t, code), index - 1); if (UNLIKELY(exception)) goto throw_; pushObject(t, makeObjectArray(t, class_, count, true)); } else { object message = makeString(t, "%d", count); exception = makeNegativeArraySizeException(t, message); goto throw_; } } goto loop; case areturn: { object result = popObject(t); if (frame > base) { popFrame(t); pushObject(t, result); goto loop; } else { return result; } } goto loop; case arraylength: { object array = popObject(t); if (LIKELY(array)) { pushInt(t, cast(array, BytesPerWord)); } else { exception = makeNullPointerException(t); goto throw_; } } goto loop; case astore: { setLocalObject(t, codeBody(t, code, ip++), popObject(t)); } goto loop; case astore_0: { setLocalObject(t, 0, popObject(t)); } goto loop; case astore_1: { setLocalObject(t, 1, popObject(t)); } goto loop; case astore_2: { setLocalObject(t, 2, popObject(t)); } goto loop; case astore_3: { setLocalObject(t, 3, popObject(t)); } goto loop; case athrow: { exception = popObject(t); if (UNLIKELY(exception == 0)) { exception = makeNullPointerException(t); } } goto throw_; case baload: { int32_t index = popInt(t); object array = popObject(t); if (LIKELY(array)) { if (LIKELY(index >= 0 and static_cast(index) < byteArrayLength(t, array))) { pushInt(t, byteArrayBody(t, array, index)); } else { object message = makeString(t, "%d not in [0,%d)", index, byteArrayLength(t, array)); exception = makeArrayIndexOutOfBoundsException(t, message); goto throw_; } } else { exception = makeNullPointerException(t); goto throw_; } } goto loop; case bastore: { int8_t value = popInt(t); int32_t index = popInt(t); object array = popObject(t); if (LIKELY(array)) { if (LIKELY(index >= 0 and static_cast(index) < byteArrayLength(t, array))) { byteArrayBody(t, array, index) = value; } else { object message = makeString(t, "%d not in [0,%d)", index, byteArrayLength(t, array)); exception = makeArrayIndexOutOfBoundsException(t, message); goto throw_; } } else { exception = makeNullPointerException(t); goto throw_; } } goto loop; case bipush: { pushInt(t, static_cast(codeBody(t, code, ip++))); } goto loop; case caload: { int32_t index = popInt(t); object array = popObject(t); if (LIKELY(array)) { if (LIKELY(index >= 0 and static_cast(index) < charArrayLength(t, array))) { pushInt(t, charArrayBody(t, array, index)); } else { object message = makeString(t, "%d not in [0,%d)", index, charArrayLength(t, array)); exception = makeArrayIndexOutOfBoundsException(t, message); goto throw_; } } else { exception = makeNullPointerException(t); goto throw_; } } goto loop; case castore: { uint16_t value = popInt(t); int32_t index = popInt(t); object array = popObject(t); if (LIKELY(array)) { if (LIKELY(index >= 0 and static_cast(index) < charArrayLength(t, array))) { charArrayBody(t, array, index) = value; } else { object message = makeString(t, "%d not in [0,%d)", index, charArrayLength(t, array)); exception = makeArrayIndexOutOfBoundsException(t, message); goto throw_; } } else { exception = makeNullPointerException(t); goto throw_; } } goto loop; case checkcast: { uint16_t index = codeReadInt16(t, ip); if (peekObject(t, sp - 1)) { object class_ = resolveClass(t, codePool(t, code), index - 1); if (UNLIKELY(exception)) goto throw_; if (not instanceOf(t, class_, peekObject(t, sp - 1))) { object message = makeString (t, "%s as %s", &byteArrayBody (t, className(t, objectClass(t, peekObject(t, sp - 1))), 0), &byteArrayBody(t, className(t, class_), 0)); exception = makeClassCastException(t, message); goto throw_; } } } goto loop; case d2f: { pushFloat(t, static_cast(popDouble(t))); } goto loop; case d2i: { pushInt(t, static_cast(popDouble(t))); } goto loop; case d2l: { pushLong(t, static_cast(popDouble(t))); } goto loop; case dadd: { double b = popDouble(t); double a = popDouble(t); pushDouble(t, a + b); } goto loop; case daload: { int32_t index = popInt(t); object array = popObject(t); if (LIKELY(array)) { if (LIKELY(index >= 0 and static_cast(index) < doubleArrayLength(t, array))) { double d; memcpy(&d, &doubleArrayBody(t, array, index), sizeof(double)); pushDouble(t, d); } else { object message = makeString(t, "%d not in [0,%d)", index, doubleArrayLength(t, array)); exception = makeArrayIndexOutOfBoundsException(t, message); goto throw_; } } else { exception = makeNullPointerException(t); goto throw_; } } goto loop; case dastore: { double value = popDouble(t); int32_t index = popInt(t); object array = popObject(t); if (LIKELY(array)) { if (LIKELY(index >= 0 and static_cast(index) < doubleArrayLength(t, array))) { memcpy(&doubleArrayBody(t, array, index), &value, sizeof(uint64_t)); } else { object message = makeString(t, "%d not in [0,%d)", index, doubleArrayLength(t, array)); exception = makeArrayIndexOutOfBoundsException(t, message); goto throw_; } } else { exception = makeNullPointerException(t); goto throw_; } } goto loop; case dcmpg: { double b = popDouble(t); double a = popDouble(t); pushInt(t, (a > b ? 1 : 0)); } goto loop; case dcmpl: { double b = popDouble(t); double a = popDouble(t); pushInt(t, (a < b ? 1 : 0)); } goto loop; case dconst_0: { pushDouble(t, 0); } goto loop; case dconst_1: { pushDouble(t, 1); } goto loop; case ddiv: { double b = popDouble(t); double a = popDouble(t); pushDouble(t, a / b); } goto loop; case dmul: { double b = popDouble(t); double a = popDouble(t); pushDouble(t, a * b); } goto loop; case dneg: { double a = popDouble(t); pushDouble(t, - a); } goto loop; case vm::drem: { double b = popDouble(t); double a = popDouble(t); pushDouble(t, fmod(a, b)); } goto loop; case dsub: { double b = popDouble(t); double a = popDouble(t); pushDouble(t, a - b); } goto loop; case dup: { if (DebugStack) { fprintf(stderr, "dup\n"); } memcpy(stack + ((sp ) * 2), stack + ((sp - 1) * 2), BytesPerWord * 2); ++ sp; } goto loop; case dup_x1: { if (DebugStack) { fprintf(stderr, "dup_x1\n"); } memcpy(stack + ((sp ) * 2), stack + ((sp - 1) * 2), BytesPerWord * 2); memcpy(stack + ((sp - 1) * 2), stack + ((sp - 2) * 2), BytesPerWord * 2); memcpy(stack + ((sp - 2) * 2), stack + ((sp ) * 2), BytesPerWord * 2); ++ sp; } goto loop; case dup_x2: { if (DebugStack) { fprintf(stderr, "dup_x2\n"); } memcpy(stack + ((sp ) * 2), stack + ((sp - 1) * 2), BytesPerWord * 2); memcpy(stack + ((sp - 1) * 2), stack + ((sp - 2) * 2), BytesPerWord * 2); memcpy(stack + ((sp - 2) * 2), stack + ((sp - 3) * 2), BytesPerWord * 2); memcpy(stack + ((sp - 3) * 2), stack + ((sp ) * 2), BytesPerWord * 2); ++ sp; } goto loop; case dup2: { if (DebugStack) { fprintf(stderr, "dup2\n"); } memcpy(stack + ((sp + 1) * 2), stack + ((sp - 2) * 2), BytesPerWord * 4); sp += 2; } goto loop; case dup2_x1: { if (DebugStack) { fprintf(stderr, "dup2_x1\n"); } memcpy(stack + ((sp + 1) * 2), stack + ((sp - 1) * 2), BytesPerWord * 2); memcpy(stack + ((sp ) * 2), stack + ((sp - 2) * 2), BytesPerWord * 2); memcpy(stack + ((sp - 1) * 2), stack + ((sp - 3) * 2), BytesPerWord * 2); memcpy(stack + ((sp - 3) * 2), stack + ((sp ) * 2), BytesPerWord * 4); sp += 2; } goto loop; case dup2_x2: { if (DebugStack) { fprintf(stderr, "dup2_x2\n"); } memcpy(stack + ((sp + 1) * 2), stack + ((sp - 1) * 2), BytesPerWord * 2); memcpy(stack + ((sp ) * 2), stack + ((sp - 2) * 2), BytesPerWord * 2); memcpy(stack + ((sp - 1) * 2), stack + ((sp - 3) * 2), BytesPerWord * 2); memcpy(stack + ((sp - 2) * 2), stack + ((sp - 4) * 2), BytesPerWord * 2); memcpy(stack + ((sp - 4) * 2), stack + ((sp ) * 2), BytesPerWord * 4); sp += 2; } goto loop; case f2d: { pushDouble(t, popFloat(t)); } goto loop; case f2i: { pushInt(t, static_cast(popFloat(t))); } goto loop; case f2l: { pushLong(t, static_cast(popFloat(t))); } goto loop; case fadd: { float b = popFloat(t); float a = popFloat(t); pushFloat(t, a + b); } goto loop; case faload: { int32_t index = popInt(t); object array = popObject(t); if (LIKELY(array)) { if (LIKELY(index >= 0 and static_cast(index) < floatArrayLength(t, array))) { float f; memcpy(&f, &floatArrayBody(t, array, index), sizeof(float)); pushFloat(t, f); } else { object message = makeString(t, "%d not in [0,%d)", index, floatArrayLength(t, array)); exception = makeArrayIndexOutOfBoundsException(t, message); goto throw_; } } else { exception = makeNullPointerException(t); goto throw_; } } goto loop; case fastore: { float value = popFloat(t); int32_t index = popInt(t); object array = popObject(t); if (LIKELY(array)) { if (LIKELY(index >= 0 and static_cast(index) < floatArrayLength(t, array))) { memcpy(&floatArrayBody(t, array, index), &value, sizeof(uint32_t)); } else { object message = makeString(t, "%d not in [0,%d)", index, floatArrayLength(t, array)); exception = makeArrayIndexOutOfBoundsException(t, message); goto throw_; } } else { exception = makeNullPointerException(t); goto throw_; } } goto loop; case fcmpg: { float b = popFloat(t); float a = popFloat(t); pushInt(t, (a > b ? 1 : 0)); } goto loop; case fcmpl: { float b = popFloat(t); float a = popFloat(t); pushInt(t, (a < b ? 1 : 0)); } goto loop; case fconst_0: { pushFloat(t, 0); } goto loop; case fconst_1: { pushFloat(t, 1); } goto loop; case fconst_2: { pushFloat(t, 2); } goto loop; case fdiv: { float b = popFloat(t); float a = popFloat(t); pushFloat(t, a / b); } goto loop; case fmul: { float b = popFloat(t); float a = popFloat(t); pushFloat(t, a * b); } goto loop; case fneg: { float a = popFloat(t); pushFloat(t, - a); } goto loop; case frem: { float b = popFloat(t); float a = popFloat(t); pushFloat(t, fmodf(a, b)); } goto loop; case fsub: { float b = popFloat(t); float a = popFloat(t); pushFloat(t, a - b); } goto loop; case getfield: { if (LIKELY(peekObject(t, sp - 1))) { uint16_t index = codeReadInt16(t, ip); object field = resolveField(t, codePool(t, code), index - 1); if (UNLIKELY(exception)) goto throw_; object instance = popObject(t); switch (fieldCode(t, field)) { case ByteField: case BooleanField: pushInt(t, cast(instance, fieldOffset(t, field))); break; case CharField: case ShortField: pushInt(t, cast(instance, fieldOffset(t, field))); break; case FloatField: case IntField: pushInt(t, cast(instance, fieldOffset(t, field))); break; case DoubleField: case LongField: pushLong(t, cast(instance, fieldOffset(t, field))); break; case ObjectField: pushObject(t, cast(instance, fieldOffset(t, field))); break; default: abort(t); } } else { exception = makeNullPointerException(t); goto throw_; } } goto loop; case getstatic: { uint16_t index = codeReadInt16(t, ip); object field = resolveField(t, codePool(t, code), index - 1); if (UNLIKELY(exception)) goto throw_; if (UNLIKELY(classInit(t, fieldClass(t, field), 3))) goto invoke; object v = arrayBody(t, classStaticTable(t, fieldClass(t, field)), fieldOffset(t, field)); switch (fieldCode(t, field)) { case ByteField: case BooleanField: case CharField: case ShortField: case FloatField: case IntField: pushInt(t, intValue(t, v)); break; case DoubleField: case LongField: pushLong(t, longValue(t, v)); break; case ObjectField: pushObject(t, v); break; default: abort(t); } } goto loop; case goto_: { int16_t offset = codeReadInt16(t, ip); ip = (ip - 3) + offset; } goto loop; case goto_w: { int32_t offset = codeReadInt32(t, ip); ip = (ip - 5) + offset; } goto loop; case i2b: { pushInt(t, static_cast(popInt(t))); } goto loop; case i2c: { pushInt(t, static_cast(popInt(t))); } goto loop; case i2l: { pushLong(t, popInt(t)); } goto loop; case i2s: { pushInt(t, static_cast(popInt(t))); } goto loop; case iadd: { int32_t b = popInt(t); int32_t a = popInt(t); pushInt(t, a + b); } goto loop; case iaload: { int32_t index = popInt(t); object array = popObject(t); if (LIKELY(array)) { if (LIKELY(index >= 0 and static_cast(index) < intArrayLength(t, array))) { pushInt(t, intArrayBody(t, array, index)); } else { object message = makeString(t, "%d not in [0,%d)", index, intArrayLength(t, array)); exception = makeArrayIndexOutOfBoundsException(t, message); goto throw_; } } else { exception = makeNullPointerException(t); goto throw_; } } goto loop; case iand: { int32_t b = popInt(t); int32_t a = popInt(t); pushInt(t, a & b); } goto loop; case iastore: { int32_t value = popInt(t); int32_t index = popInt(t); object array = popObject(t); if (LIKELY(array)) { if (LIKELY(index >= 0 and static_cast(index) < intArrayLength(t, array))) { intArrayBody(t, array, index) = value; } else { object message = makeString(t, "%d not in [0,%d)", index, intArrayLength(t, array)); exception = makeArrayIndexOutOfBoundsException(t, message); goto throw_; } } else { exception = makeNullPointerException(t); goto throw_; } } goto loop; case iconst_m1: { pushInt(t, static_cast(-1)); } goto loop; case iconst_0: { pushInt(t, 0); } goto loop; case iconst_1: { pushInt(t, 1); } goto loop; case iconst_2: { pushInt(t, 2); } goto loop; case iconst_3: { pushInt(t, 3); } goto loop; case iconst_4: { pushInt(t, 4); } goto loop; case iconst_5: { pushInt(t, 5); } goto loop; case idiv: { int32_t b = popInt(t); int32_t a = popInt(t); pushInt(t, a / b); } goto loop; case if_acmpeq: { uint8_t offset1 = codeBody(t, code, ip++); uint8_t offset2 = codeBody(t, code, ip++); object b = popObject(t); object a = popObject(t); if (a == b) { ip = (ip - 3) + static_cast(((offset1 << 8) | offset2)); } } goto loop; case if_acmpne: { uint8_t offset1 = codeBody(t, code, ip++); uint8_t offset2 = codeBody(t, code, ip++); object b = popObject(t); object a = popObject(t); if (a != b) { ip = (ip - 3) + static_cast(((offset1 << 8) | offset2)); } } goto loop; case if_icmpeq: { uint8_t offset1 = codeBody(t, code, ip++); uint8_t offset2 = codeBody(t, code, ip++); int32_t b = popInt(t); int32_t a = popInt(t); if (a == b) { ip = (ip - 3) + static_cast(((offset1 << 8) | offset2)); } } goto loop; case if_icmpne: { uint8_t offset1 = codeBody(t, code, ip++); uint8_t offset2 = codeBody(t, code, ip++); int32_t b = popInt(t); int32_t a = popInt(t); if (a != b) { ip = (ip - 3) + static_cast(((offset1 << 8) | offset2)); } } goto loop; case if_icmpgt: { uint8_t offset1 = codeBody(t, code, ip++); uint8_t offset2 = codeBody(t, code, ip++); int32_t b = popInt(t); int32_t a = popInt(t); if (a > b) { ip = (ip - 3) + static_cast(((offset1 << 8) | offset2)); } } goto loop; case if_icmpge: { uint8_t offset1 = codeBody(t, code, ip++); uint8_t offset2 = codeBody(t, code, ip++); int32_t b = popInt(t); int32_t a = popInt(t); if (a >= b) { ip = (ip - 3) + static_cast(((offset1 << 8) | offset2)); } } goto loop; case if_icmplt: { uint8_t offset1 = codeBody(t, code, ip++); uint8_t offset2 = codeBody(t, code, ip++); int32_t b = popInt(t); int32_t a = popInt(t); if (a < b) { ip = (ip - 3) + static_cast(((offset1 << 8) | offset2)); } } goto loop; case if_icmple: { uint8_t offset1 = codeBody(t, code, ip++); uint8_t offset2 = codeBody(t, code, ip++); int32_t b = popInt(t); int32_t a = popInt(t); if (a <= b) { ip = (ip - 3) + static_cast(((offset1 << 8) | offset2)); } } goto loop; case ifeq: { uint8_t offset1 = codeBody(t, code, ip++); uint8_t offset2 = codeBody(t, code, ip++); if (popInt(t) == 0) { ip = (ip - 3) + static_cast(((offset1 << 8) | offset2)); } } goto loop; case ifne: { uint8_t offset1 = codeBody(t, code, ip++); uint8_t offset2 = codeBody(t, code, ip++); if (popInt(t)) { ip = (ip - 3) + static_cast(((offset1 << 8) | offset2)); } } goto loop; case ifgt: { uint8_t offset1 = codeBody(t, code, ip++); uint8_t offset2 = codeBody(t, code, ip++); if (static_cast(popInt(t)) > 0) { ip = (ip - 3) + static_cast(((offset1 << 8) | offset2)); } } goto loop; case ifge: { uint8_t offset1 = codeBody(t, code, ip++); uint8_t offset2 = codeBody(t, code, ip++); if (static_cast(popInt(t)) >= 0) { ip = (ip - 3) + static_cast(((offset1 << 8) | offset2)); } } goto loop; case iflt: { uint8_t offset1 = codeBody(t, code, ip++); uint8_t offset2 = codeBody(t, code, ip++); if (static_cast(popInt(t)) < 0) { ip = (ip - 3) + static_cast(((offset1 << 8) | offset2)); } } goto loop; case ifle: { uint8_t offset1 = codeBody(t, code, ip++); uint8_t offset2 = codeBody(t, code, ip++); if (static_cast(popInt(t)) <= 0) { ip = (ip - 3) + static_cast(((offset1 << 8) | offset2)); } } goto loop; case ifnonnull: { uint8_t offset1 = codeBody(t, code, ip++); uint8_t offset2 = codeBody(t, code, ip++); if (popObject(t)) { ip = (ip - 3) + static_cast(((offset1 << 8) | offset2)); } } goto loop; case ifnull: { uint8_t offset1 = codeBody(t, code, ip++); uint8_t offset2 = codeBody(t, code, ip++); if (popObject(t) == 0) { ip = (ip - 3) + static_cast(((offset1 << 8) | offset2)); } } goto loop; case iinc: { uint8_t index = codeBody(t, code, ip++); int8_t c = codeBody(t, code, ip++); setLocalInt(t, index, localInt(t, index) + c); } goto loop; case iload: case fload: { pushInt(t, localInt(t, codeBody(t, code, ip++))); } goto loop; case iload_0: case fload_0: { pushInt(t, localInt(t, 0)); } goto loop; case iload_1: case fload_1: { pushInt(t, localInt(t, 1)); } goto loop; case iload_2: case fload_2: { pushInt(t, localInt(t, 2)); } goto loop; case iload_3: case fload_3: { pushInt(t, localInt(t, 3)); } goto loop; case imul: { int32_t b = popInt(t); int32_t a = popInt(t); pushInt(t, a * b); } goto loop; case ineg: { pushInt(t, - popInt(t)); } goto loop; case instanceof: { uint8_t index1 = codeBody(t, code, ip++); uint8_t index2 = codeBody(t, code, ip++); if (peekObject(t, sp - 1)) { uint16_t index = (index1 << 8) | index2; object class_ = resolveClass(t, codePool(t, code), index - 1); if (UNLIKELY(exception)) goto throw_; if (instanceOf(t, class_, popObject(t))) { pushInt(t, 1); } else { pushInt(t, 0); } } else { popObject(t); pushInt(t, 0); } } goto loop; case invokeinterface: { uint16_t index = codeReadInt16(t, ip); ip += 2; object method = resolveMethod(t, codePool(t, code), index - 1); if (UNLIKELY(exception)) goto throw_; unsigned parameterFootprint = methodParameterFootprint(t, method); if (LIKELY(peekObject(t, sp - parameterFootprint))) { code = findInterfaceMethod (t, method, objectClass(t, peekObject(t, sp - parameterFootprint))); goto invoke; } else { exception = makeNullPointerException(t); goto throw_; } } goto loop; case invokespecial: { uint16_t index = codeReadInt16(t, ip); object method = resolveMethod(t, codePool(t, code), index - 1); if (UNLIKELY(exception)) goto throw_; unsigned parameterFootprint = methodParameterFootprint(t, method); if (LIKELY(peekObject(t, sp - parameterFootprint))) { object class_ = methodClass(t, frameMethod(t, frame)); if (isSpecialMethod(t, method, class_)) { class_ = classSuper(t, class_); if (classVirtualTable(t, class_) == 0) { PROTECT(t, method); PROTECT(t, class_); resolveClass(t, className(t, class_)); if (UNLIKELY(exception)) goto throw_; if (UNLIKELY(classInit(t, class_, 3))) goto invoke; } code = findMethod(t, method, class_); } else { code = method; } goto invoke; } else { exception = makeNullPointerException(t); goto throw_; } } goto loop; case invokestatic: { uint16_t index = codeReadInt16(t, ip); object method = resolveMethod(t, codePool(t, code), index - 1); if (UNLIKELY(exception)) goto throw_; if (UNLIKELY(classInit(t, methodClass(t, method), 3))) goto invoke; code = method; } goto invoke; case invokevirtual: { uint16_t index = codeReadInt16(t, ip); object method = resolveMethod(t, codePool(t, code), index - 1); if (UNLIKELY(exception)) goto throw_; unsigned parameterFootprint = methodParameterFootprint(t, method); if (LIKELY(peekObject(t, sp - parameterFootprint))) { object class_ = objectClass(t, peekObject(t, sp - parameterFootprint)); if (classVirtualTable(t, class_) == 0) { PROTECT(t, method); PROTECT(t, class_); resolveClass(t, className(t, class_)); if (UNLIKELY(exception)) goto throw_; if (UNLIKELY(classInit(t, class_, 3))) goto invoke; } code = findMethod(t, method, class_); goto invoke; } else { exception = makeNullPointerException(t); goto throw_; } } goto loop; case ior: { int32_t b = popInt(t); int32_t a = popInt(t); pushInt(t, a | b); } goto loop; case irem: { int32_t b = popInt(t); int32_t a = popInt(t); pushInt(t, a % b); } goto loop; case ireturn: case freturn: { int32_t result = popInt(t); if (frame > base) { popFrame(t); pushInt(t, result); goto loop; } else { return makeInt(t, result); } } goto loop; case ishl: { int32_t b = popInt(t); int32_t a = popInt(t); pushInt(t, a << b); } goto loop; case ishr: { int32_t b = popInt(t); int32_t a = popInt(t); pushInt(t, a >> b); } goto loop; case istore: case fstore: { setLocalInt(t, codeBody(t, code, ip++), popInt(t)); } goto loop; case istore_0: case fstore_0: { setLocalInt(t, 0, popInt(t)); } goto loop; case istore_1: case fstore_1: { setLocalInt(t, 1, popInt(t)); } goto loop; case istore_2: case fstore_2: { setLocalInt(t, 2, popInt(t)); } goto loop; case istore_3: case fstore_3: { setLocalInt(t, 3, popInt(t)); } goto loop; case isub: { int32_t b = popInt(t); int32_t a = popInt(t); pushInt(t, a - b); } goto loop; case iushr: { int32_t b = popInt(t); int32_t a = popInt(t); pushInt(t, static_cast(a >> b)); } goto loop; case ixor: { int32_t b = popInt(t); int32_t a = popInt(t); pushInt(t, a ^ b); } goto loop; case jsr: { uint8_t offset1 = codeBody(t, code, ip++); uint8_t offset2 = codeBody(t, code, ip++); pushInt(t, ip); ip = (ip - 3) + static_cast(((offset1 << 8) | offset2)); } goto loop; case jsr_w: { uint8_t offset1 = codeBody(t, code, ip++); uint8_t offset2 = codeBody(t, code, ip++); uint8_t offset3 = codeBody(t, code, ip++); uint8_t offset4 = codeBody(t, code, ip++); pushInt(t, ip); ip = (ip - 3) + static_cast ((offset1 << 24) | (offset2 << 16) | (offset3 << 8) | offset4); } goto loop; case l2i: { pushInt(t, static_cast(popLong(t))); } goto loop; case ladd: { int64_t b = popLong(t); int64_t a = popLong(t); pushLong(t, a + b); } goto loop; case laload: { int32_t index = popInt(t); object array = popObject(t); if (LIKELY(array)) { if (LIKELY(index >= 0 and static_cast(index) < longArrayLength(t, array))) { pushLong(t, longArrayBody(t, array, index)); } else { object message = makeString(t, "%d not in [0,%d)", index, longArrayLength(t, array)); exception = makeArrayIndexOutOfBoundsException(t, message); goto throw_; } } else { exception = makeNullPointerException(t); goto throw_; } } goto loop; case land: { int64_t b = popLong(t); int64_t a = popLong(t); pushLong(t, a & b); } goto loop; case lastore: { int64_t value = popLong(t); int32_t index = popInt(t); object array = popObject(t); if (LIKELY(array)) { if (LIKELY(index >= 0 and static_cast(index) < longArrayLength(t, array))) { longArrayBody(t, array, index) = value; } else { object message = makeString(t, "%d not in [0,%d)", index, longArrayLength(t, array)); exception = makeArrayIndexOutOfBoundsException(t, message); goto throw_; } } else { exception = makeNullPointerException(t); goto throw_; } } goto loop; case lcmp: { int64_t b = popLong(t); int64_t a = popLong(t); pushInt(t, a > b ? 1 : a == b ? 0 : -1); } goto loop; case lconst_0: { pushLong(t, 0); } goto loop; case lconst_1: { pushLong(t, 1); } goto loop; case ldc: case ldc_w: { uint16_t index; if (instruction == ldc) { index = codeBody(t, code, ip++); } else { uint8_t index1 = codeBody(t, code, ip++); uint8_t index2 = codeBody(t, code, ip++); index = (index1 << 8) | index2; } object v = arrayBody(t, codePool(t, code), index - 1); if (objectClass(t, v) == arrayBody(t, t->vm->types, Machine::IntType)) { pushInt(t, intValue(t, v)); } else if (objectClass(t, v) == arrayBody(t, t->vm->types, Machine::FloatType)) { pushInt(t, floatValue(t, v)); } else if (objectClass(t, v) == arrayBody(t, t->vm->types, Machine::StringType)) { pushObject(t, v); } else { object class_ = resolveClass(t, codePool(t, code), index - 1); if (UNLIKELY(exception)) goto throw_; pushObject(t, class_); } } goto loop; case ldc2_w: { uint8_t index1 = codeBody(t, code, ip++); uint8_t index2 = codeBody(t, code, ip++); object v = arrayBody(t, codePool(t, code), ((index1 << 8) | index2) - 1); if (objectClass(t, v) == arrayBody(t, t->vm->types, Machine::LongType)) { pushLong(t, longValue(t, v)); } else if (objectClass(t, v) == arrayBody(t, t->vm->types, Machine::DoubleType)) { pushLong(t, doubleValue(t, v)); } else { abort(t); } } goto loop; case ldiv_: { int64_t b = popLong(t); int64_t a = popLong(t); pushLong(t, a / b); } goto loop; case lload: case dload: { pushLong(t, localLong(t, codeBody(t, code, ip++))); } goto loop; case lload_0: case dload_0: { pushLong(t, localLong(t, 0)); } goto loop; case lload_1: case dload_1: { pushLong(t, localLong(t, 1)); } goto loop; case lload_2: case dload_2: { pushLong(t, localLong(t, 2)); } goto loop; case lload_3: case dload_3: { pushLong(t, localLong(t, 3)); } goto loop; case lmul: { int64_t b = popLong(t); int64_t a = popLong(t); pushLong(t, a * b); } goto loop; case lneg: { pushLong(t, - popLong(t)); } goto loop; case lookupswitch: { int32_t base = ip - 1; ip += 3; ip -= (ip % 4); int32_t default_ = codeReadInt32(t, ip); int32_t pairCount = codeReadInt32(t, ip); int32_t key = popInt(t); int32_t bottom = 0; int32_t top = pairCount; for (int32_t span = top - bottom; span; span = top - bottom) { int32_t middle = bottom + (span / 2); unsigned index = ip + (middle * 8); int32_t k = codeReadInt32(t, index); if (key < k) { top = middle; } else if (key > k) { bottom = middle + 1; } else { ip = base + codeReadInt32(t, index); goto loop; } } ip = base + default_; } goto loop; case lor: { int64_t b = popLong(t); int64_t a = popLong(t); pushLong(t, a | b); } goto loop; case lrem: { int64_t b = popLong(t); int64_t a = popLong(t); pushLong(t, a % b); } goto loop; case lreturn: case dreturn: { int64_t result = popLong(t); if (frame > base) { popFrame(t); pushLong(t, result); goto loop; } else { return makeLong(t, result); } } goto loop; case lshl: { int64_t b = popLong(t); int64_t a = popLong(t); pushLong(t, a << b); } goto loop; case lshr: { int64_t b = popLong(t); int64_t a = popLong(t); pushLong(t, a >> b); } goto loop; case lstore: case dstore: { setLocalLong(t, codeBody(t, code, ip++), popLong(t)); } goto loop; case lstore_0: case dstore_0:{ setLocalLong(t, 0, popLong(t)); } goto loop; case lstore_1: case dstore_1: { setLocalLong(t, 1, popLong(t)); } goto loop; case lstore_2: case dstore_2: { setLocalLong(t, 2, popLong(t)); } goto loop; case lstore_3: case dstore_3: { setLocalLong(t, 3, popLong(t)); } goto loop; case lsub: { int64_t b = popLong(t); int64_t a = popLong(t); pushLong(t, a - b); } goto loop; case lushr: { uint64_t b = popLong(t); uint64_t a = popLong(t); pushLong(t, a >> b); } goto loop; case lxor: { int64_t b = popLong(t); int64_t a = popLong(t); pushLong(t, a ^ b); } goto loop; case monitorenter: { object o = popObject(t); if (LIKELY(o)) { acquire(t, o); } else { exception = makeNullPointerException(t); goto throw_; } } goto loop; case monitorexit: { object o = popObject(t); if (LIKELY(o)) { release(t, o); } else { exception = makeNullPointerException(t); goto throw_; } } goto loop; case new_: { uint16_t index = codeReadInt16(t, ip); object class_ = resolveClass(t, codePool(t, code), index - 1); if (UNLIKELY(exception)) goto throw_; if (UNLIKELY(classInit(t, class_, 3))) goto invoke; pushObject(t, make(t, class_)); } goto loop; case newarray: { int32_t count = popInt(t); if (LIKELY(count >= 0)) { uint8_t type = codeBody(t, code, ip++); object array; switch (type) { case T_BOOLEAN: array = makeBooleanArray(t, count, true); break; case T_CHAR: array = makeCharArray(t, count, true); break; case T_FLOAT: array = makeFloatArray(t, count, true); break; case T_DOUBLE: array = makeDoubleArray(t, count, true); break; case T_BYTE: array = makeByteArray(t, count, true); break; case T_SHORT: array = makeShortArray(t, count, true); break; case T_INT: array = makeIntArray(t, count, true); break; case T_LONG: array = makeLongArray(t, count, true); break; default: abort(t); } pushObject(t, array); } else { object message = makeString(t, "%d", count); exception = makeNegativeArraySizeException(t, message); goto throw_; } } goto loop; case nop: goto loop; case pop_: { -- sp; } goto loop; case pop2: { sp -= 2; } goto loop; case putfield: { uint16_t index = codeReadInt16(t, ip); object field = resolveField(t, codePool(t, code), index - 1); if (UNLIKELY(exception)) goto throw_; switch (fieldCode(t, field)) { case ByteField: case BooleanField: case CharField: case ShortField: case FloatField: case IntField: { int32_t value = popInt(t); object o = popObject(t); if (LIKELY(o)) { switch (fieldCode(t, field)) { case ByteField: case BooleanField: cast(o, fieldOffset(t, field)) = value; break; case CharField: case ShortField: cast(o, fieldOffset(t, field)) = value; break; case FloatField: case IntField: cast(o, fieldOffset(t, field)) = value; break; } } else { exception = makeNullPointerException(t); goto throw_; } } break; case DoubleField: case LongField: { int64_t value = popLong(t); object o = popObject(t); if (LIKELY(o)) { cast(o, fieldOffset(t, field)) = value; } else { exception = makeNullPointerException(t); goto throw_; } } break; case ObjectField: { object value = popObject(t); object o = popObject(t); if (LIKELY(o)) { set(t, cast(o, fieldOffset(t, field)), value); } else { exception = makeNullPointerException(t); goto throw_; } } break; default: abort(t); } } goto loop; case putstatic: { uint16_t index = codeReadInt16(t, ip); object field = resolveField(t, codePool(t, code), index - 1); if (UNLIKELY(exception)) goto throw_; if (UNLIKELY(classInit(t, fieldClass(t, field), 3))) goto invoke; PROTECT(t, field); object v; switch (fieldCode(t, field)) { case ByteField: case BooleanField: case CharField: case ShortField: case FloatField: case IntField: { v = makeInt(t, popInt(t)); } break; case DoubleField: case LongField: { v = makeLong(t, popLong(t)); } break; case ObjectField: v = popObject(t); break; default: abort(t); } set(t, arrayBody(t, classStaticTable(t, fieldClass(t, field)), fieldOffset(t, field)), v); } goto loop; case ret: { ip = localInt(t, codeBody(t, code, ip)); } goto loop; case return_: { if (frame > base) { popFrame(t); goto loop; } else { return 0; } } goto loop; case saload: { int32_t index = popInt(t); object array = popObject(t); if (LIKELY(array)) { if (LIKELY(index >= 0 and static_cast(index) < shortArrayLength(t, array))) { pushInt(t, shortArrayBody(t, array, index)); } else { object message = makeString(t, "%d not in [0,%d)", index, shortArrayLength(t, array)); exception = makeArrayIndexOutOfBoundsException(t, message); goto throw_; } } else { exception = makeNullPointerException(t); goto throw_; } } goto loop; case sastore: { int16_t value = popInt(t); int32_t index = popInt(t); object array = popObject(t); if (LIKELY(array)) { if (LIKELY(index >= 0 and static_cast(index) < shortArrayLength(t, array))) { shortArrayBody(t, array, index) = value; } else { object message = makeString(t, "%d not in [0,%d)", index, shortArrayLength(t, array)); exception = makeArrayIndexOutOfBoundsException(t, message); goto throw_; } } else { exception = makeNullPointerException(t); goto throw_; } } goto loop; case sipush: { uint8_t byte1 = codeBody(t, code, ip++); uint8_t byte2 = codeBody(t, code, ip++); pushInt(t, static_cast((byte1 << 8) | byte2)); } goto loop; case swap: { uintptr_t tmp[2]; memcpy(tmp , stack + ((sp - 1) * 2), BytesPerWord * 2); memcpy(stack + ((sp - 1) * 2), stack + ((sp - 2) * 2), BytesPerWord * 2); memcpy(stack + ((sp - 2) * 2), tmp , BytesPerWord * 2); } goto loop; case tableswitch: { int32_t base = ip - 1; ip += 3; ip -= (ip % 4); int32_t default_ = codeReadInt32(t, ip); int32_t bottom = codeReadInt32(t, ip); int32_t top = codeReadInt32(t, ip); int32_t key = popInt(t); if (key >= bottom and key <= top) { unsigned index = ip + ((key - bottom) * 4); ip = base + codeReadInt32(t, index); } else { ip = base + default_; } } goto loop; case wide: goto wide; default: abort(t); } wide: switch (codeBody(t, code, ip++)) { case aload: { pushObject(t, localObject(t, codeReadInt16(t, ip))); } goto loop; case astore: { setLocalObject(t, codeReadInt16(t, ip), popObject(t)); } goto loop; case iinc: { uint16_t index = codeReadInt16(t, ip); uint16_t count = codeReadInt16(t, ip); setLocalInt(t, index, localInt(t, index) + count); } goto loop; case iload: { pushInt(t, localInt(t, codeReadInt16(t, ip))); } goto loop; case istore: { setLocalInt(t, codeReadInt16(t, ip), popInt(t)); } goto loop; case lload: { pushLong(t, localLong(t, codeReadInt16(t, ip))); } goto loop; case lstore: { setLocalLong(t, codeReadInt16(t, ip), popLong(t)); } goto loop; case ret: { ip = localInt(t, codeReadInt16(t, ip)); } goto loop; default: abort(t); } invoke: { if (methodFlags(t, code) & ACC_NATIVE) { invokeNative(t, code); if (UNLIKELY(exception)) goto throw_; } else { checkStack(t, code); if (UNLIKELY(exception)) goto throw_; pushFrame(t, code); } } goto loop; throw_: if (DebugRun) { fprintf(stderr, "throw\n"); } pokeInt(t, t->frame + FrameIpOffset, t->ip); for (; frame >= base; popFrame(t)) { code = methodCode(t, frameMethod(t, frame)); object eht = codeExceptionHandlerTable(t, code); if (eht) { for (unsigned i = 0; i < exceptionHandlerTableLength(t, eht); ++i) { ExceptionHandler* eh = exceptionHandlerTableBody(t, eht, i); if (frameIp(t, frame) - 1 >= exceptionHandlerStart(eh) and frameIp(t, frame) - 1 < exceptionHandlerEnd(eh)) { object catchType = 0; if (exceptionHandlerCatchType(eh)) { object e = exception; exception = 0; PROTECT(t, e); PROTECT(t, eht); catchType = resolveClass (t, codePool(t, code), exceptionHandlerCatchType(eh) - 1); if (catchType) { eh = exceptionHandlerTableBody(t, eht, i); exception = e; } else { // can't find what we're supposed to catch - move on. continue; } } if (catchType == 0 or instanceOf(t, catchType, exception)) { sp = frame + FrameFootprint; ip = exceptionHandlerIp(eh); pushObject(t, exception); exception = 0; goto loop; } } } } } return 0; } void run(Thread* t, const char* className, int argc, const char** argv) { enter(t, Thread::ActiveState); object args = makeObjectArray (t, arrayBody(t, t->vm->types, Machine::StringType), argc, true); PROTECT(t, args); for (int i = 0; i < argc; ++i) { object arg = makeString(t, "%s", argv[i]); set(t, objectArrayBody(t, args, i), arg); } run(t, className, "main", "([Ljava/lang/String;)V", 0, args); } void pushArguments(Thread* t, object this_, const char* spec, va_list a) { if (this_) { pushObject(t, this_); } const char* s = spec; ++ s; // skip '(' while (*s and *s != ')') { switch (*s) { case 'L': while (*s and *s != ';') ++ s; ++ s; pushObject(t, va_arg(a, object)); break; case '[': while (*s == '[') ++ s; switch (*s) { case 'L': while (*s and *s != ';') ++ s; ++ s; break; default: ++ s; break; } pushObject(t, va_arg(a, object)); break; case 'J': case 'D': ++ s; pushLong(t, va_arg(a, uint64_t)); break; default: ++ s; pushInt(t, va_arg(a, uint32_t)); break; } } } void pushArguments(Thread* t, object this_, const char* spec, object a) { if (this_) { pushObject(t, this_); } unsigned index = 0; const char* s = spec; ++ s; // skip '(' while (*s and *s != ')') { switch (*s) { case 'L': while (*s and *s != ';') ++ s; ++ s; pushObject(t, objectArrayBody(t, a, index++)); break; case '[': while (*s == '[') ++ s; switch (*s) { case 'L': while (*s and *s != ';') ++ s; ++ s; break; default: ++ s; break; } pushObject(t, objectArrayBody(t, a, index++)); break; case 'J': case 'D': ++ s; pushLong(t, cast(objectArrayBody(t, a, index++), BytesPerWord)); break; default: ++ s; pushInt(t, cast(objectArrayBody(t, a, index++), BytesPerWord)); break; } } } inline unsigned returnCode(Thread* t, object method) { const char* s = reinterpret_cast (&byteArrayBody(t, methodSpec(t, method), 0)); while (*s and *s != ')') ++ s; return fieldCode(t, s[1]); } object invoke(Thread* t, object method) { PROTECT(t, method); object class_; PROTECT(t, class_); if (methodFlags(t, method) & ACC_STATIC) { class_ = methodClass(t, method); } else { unsigned parameterFootprint = methodParameterFootprint(t, method); class_ = objectClass(t, peekObject(t, t->sp - parameterFootprint)); if (classFlags(t, methodClass(t, method)) & ACC_INTERFACE) { method = findInterfaceMethod(t, method, class_); } else { if (classVirtualTable(t, class_) == 0) { resolveClass(t, className(t, class_)); if (UNLIKELY(t->exception)) return 0; } method = findMethod(t, method, class_); } } initClass(t, class_); object result = 0; if (methodFlags(t, method) & ACC_NATIVE) { unsigned returnCode = invokeNative(t, method); if (LIKELY(t->exception == 0)) { switch (returnCode) { case ByteField: case BooleanField: case CharField: case ShortField: case FloatField: case IntField: result = makeInt(t, popInt(t)); break; case LongField: case DoubleField: result = makeLong(t, popLong(t)); break; case ObjectField: result = popObject(t); break; case VoidField: result = 0; break; default: abort(t); }; } } else { checkStack(t, method); if (LIKELY(t->exception == 0)) { pushFrame(t, method); result = ::run(t); if (LIKELY(t->exception == 0)) { popFrame(t); } } } switch (returnCode(t, method)) { case ByteField: return makeByte(t, static_cast(intValue(t, result))); case BooleanField: return makeBoolean(t, static_cast(intValue(t, result))); case CharField: return makeChar(t, static_cast(intValue(t, result))); case ShortField: return makeShort(t, static_cast(intValue(t, result))); case FloatField: return makeFloat(t, static_cast(intValue(t, result))); case DoubleField: return makeDouble(t, static_cast(longValue(t, result))); case ObjectField: case IntField: case LongField: case VoidField: return result; default: abort(t); } } } // namespace namespace vm { object run(Thread* t, object method, object this_, ...) { assert(t, t->state == Thread::ActiveState or t->state == Thread::ExclusiveState); assert(t, ((methodFlags(t, method) & ACC_STATIC) == 0) xor (this_ == 0)); if (UNLIKELY(t->sp + methodParameterFootprint(t, method) + 1 > Thread::StackSizeInWords / 2)) { t->exception = makeStackOverflowError(t); return 0; } va_list a; va_start(a, this_); const char* spec = reinterpret_cast (&byteArrayBody(t, methodSpec(t, method), 0)); pushArguments(t, this_, spec, a); va_end(a); return invoke(t, method); } object run2(Thread* t, object method, object this_, object arguments) { assert(t, t->state == Thread::ActiveState or t->state == Thread::ExclusiveState); assert(t, ((methodFlags(t, method) & ACC_STATIC) == 0) xor (this_ == 0)); if (UNLIKELY(t->sp + methodParameterFootprint(t, method) + 1 > Thread::StackSizeInWords / 2)) { t->exception = makeStackOverflowError(t); return 0; } const char* spec = reinterpret_cast (&byteArrayBody(t, methodSpec(t, method), 0)); pushArguments(t, this_, spec, arguments); return invoke(t, method); } object run(Thread* t, const char* className, const char* methodName, const char* methodSpec, object this_, ...) { assert(t, t->state == Thread::ActiveState or t->state == Thread::ExclusiveState); if (UNLIKELY(t->sp + parameterFootprint(methodSpec) + 1 > Thread::StackSizeInWords / 2)) { t->exception = makeStackOverflowError(t); return 0; } va_list a; va_start(a, this_); pushArguments(t, this_, methodSpec, a); va_end(a); object class_ = resolveClass(t, makeByteArray(t, "%s", className)); if (LIKELY(t->exception == 0)) { PROTECT(t, class_); object name = makeByteArray(t, methodName); PROTECT(t, name); object spec = makeByteArray(t, methodSpec); object reference = makeReference(t, class_, name, spec); object method = findMethodInClass(t, class_, reference); if (LIKELY(t->exception == 0)) { assert(t, ((methodFlags(t, method) & ACC_STATIC) == 0) xor (this_ == 0)); return invoke(t, method); } } return 0; } int run(System* system, Heap* heap, Finder* finder, const char* className, int argc, const char** argv) { Machine m(system, heap, finder); Thread* t = new (system->allocate(sizeof(Thread))) Thread(&m, 0, 0); enter(t, Thread::ActiveState); ::run(t, className, argc, argv); int exitCode = 0; if (t->exception) { exitCode = -1; printTrace(t, t->exception); } exit(t); return exitCode; } } // namespace vm