namespace { object run(Thread* t) { unsigned ip = 0; #define PUSH(x) t->stack[(t->sp)++] = x #define POP(x) x = t->stack[--(t->sp)] loop: switch (codeBody(t->code)[ip++]) { case aaload: { object index; POP(index); object array; POP(array); if (array) { int32_t i = intValue(index); if (i >= 0 and i < objectArrayLength(array)) { PUSH(objectArrayBody(array)[i]); } else { object message = makeString(t, "%d not in [0,%d]", i, objectArrayLength(array)); t->exception = makeArrayIndexOutOfBoundsException(t, message); goto throw_; } } else { t->exception = makeNullPointerException(t, 0); goto throw_; } } goto loop; case aastore: { object value; POP(value); object index; POP(index); object array; POP(array); int32_t i = intValue(index); if (array) { if (i >= 0 and i < objectArrayLength(array)) { set(t, objectArrayBody(array)[i], value); } else { object message = makeString(t, "%d not in [0,%d]", i, objectArrayLength(array)); t->exception = makeArrayIndexOutOfBoundsException(t, message); goto throw_; } } else { t->exception = makeNullPointerException(t, 0); goto throw_; } } goto loop; case aconst_null: { PUSH(0); } goto loop; case aload: { PUSH(frameBody(t->frame)[codeBody(t->code)[ip++]]); } goto loop; case aload_0: { PUSH(frameBody(t->frame)[0]); } goto loop; case aload_1: { PUSH(frameBody(t->frame)[1]); } goto loop; case aload_2: { PUSH(frameBody(t->frame)[2]); } goto loop; case aload_3: { PUSH(frameBody(t->frame)[3]); } goto loop; case anewarray: { object count; POP(count); int32_t c = intValue(count); if (c >= 0) { uint8_t index1 = codeBody(t->code)[ip++]; uint8_t index2 = codeBody(t->code)[ip++]; uint16_t index = (index1 << 8) | index2; object class_ = resolveClass(t, codePool(t->code), index); if (t->exception) goto throw_; object array = makeObjectArray(t, class_, c); memset(objectArrayBody(array), 0, c * 4); PUSH(array); } else { object message = makeString(t, "%d", c); t->exception = makeNegativeArrayStoreException(t, message); goto throw_; } } goto loop; case areturn: { object value; POP(value); if (t->sp) { POP(t->frame); t->code = frameCode(t->frame); ip = frameIp(t->frame); PUSH(value); goto loop; } else { return value; } } goto loop; case arraylength: { object array; POP(array); if (array) { PUSH(makeInt(t, arrayLength(array))); } else { t->exception = makeNullPointerException(t, 0); goto throw_; } } UNREACHABLE; case astore: { object value; POP(value); set(t, frameBody(t->frame)[codeBody(t->code)[ip++]], value); } goto loop; case astore_0: { object value; POP(value); set(t, frameBody(t->frame)[0], value); } goto loop; case astore_1: { object value; POP(value); set(t, frameBody(t->frame)[1], value); } goto loop; case astore_2: { object value; POP(value); set(t, frameBody(t->frame)[2], value); } goto loop; case astore_3: { object value; POP(value); set(t, frameBody(t->frame)[3], value); } goto loop; case athrow: { POP(t->exception); if (t->exception == 0) { t->exception = makeNullPointerException(t, 0); } goto throw_; } UNREACHABLE; case baload: { object index; POP(index); object array; POP(array); if (array) { int32_t i = intValue(index); if (i >= 0 and i < byteArrayLength(array)) { PUSH(makeByte(t, byteArrayBody(array)[i])); } else { object message = makeString(t, "%d not in [0,%d]", i, byteArrayLength(array)); t->exception = makeArrayIndexOutOfBoundsException(t, message); goto throw_; } } else { t->exception = makeNullPointerException(t, 0); goto throw_; } } goto loop; case bastore: { object value; POP(value); object index; POP(index); object array; POP(array); int32_t i = intValue(index); if (array) { if (i >= 0 and i < byteArrayLength(array)) { byteArrayBody(array)[i] = intValue(value); } else { object message = makeString(t, "%d not in [0,%d]", i, byteArrayLength(array)); t->exception = makeArrayIndexOutOfBoundsException(t, message); goto throw_; } } else { t->exception = makeNullPointerException(t, 0); goto throw_; } } goto loop; case bipush: { PUSH(makeInt(t, codeBody(t->code)[ip++])); } goto loop; case caload: { object index; POP(index); object array; POP(array); if (array) { int32_t i = intValue(index); if (i >= 0 and i < charArrayLength(array)) { PUSH(makeInt(t, charArrayBody(array)[i])); } else { object message = makeString(t, "%d not in [0,%d]", i, charArrayLength(array)); t->exception = makeArrayIndexOutOfBoundsException(t, message); goto throw_; } } else { t->exception = makeNullPointerException(t, 0); goto throw_; } } goto loop; case castore: { object value; POP(value); object index; POP(index); object array; POP(array); int32_t i = intValue(index); if (array) { if (i >= 0 and i < charArrayLength(array)) { charArrayBody(array)[i] = intValue(value); } else { object message = makeString(t, "%d not in [0,%d]", i, charArrayLength(array)); t->exception = makeArrayIndexOutOfBoundsException(t, message); goto throw_; } } else { t->exception = makeNullPointerException(t, 0); goto throw_; } } goto loop; case checkcast: { uint8_t index1 = codeBody(t->code)[ip++]; uint8_t index2 = codeBody(t->code)[ip++]; if (t->stack[t->sp - 1]) { uint16_t index = (index1 << 8) | index2; object class_ = resolveClass(t, codePool(t->code), index); if (t->exception) goto throw_; if (not instanceOf(t, class_, t->stack[t->sp - 1])) { t->exception = makeClassCastException(t, 0); goto throw_; } } } goto loop; case dup: { object value = t->stack[t->sp - 1]; PUSH(value); } goto loop; case dup_x1: { object first; POP(first); object second; POP(second); PUSH(first); PUSH(second); PUSH(first); } goto loop; case dup_x2: { object first; POP(first); object second; POP(second); object third; POP(third); PUSH(first); PUSH(third); PUSH(second); PUSH(first); } goto loop; case dup2: { object first = t->stack[t->sp - 1]; if (isLongOrDouble(first)) { PUSH(first); } else { object second = t->stack[t->sp - 2]; PUSH(second); PUSH(first); } } goto loop; case dup2_x1: { object first; POP(first); object second; POP(second); if (isLongOrDouble(first)) { PUSH(first); PUSH(second); PUSH(first); } else { object third; POP(third); PUSH(second); PUSH(first); PUSH(third); PUSH(second); PUSH(first); } } goto loop; case dup2_x2: { object first; POP(first); object second; POP(second); if (isLongOrDouble(first)) { if (isLongOrDouble(second)) { PUSH(first); PUSH(second); PUSH(first); } else { object third; POP(third); PUSH(first); PUSH(third); PUSH(second); PUSH(first); } } else { object third; POP(third); if (isLongOrDouble(third)) { PUSH(second); PUSH(first); PUSH(third); PUSH(second); PUSH(first); } else { object fourth; POP(fourth); PUSH(second); PUSH(first); PUSH(fourth); PUSH(third); PUSH(second); PUSH(first); } } } goto loop; case getfield: { object instance; POP(instance); if (instance) { uint8_t index1 = codeBody(t->code)[ip++]; uint8_t index2 = codeBody(t->code)[ip++]; uint16_t index = (index1 << 8) | index2; object field = resolveField(t, codePool(t->code), index); if (t->exception) goto throw_; PUSH(getField(instance, field)); } else { t->exception = makeNullPointerException(t, 0); goto throw_; } } goto loop; case getstatic: { if (instance) { uint8_t index1 = codeBody(t->code)[ip++]; uint8_t index2 = codeBody(t->code)[ip++]; uint16_t index = (index1 << 8) | index2; object field = resolveField(t, codePool(t->code), index); if (t->exception) goto throw_; if (not classInitialized(fieldClass(field))) { frameIp(t->frame) = ip - 3; PUSH(t->frame); t->code = classInitializer(fieldClass(field)); t->frame = makeFrame(t, t->code); ip = 0; } PUSH(getStatic(field)); } else { t->exception = makeNullPointerException(t, 0); goto throw_; } } goto loop; case goto_: { uint8_t offset1 = codeBody(t->code)[ip++]; uint8_t offset2 = codeBody(t->code)[ip++]; ip = (ip - 1) + ((offset1 << 8) | offset2); } goto loop; case goto_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++]; ip = (ip - 1) + ((offset1 << 24) | (offset2 << 16) | (offset3 << 8) | offset4); } goto loop; case i2b: { object v; POP(v); PUSH(makeInt(t, static_cast(intValue(v)))); } goto loop; case i2c: { object v; POP(v); PUSH(makeInt(t, static_cast(intValue(v)))); } goto loop; case i2l: { object v; POP(v); PUSH(makeLong(t, intValue(v))); } goto loop; case i2s: { object v; POP(v); PUSH(makeInt(t, static_cast(intValue(v)))); } goto loop; case iadd: { object b; POP(b); object a; POP(a); PUSH(makeInt(t, intValue(a) + intValue(b))); } goto loop; case iaload: { object index; POP(index); object array; POP(array); if (array) { int32_t i = intValue(index); if (i >= 0 and i < intArrayLength(array)) { PUSH(makeInt(t, intArrayBody(array)[i])); } else { object message = makeString(t, "%d not in [0,%d]", i, intArrayLength(array)); t->exception = makeArrayIndexOutOfBoundsException(t, message); goto throw_; } } else { t->exception = makeNullPointerException(t, 0); goto throw_; } } goto loop; case iand: { object b; POP(b); object a; POP(a); PUSH(makeInt(t, intValue(a) & intValue(b))); } goto loop; case iastore: { object value; POP(value); object index; POP(index); object array; POP(array); int32_t i = intValue(index); if (array) { if (i >= 0 and i < intArrayLength(array)) { intArrayBody(array)[i] = intValue(value); } else { object message = makeString(t, "%d not in [0,%d]", i, intArrayLength(array)); t->exception = makeArrayIndexOutOfBoundsException(t, message); goto throw_; } } else { t->exception = makeNullPointerException(t, 0); goto throw_; } } goto loop; case iconst_0: { PUSH(makeInt(0)); } goto loop; case iconst_1: { PUSH(makeInt(1)); } goto loop; case iconst_2: { PUSH(makeInt(2)); } goto loop; case iconst_3: { PUSH(makeInt(3)); } goto loop; case iconst_4: { PUSH(makeInt(4)); } goto loop; case iconst_5: { PUSH(makeInt(5)); } goto loop; case idiv: { object b; POP(b); object a; POP(a); PUSH(makeInt(t, intValue(a) / intValue(b))); } goto loop; case if_acmpeq: { uint8_t offset1 = codeBody(t->code)[ip++]; uint8_t offset2 = codeBody(t->code)[ip++]; object b; POP(b); object a; POP(a); if (a == b) { ip = (ip - 1) + ((offset1 << 8) | offset2); } } goto loop; case if_acmpne: { uint8_t offset1 = codeBody(t->code)[ip++]; uint8_t offset2 = codeBody(t->code)[ip++]; object b; POP(b); object a; POP(a); if (a != b) { ip = (ip - 1) + ((offset1 << 8) | offset2); } } goto loop; case if_icmpeq: { uint8_t offset1 = codeBody(t->code)[ip++]; uint8_t offset2 = codeBody(t->code)[ip++]; object b; POP(b); object a; POP(a); if (intValue(a) == intValue(b)) { ip = (ip - 1) + ((offset1 << 8) | offset2); } } goto loop; case if_icmpne: { uint8_t offset1 = codeBody(t->code)[ip++]; uint8_t offset2 = codeBody(t->code)[ip++]; object b; POP(b); object a; POP(a); if (intValue(a) != intValue(b)) { ip = (ip - 1) + ((offset1 << 8) | offset2); } } goto loop; case if_icmpgt: { uint8_t offset1 = codeBody(t->code)[ip++]; uint8_t offset2 = codeBody(t->code)[ip++]; object b; POP(b); object a; POP(a); if (intValue(a) > intValue(b)) { ip = (ip - 1) + ((offset1 << 8) | offset2); } } goto loop; case if_icmpge: { uint8_t offset1 = codeBody(t->code)[ip++]; uint8_t offset2 = codeBody(t->code)[ip++]; object b; POP(b); object a; POP(a); if (intValue(a) >= intValue(b)) { ip = (ip - 1) + ((offset1 << 8) | offset2); } } goto loop; case if_icmplt: { uint8_t offset1 = codeBody(t->code)[ip++]; uint8_t offset2 = codeBody(t->code)[ip++]; object b; POP(b); object a; POP(a); if (intValue(a) < intValue(b)) { ip = (ip - 1) + ((offset1 << 8) | offset2); } } goto loop; case if_icmple: { uint8_t offset1 = codeBody(t->code)[ip++]; uint8_t offset2 = codeBody(t->code)[ip++]; object b; POP(b); object a; POP(a); if (intValue(a) < intValue(b)) { ip = (ip - 1) + ((offset1 << 8) | offset2); } } goto loop; case ifeq: { uint8_t offset1 = codeBody(t->code)[ip++]; uint8_t offset2 = codeBody(t->code)[ip++]; object v; POP(v); if (intValue(v) == 0) { ip = (ip - 1) + ((offset1 << 8) | offset2); } } goto loop; case ifne: { uint8_t offset1 = codeBody(t->code)[ip++]; uint8_t offset2 = codeBody(t->code)[ip++]; object v; POP(v); if (intValue(v)) { ip = (ip - 1) + ((offset1 << 8) | offset2); } } goto loop; case ifgt: { uint8_t offset1 = codeBody(t->code)[ip++]; uint8_t offset2 = codeBody(t->code)[ip++]; object v; POP(v); if (intValue(v) > 0) { ip = (ip - 1) + ((offset1 << 8) | offset2); } } goto loop; case ifge: { uint8_t offset1 = codeBody(t->code)[ip++]; uint8_t offset2 = codeBody(t->code)[ip++]; object v; POP(v); if (intValue(v) >= 0) { ip = (ip - 1) + ((offset1 << 8) | offset2); } } goto loop; case iflt: { uint8_t offset1 = codeBody(t->code)[ip++]; uint8_t offset2 = codeBody(t->code)[ip++]; object v; POP(v); if (intValue(v) < 0) { ip = (ip - 1) + ((offset1 << 8) | offset2); } } goto loop; case ifle: { uint8_t offset1 = codeBody(t->code)[ip++]; uint8_t offset2 = codeBody(t->code)[ip++]; object v; POP(v); if (intValue(v) <= 0) { ip = (ip - 1) + ((offset1 << 8) | offset2); } } goto loop; case ifnonnull: { uint8_t offset1 = codeBody(t->code)[ip++]; uint8_t offset2 = codeBody(t->code)[ip++]; object v; POP(v); if (v) { ip = (ip - 1) + ((offset1 << 8) | offset2); } } goto loop; case ifnull: { uint8_t offset1 = codeBody(t->code)[ip++]; uint8_t offset2 = codeBody(t->code)[ip++]; object v; POP(v); if (v == 0) { ip = (ip - 1) + ((offset1 << 8) | offset2); } } goto loop; case iinc: { uint8_t index = codeBody(t->code)[ip++]; int8_t c = codeBody(t->code)[ip++]; int32_t v = intValue(frameBody(t->frame)[index]); frameBody(t->frame)[index] = makeInt(t, v + c); } goto loop; case iload: { PUSH(makeInt(t, intValue(frameBody(t->frame)[codeBody(t->code)[ip++]]))); } goto loop; case iload_0: { PUSH(makeInt(t, intValue(frameBody(t->frame)[0]))); } goto loop; case iload_1: { PUSH(makeInt(t, intValue(frameBody(t->frame)[1]))); } goto loop; case iload_2: { PUSH(makeInt(t, intValue(frameBody(t->frame)[2]))); } goto loop; case iload_3: { PUSH(makeInt(t, intValue(frameBody(t->frame)[3]))); } goto loop; case imul: { object b; POP(b); object a; POP(a); PUSH(makeInt(t, intValue(a) * intValue(b))); } goto loop; case ineg: { object v; POP(v); PUSH(makeInt(t, - intValue(v))); } goto loop; case instanceof: { uint8_t index1 = codeBody(t->code)[ip++]; uint8_t index2 = codeBody(t->code)[ip++]; if (t->stack[t->sp - 1]) { uint16_t index = (index1 << 8) | index2; object class_ = resolveClass(t, codePool(t->code), index); if (t->exception) goto throw_; if (instanceOf(t, class_, t->stack[t->sp - 1])) { PUSH(makeInt(t, 1)); } else { PUSH(makeInt(t, 0)); } } else { PUSH(makeInt(t, 0)); } } goto loop; case invokeinterface: { // todo } goto loop; default: UNREACHABLE; } throw_: for (; t->frame; t->frame = frameNext(t->frame)) { t->code = frameCode(t->frame); object eht = codeExceptionHandlerTable(t->code); if (eht) { for (unsigned i = 0; i < exceptionHandleTableLength(eht); ++i) { ExceptionHandler* eh = exceptionHandlerTableBody(eht)[i]; uint16_t catchType = exceptionHandlerCatchType(eh); if (catchType == 0 or instanceOf(rawArrayBody(codePool(t->code))[catchType], t->exception)) { t->sp = frameStackBase(t->frame); ip = exceptionHandlerIp(eh); PUSH(t->exception); t->exception = 0; goto loop; } } } } t->code = defaultExceptionHandler(t); t->frame = makeFrame(t, t->code); t->sp = 0; ip = 0; PUSH(t->exception); t->exception = 0; goto loop; } } // namespace