lots of bugfixes

This commit is contained in:
Joel Dice 2007-06-20 19:38:02 -06:00
parent ccd71b0d99
commit 95c4bff51b
5 changed files with 498 additions and 347 deletions

View File

@ -8,6 +8,7 @@ cxx = g++
cc = gcc
vg = nice valgrind --leak-check=full --num-callers=32 --db-attach=yes \
--freelist-vol=100000000
javac = javac
warnings = -Wall -Wextra -Werror -Wold-style-cast -Wunused-parameter \
-Winit-self -Wconversion
@ -51,7 +52,6 @@ interpreter-sources = \
$(src)/main.cpp
interpreter-objects = $(call cpp-objects,$(interpreter-sources),$(src))
interpreter-cflags = $(slow) $(cflags)
input = Test
generator-headers = \
$(src)/input.h \
@ -74,32 +74,42 @@ fast-objects = $(patsubst $(bld)/%,$(bld)/fast-%,$(interpreter-objects))
fast-executable = $(bld)/fast-vm
fast-cflags = $(fast) $(cflags)
input = $(bld)/classes/Test.class
input-depends = \
$(bld)/classes/java/lang/Object.class \
$(bld)/classes/java/lang/Class.class \
$(bld)/classes/vm/VM.class
gen-run-arg = $(shell echo $(1) | sed -e 's:$(bld)/classes/\(.*\)\.class:\1:')
.PHONY: build
build: $(executable)
$(input): $(input-depends)
.PHONY: run
run: $(executable)
$(<) $(input)
run: $(executable) $(input)
$(<) -cp $(bld)/classes $(call gen-run-arg,$(input))
.PHONY: debug
debug: $(executable)
gdb --args $(<) $(input)
debug: $(executable) $(input)
gdb --args $(<) -cp $(bld)/classes $(call gen-run-arg,$(input))
.PHONY: fast
fast: $(fast-executable)
ls -lh $(<)
.PHONY: vg
vg: $(executable)
$(vg) $(<) $(input)
vg: $(executable) $(input)
$(vg) $(<) -cp $(bld)/classes $(call gen-run-arg,$(input))
.PHONY: test
test: $(test-executable)
$(vg) $(<) $(input)
test: $(test-executable) $(input)
$(vg) $(<) -cp $(bld)/classes $(call gen-run-arg,$(input))
.PHONY: stress
stress: $(stress-executable)
$(vg) $(<) $(input)
stress: $(stress-executable) $(input)
$(vg) $(<) -cp $(bld)/classes $(call gen-run-arg,$(input))
.PHONY: run-all
run-all: $(executable)
@ -136,6 +146,11 @@ $(bld)/fast-vm.o: \
$(bld)/type-generator.o: \
$(generator-headers)
$(bld)/classes/%.class: $(inp)/%.java
@echo "compiling $(@)"
@mkdir -p $(dir $(@))
$(javac) -bootclasspath $(inp) -classpath $(inp) -d $(bld)/classes $(<)
$(stdcpp-objects): $(bld)/%.o: $(src)/%.cpp
@echo "compiling $(@)"
@mkdir -p $(dir $(@))

View File

@ -4,191 +4,210 @@
namespace vm {
enum OpCode {
aaload,
aastore,
aconst_null,
aload,
aload_0,
aload_1,
aload_2,
aload_3,
anewarray,
areturn,
arraylength,
astore,
astore_0,
astore_1,
astore_2,
astore_3,
athrow,
baload,
bastore,
bipush,
breakpoint,
caload,
castore,
checkcast,
d2f,
d2i,
d2l,
dadd,
daload,
dastore,
dcmpg,
dcmpl,
dconst,
ddiv,
dload,
dmul,
dneg,
drem,
dreturn,
dstore,
dsub,
dup,
dup_x1,
dup_x2,
dup2,
dup2_x1,
dup2_x2,
f2d,
f2i,
f2l,
fadd,
faload,
fastore,
fcmpg,
fcmpl,
fconst,
fdiv,
fload,
fmul,
fneg,
frem,
freturn,
fstore,
fsub,
getfield,
getstatic,
goto_,
goto_w,
i2b,
i2c,
i2d,
i2f,
i2l,
i2s,
iadd,
iaload,
iand,
iastore,
iconst_0,
iconst_1,
iconst_2,
iconst_3,
iconst_4,
iconst_5,
idiv,
if_,
if_acmpeq,
if_acmpne,
if_icmpeq,
if_icmpne,
if_icmpgt,
if_icmpge,
if_icmplt,
if_icmple,
ifeq,
ifge,
ifgt,
ifle,
iflt,
ifne,
ifnonnull,
ifnull,
iinc,
iload,
iload_0,
iload_1,
iload_2,
iload_3,
impdep1,
impdep2,
imul,
ineg,
instanceof,
invokeinterface,
invokespecial,
invokestatic,
invokevirtual,
ior,
irem,
ireturn,
ishl,
ishr,
istore,
istore_0,
istore_1,
istore_2,
istore_3,
isub,
iushr,
ixor,
jsr,
jsr_w,
l2d,
l2f,
l2i,
ladd,
laload,
land,
lastore,
lcmp,
lconst_0,
lconst_1,
ldc,
ldc_w,
ldc2_w,
ldiv,
lload,
lload_0,
lload_1,
lload_2,
lload_3,
lmul,
lneg,
lookupswitch,
lor,
lrem,
lreturn,
lshl,
lshr,
lstore,
lstore_0,
lstore_1,
lstore_2,
lstore_3,
lsub,
lushr,
lxor,
monitorenter,
monitorexit,
multianewarray,
new_,
newarray,
nop,
pop,
pop2,
putfield,
putstatic,
ret,
return_,
saload,
sastore,
sipush,
swap,
tableswitch,
wide
aaload = 0x32,
aastore = 0x53,
aconst_null = 0x01,
aload = 0x19,
aload_0 = 0x2a,
aload_1 = 0x2b,
aload_2 = 0x2c,
aload_3 = 0x2d,
anewarray = 0xbd,
areturn = 0xb0,
arraylength = 0xbe,
astore = 0x3a,
astore_0 = 0x4b,
astore_1 = 0x4c,
astore_2 = 0x4d,
astore_3 = 0x4e,
athrow = 0xbf,
baload = 0x33,
bastore = 0x54,
bipush = 0x10,
breakpoint = 0xca,
caload = 0x34,
castore = 0x55,
checkcast = 0xc0,
d2f = 0x90,
d2i = 0x8e,
d2l = 0x8f,
dadd = 0x63,
daload = 0x31,
dastore = 0x52,
dcmpg = 0x98,
dcmpl = 0x97,
dconst_0 = 0x0e,
dconst_1 = 0x0f,
ddiv = 0x6f,
dload = 0x18,
dload_0 = 0x26,
dload_1 = 0x27,
dload_2 = 0x28,
dload_3 = 0x29,
dmul = 0x6b,
dneg = 0x77,
drem = 0x73,
dreturn = 0xaf,
dstore = 0x39,
dstore_0 = 0x47,
dstore_1 = 0x48,
dstore_2 = 0x49,
dstore_3 = 0x4a,
dsub = 0x67,
dup = 0x59,
dup_x1 = 0x5a,
dup_x2 = 0x5b,
dup2 = 0x5c,
dup2_x1 = 0x5d,
dup2_x2 = 0x5e,
f2d = 0x8d,
f2i = 0x8b,
f2l = 0x8c,
fadd = 0x62,
faload = 0x30,
fastore = 0x51,
fcmpg = 0x96,
fcmpl = 0x95,
fconst_0 = 0x0b,
fconst_1 = 0x0c,
fconst_2 = 0x0d,
fdiv = 0x6e,
fload = 0x17,
fload_0 = 0x22,
fload_1 = 0x23,
fload_2 = 0x24,
fload_3 = 0x25,
fmul = 0x6a,
fneg = 0x76,
frem = 0x72,
freturn = 0xae,
fstore = 0x38,
fstore_0 = 0x43,
fstore_1 = 0x44,
fstore_2 = 0x45,
fstore_3 = 0x46,
fsub = 0x66,
getfield = 0xb4,
getstatic = 0xb2,
goto_ = 0xa7,
goto_w = 0xc8,
i2b = 0x91,
i2c = 0x92,
i2d = 0x87,
i2f = 0x86,
i2l = 0x85,
i2s = 0x93,
iadd = 0x60,
iaload = 0x2e,
iand = 0x73,
iastore = 0x4f,
iconst_m1 = 0x02,
iconst_0 = 0x03,
iconst_1 = 0x04,
iconst_2 = 0x05,
iconst_3 = 0x06,
iconst_4 = 0x07,
iconst_5 = 0x08,
idiv = 0x6c,
if_acmpeq = 0xa5,
if_acmpne = 0xa6,
if_icmpeq = 0x9f,
if_icmpne = 0xa0,
if_icmpgt = 0xa1,
if_icmpge = 0xa2,
if_icmplt = 0xa3,
if_icmple = 0xa4,
ifeq = 0x99,
ifge = 0x9c,
ifgt = 0x9d,
ifle = 0x9e,
iflt = 0x9b,
ifne = 0x9a,
ifnonnull = 0xc7,
ifnull = 0xc6,
iinc = 0x84,
iload = 0x15,
iload_0 = 0x1a,
iload_1 = 0x1b,
iload_2 = 0x1c,
iload_3 = 0x1d,
impdep1 = 0xfe,
impdep2 = 0xff,
imul = 0x68,
ineg = 0x74,
instanceof = 0xc1,
invokeinterface = 0xb9,
invokespecial = 0xb7,
invokestatic = 0xb8,
invokevirtual = 0xb6,
ior = 0x80,
irem = 0x70,
ireturn = 0xac,
ishl = 0x78,
ishr = 0x7a,
istore = 0x36,
istore_0 = 0x3b,
istore_1 = 0x3c,
istore_2 = 0x3d,
istore_3 = 0x3e,
isub = 0x64,
iushr = 0x7c,
ixor = 0x82,
jsr = 0xa8,
jsr_w = 0xc9,
l2d = 0x8a,
l2f = 0x89,
l2i = 0x88,
ladd = 0x61,
laload = 0x2f,
land = 0x7f,
lastore = 0x50,
lcmp = 0x94,
lconst_0 = 0x09,
lconst_1 = 0x0a,
ldc = 0x12,
ldc_w = 0x13,
ldc2_w = 0x14,
ldiv = 0x6d,
lload = 0x16,
lload_0 = 0x1e,
lload_1 = 0x1f,
lload_2 = 0x20,
lload_3 = 0x21,
lmul = 0x69,
lneg = 0x75,
lookupswitch = 0xab,
lor = 0x81,
lrem = 0x71,
lreturn = 0xad,
lshl = 0x79,
lshr = 0x7b,
lstore = 0x37,
lstore_0 = 0x3f,
lstore_1 = 0x40,
lstore_2 = 0x41,
lstore_3 = 0x42,
lsub = 0x65,
lushr = 0x7d,
lxor = 0x83,
monitorenter = 0xc2,
monitorexit = 0xc3,
multianewarray = 0xc5,
new_ = 0xbb,
newarray = 0xbc,
nop = 0x00,
pop = 0x57,
pop2 = 0x58,
putfield = 0xb5,
putstatic = 0xb3,
ret = 0xa9,
return_ = 0xb1,
saload = 0x35,
sastore = 0x56,
sipush = 0x11,
swap = 0x5f,
tableswitch = 0xaa,
wide = 0xc4
};
enum TypeCode {

View File

@ -42,13 +42,19 @@ class System: public vm::System {
if (up == 0) abort();
*up = *size;
count += *up;
return up + 1;
}
virtual void free(const void* p) {
if (p) {
const uintptr_t* up = static_cast<const uintptr_t*>(p) - 1;
if (count < *up) {
abort();
}
count -= *up;
::free(const_cast<uintptr_t*>(up));
}
}
@ -71,15 +77,18 @@ class System: public vm::System {
};
const char*
append(vm::System* s, const char* a, const char* b, const char* c)
append(vm::System* s, const char* a, const char* b, const char* c,
const char* d)
{
unsigned al = strlen(a);
unsigned bl = strlen(b);
unsigned cl = strlen(c);
char* p = static_cast<char*>(s->allocate(al + bl + cl + 1));
unsigned dl = strlen(d);
char* p = static_cast<char*>(s->allocate(al + bl + cl + dl + 1));
memcpy(p, a, al);
memcpy(p + al, b, bl);
memcpy(p + al + bl, c, cl + 1);
memcpy(p + al + bl, c, cl);
memcpy(p + al + bl + cl, d, dl + 1);
return p;
}
@ -92,7 +101,8 @@ class ClassFinder: public vm::ClassFinder {
class Data: public vm::ClassFinder::Data {
public:
Data(uint8_t* start, size_t length):
Data(vm::System* system, uint8_t* start, size_t length):
system(system),
start_(start),
length_(length)
{ }
@ -109,18 +119,22 @@ class ClassFinder: public vm::ClassFinder {
if (start_) {
munmap(start_, length_);
}
system->free(this);
}
vm::System* system;
uint8_t* start_;
size_t length_;
};
virtual Data* find(const char* className) {
Data* d = new (system->allocate(sizeof(Data))) Data(0, 0);
Data* d = new (system->allocate(sizeof(Data))) Data(system, 0, 0);
for (const char** p = path; *p; ++p) {
const char* file = append(system, *p, "/", className);
const char* file = append(system, *p, "/", className, ".class");
int fd = open(file, O_RDONLY);
system->free(file);
if (fd != -1) {
struct stat s;
int r = fstat(fd, &s);
@ -133,7 +147,6 @@ class ClassFinder: public vm::ClassFinder {
}
}
}
system->free(file);
}
system->free(d);
@ -183,7 +196,7 @@ parsePath(vm::System* s, const char* path)
unsigned i = 0;
for (Tokenizer t(path, ':'); t.hasMore(); ++i) {
Tokenizer::Token token(t.next());
char* p = static_cast<char*>(s->allocate(token.length));
char* p = static_cast<char*>(s->allocate(token.length + 1));
memcpy(p, token.s, token.length);
p[token.length] = 0;
v[i] = p;

View File

@ -1492,19 +1492,14 @@ writeInitializations(Output* out, Object* declarations)
out->write("t->vm->types = allocate(t, pad((");
out->write(count);
out->write(" * ");
out->write(sizeof(void*));
out->write(") + 4 + ");
out->write(sizeof(void*));
out->write("));\n");
out->write(" * sizeof(void*)) + 4 + sizeof(void*)));\n");
out->write("objectClass(t->vm->types) = 0;\n");
out->write("arrayLength(t, t->vm->types) = ");
out->write(count);
out->write(";\n");
out->write("t->vm->types = makeArray(t, ");
out->write(typeCount(declarations));
out->write(");\n\n");
out->write("memset(&arrayBody(t, t->vm->types, 0), 0, ");
out->write(count);
out->write(" * sizeof(void*));\n\n");
for (Object* p = declarations; p; p = cdr(p)) {
Object* o = car(p);

View File

@ -16,6 +16,8 @@ using namespace vm;
namespace {
static const bool Debug = true;
class Thread;
void assert(Thread*, bool);
@ -291,6 +293,8 @@ collect(Machine* m, Heap::CollectionType type)
Machine* m;
} it(m);
fprintf(stderr, "collection time!\n");
m->heap->collect(type, &it);
}
@ -759,18 +763,19 @@ find(Thread* t, object class_, object table, object reference,
object n = referenceName(t, reference);
object s = referenceSpec(t, reference);
for (unsigned i = 0; i < arrayLength(t, table); ++i) {
object field = arrayBody(t, table, i);
if (strcmp(&byteArrayBody(t, name(t, field), 0),
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, field), 0),
strcmp(&byteArrayBody(t, spec(t, o), 0),
&byteArrayBody(t, s, 0)) == 0)
{
return field;
return o;
}
}
object message = makeString
(t, "%s (%s) not found in %s",
(t, "%s:%s not found in %s",
&byteArrayBody(t, n, 0),
&byteArrayBody(t, s, 0),
&byteArrayBody(t, className(t, class_), 0));
@ -966,6 +971,126 @@ listAppend(Thread* t, object list, object value)
set(t, listRear(t, list), p);
}
object
parsePool(Thread* t, Stream& s)
{
unsigned poolCount = s.read2() - 1;
object pool = makeArray(t, poolCount);
PROTECT(t, pool);
for (unsigned i = 0; i < poolCount; ++i) {
unsigned c = s.read1();
switch (c) {
case CONSTANT_Integer: {
object value = makeInt(t, s.read4());
set(t, arrayBody(t, pool, i), value);
} break;
case CONSTANT_Float: {
object value = makeFloat(t, s.readFloat());
set(t, arrayBody(t, pool, i), value);
} break;
case CONSTANT_Long: {
object value = makeLong(t, s.read8());
set(t, arrayBody(t, pool, i), value);
} break;
case CONSTANT_Double: {
object value = makeLong(t, s.readDouble());
set(t, arrayBody(t, pool, i), value);
} break;
case CONSTANT_Utf8: {
unsigned length = s.read2();
object value = makeByteArray(t, length + 1);
s.read(reinterpret_cast<uint8_t*>(&byteArrayBody(t, value, 0)), length);
byteArrayBody(t, value, length) = 0;
set(t, arrayBody(t, pool, i), value);
} break;
case CONSTANT_Class: {
object value = makeIntArray(t, 2);
intArrayBody(t, value, 0) = c;
intArrayBody(t, value, 1) = s.read2();
set(t, arrayBody(t, pool, i), value);
} break;
case CONSTANT_String: {
object value = makeIntArray(t, 2);
intArrayBody(t, value, 0) = c;
intArrayBody(t, value, 1) = s.read2();
set(t, arrayBody(t, pool, i), value);
} break;
case CONSTANT_NameAndType: {
object value = makeIntArray(t, 3);
intArrayBody(t, value, 0) = c;
intArrayBody(t, value, 1) = s.read2();
intArrayBody(t, value, 2) = s.read2();
set(t, arrayBody(t, pool, i), value);
} break;
case CONSTANT_Fieldref:
case CONSTANT_Methodref:
case CONSTANT_InterfaceMethodref: {
object value = makeIntArray(t, 3);
intArrayBody(t, value, 0) = c;
intArrayBody(t, value, 1) = s.read2();
intArrayBody(t, value, 2) = s.read2();
set(t, arrayBody(t, pool, i), value);
} break;
default: abort(t);
}
}
for (unsigned i = 0; i < poolCount; ++i) {
object o = arrayBody(t, pool, i);
if (objectClass(o) == arrayBody(t, t->vm->types, Machine::IntArrayType)) {
switch (intArrayBody(t, o, 0)) {
case CONSTANT_Class: {
set(t, arrayBody(t, pool, i),
arrayBody(t, pool, intArrayBody(t, o, 1) - 1));
} break;
case CONSTANT_String: {
object bytes = arrayBody(t, pool, intArrayBody(t, o, 1) - 1);
object value = makeString(t, bytes, 0, byteArrayLength(t, bytes), 0);
set(t, arrayBody(t, pool, i), value);
} break;
case CONSTANT_NameAndType: {
object name = arrayBody(t, pool, intArrayBody(t, o, 1) - 1);
object type = arrayBody(t, pool, intArrayBody(t, o, 2) - 1);
object value = makePair(t, name, type);
set(t, arrayBody(t, pool, i), value);
} break;
}
}
}
for (unsigned i = 0; i < poolCount; ++i) {
object o = arrayBody(t, pool, i);
if (objectClass(o) == arrayBody(t, t->vm->types, Machine::IntArrayType)) {
switch (intArrayBody(t, o, 0)) {
case CONSTANT_Fieldref:
case CONSTANT_Methodref:
case CONSTANT_InterfaceMethodref: {
object c = arrayBody(t, pool, intArrayBody(t, o, 1) - 1);
object nameAndType = arrayBody(t, pool, intArrayBody(t, o, 2) - 1);
object value = makeReference
(t, c, pairFirst(t, nameAndType), pairSecond(t, nameAndType));
set(t, arrayBody(t, pool, i), value);
} break;
}
}
}
return pool;
}
void
parseInterfaceTable(Thread* t, Stream& s, object class_, object pool)
{
@ -975,17 +1100,21 @@ parseInterfaceTable(Thread* t, Stream& s, object class_, object pool)
object map = makeHashMap(t, 0, 0);
PROTECT(t, map);
object superInterfaces = classInterfaceTable(t, classSuper(t, class_));
PROTECT(t, superInterfaces);
if (classSuper(t, class_)) {
object superInterfaces = classInterfaceTable(t, classSuper(t, class_));
if (superInterfaces) {
PROTECT(t, superInterfaces);
for (unsigned i = 0; i < arrayLength(t, superInterfaces); i += 2) {
object name = interfaceName(t, arrayBody(t, superInterfaces, i));
hashMapInsert(t, map, name, name, byteArrayHash);
for (unsigned i = 0; i < arrayLength(t, superInterfaces); i += 2) {
object name = interfaceName(t, arrayBody(t, superInterfaces, i));
hashMapInsert(t, map, name, name, byteArrayHash);
}
}
}
unsigned count = s.read2();
for (unsigned i = 0; i < count; ++i) {
object name = arrayBody(t, pool, s.read2());
object name = arrayBody(t, pool, s.read2() - 1);
hashMapInsert(t, map, name, name, byteArrayHash);
}
@ -1060,8 +1189,13 @@ parseFieldTable(Thread* t, Stream& s, object class_, object pool)
unsigned count = s.read2();
if (count) {
unsigned memberOffset
= classFixedSize(t, classSuper(t, class_)) * BytesPerWord;
fprintf(stderr, "%d fields\n", count);
unsigned memberOffset = BytesPerWord;
if (classSuper(t, class_)) {
memberOffset= classFixedSize(t, classSuper(t, class_)) * BytesPerWord;
}
unsigned staticOffset = 0;
object fieldTable = makeArray(t, count);
@ -1081,8 +1215,8 @@ parseFieldTable(Thread* t, Stream& s, object class_, object pool)
object value = makeField(t,
flags,
0, // offset
arrayBody(t, pool, name),
arrayBody(t, pool, spec),
arrayBody(t, pool, name - 1),
arrayBody(t, pool, spec - 1),
class_);
if (flags & ACC_STATIC) {
@ -1185,14 +1319,18 @@ parseMethodTable(Thread* t, Stream& s, object class_, object pool)
unsigned virtualCount = 0;
object superVirtualTable = classVirtualTable(t, classSuper(t, class_));
object superVirtualTable = 0;
PROTECT(t, superVirtualTable);
if (classSuper(t, class_)) {
superVirtualTable = classVirtualTable(t, classSuper(t, class_));
}
if (superVirtualTable) {
virtualCount = arrayLength(t, superVirtualTable);
for (unsigned i = 0; i < virtualCount; ++i) {
object method = arrayBody(t, superVirtualTable, i);
hashMapInsert(t, map, method, method, byteArrayHash);
hashMapInsert(t, map, method, method, methodHash);
}
}
@ -1212,7 +1350,7 @@ parseMethodTable(Thread* t, Stream& s, object class_, object pool)
object code = 0;
unsigned attributeCount = s.read2();
for (unsigned j = 0; j < attributeCount; ++j) {
object name = arrayBody(t, pool, s.read2());
object name = arrayBody(t, pool, s.read2() - 1);
unsigned length = s.read4();
if (strcmp(reinterpret_cast<const int8_t*>("Code"),
@ -1224,14 +1362,15 @@ parseMethodTable(Thread* t, Stream& s, object class_, object pool)
}
}
object value = makeMethod(t,
flags,
0, // offset
parameterCount(t, arrayBody(t, pool, spec)),
arrayBody(t, pool, name),
arrayBody(t, pool, spec),
class_,
code);
object value = makeMethod
(t,
flags,
0, // offset
parameterCount(t, arrayBody(t, pool, spec - 1)),
arrayBody(t, pool, name - 1),
arrayBody(t, pool, spec - 1),
class_,
code);
PROTECT(t, value);
if (flags & ACC_STATIC) {
@ -1265,18 +1404,18 @@ parseMethodTable(Thread* t, Stream& s, object class_, object pool)
object vtable = makeArray(t, virtualCount);
unsigned i = 0;
if (superVirtualTable) {
unsigned i = 0;
for (; i < arrayLength(t, superVirtualTable); ++i) {
object method = arrayBody(t, superVirtualTable, i);
method = hashMapFind(t, map, method, methodHash, methodEqual);
set(t, arrayBody(t, vtable, i), method);
}
}
for (object p = listFront(t, newVirtuals); p; p = pairSecond(t, p)) {
set(t, arrayBody(t, vtable, i++), pairFirst(t, p));
}
for (object p = listFront(t, newVirtuals); p; p = pairSecond(t, p)) {
set(t, arrayBody(t, vtable, i++), pairFirst(t, p));
}
set(t, classVirtualTable(t, class_), vtable);
@ -1284,17 +1423,19 @@ parseMethodTable(Thread* t, Stream& s, object class_, object pool)
// generate interface vtables
object itable = classInterfaceTable(t, class_);
PROTECT(t, itable);
if (itable) {
PROTECT(t, itable);
for (unsigned i = 0; i < arrayLength(t, itable); i += 2) {
object methodTable = interfaceMethodTable(t, arrayBody(t, itable, i));
object vtable = arrayBody(t, itable, i + 1);
for (unsigned j = 0; j < arrayLength(t, methodTable); ++j) {
object method = arrayBody(t, methodTable, j);
method = hashMapFind(t, map, method, methodHash, methodEqual);
set(t, arrayBody(t, vtable, j), method);
for (unsigned i = 0; i < arrayLength(t, itable); i += 2) {
object methodTable = interfaceMethodTable(t, arrayBody(t, itable, i));
object vtable = arrayBody(t, itable, i + 1);
for (unsigned j = 0; j < arrayLength(t, methodTable); ++j) {
object method = arrayBody(t, methodTable, j);
method = hashMapFind(t, map, method, methodHash, methodEqual);
set(t, arrayBody(t, vtable, j), method);
}
}
}
}
@ -1322,70 +1463,7 @@ parseClass(Thread* t, const uint8_t* data, unsigned size)
s.read2(); // minor version
s.read2(); // major version
unsigned poolCount = s.read2();
object pool = makeArray(t, poolCount);
PROTECT(t, pool);
for (unsigned i = 0; i < poolCount; ++i) {
switch (s.read1()) {
case CONSTANT_Class: {
set(t, arrayBody(t, pool, i), arrayBody(t, pool, s.read2()));
} break;
case CONSTANT_Fieldref:
case CONSTANT_Methodref:
case CONSTANT_InterfaceMethodref: {
object c = arrayBody(t, pool, s.read2());
object nameAndType = arrayBody(t, pool, s.read2());
object value = makeReference
(t, c, pairFirst(t, nameAndType), pairSecond(t, nameAndType));
set(t, arrayBody(t, pool, i), value);
} break;
case CONSTANT_String: {
object bytes = arrayBody(t, pool, s.read2());
object value = makeString(t, bytes, 0, byteArrayLength(t, bytes), 0);
set(t, arrayBody(t, pool, i), value);
} break;
case CONSTANT_Integer: {
object value = makeInt(t, s.read4());
set(t, arrayBody(t, pool, i), value);
} break;
case CONSTANT_Float: {
object value = makeFloat(t, s.readFloat());
set(t, arrayBody(t, pool, i), value);
} break;
case CONSTANT_Long: {
object value = makeLong(t, s.read8());
set(t, arrayBody(t, pool, i), value);
} break;
case CONSTANT_Double: {
object value = makeLong(t, s.readDouble());
set(t, arrayBody(t, pool, i), value);
} break;
case CONSTANT_NameAndType: {
object name = arrayBody(t, pool, s.read2());
object type = arrayBody(t, pool, s.read2());
object value = makePair(t, name, type);
set(t, arrayBody(t, pool, i), value);
} break;
case CONSTANT_Utf8: {
unsigned length = s.read2();
object value = makeByteArray(t, length + 1);
s.read(reinterpret_cast<uint8_t*>(&byteArrayBody(t, value, 0)), length);
byteArrayBody(t, value, length) = 0;
set(t, arrayBody(t, pool, i), value);
} break;
default: abort(t);
}
}
object pool = parsePool(t, s);
unsigned flags = s.read2();
unsigned name = s.read2();
@ -1395,7 +1473,7 @@ parseClass(Thread* t, const uint8_t* data, unsigned size)
0, // fixed size
0, // array size
0, // object mask
arrayBody(t, pool, name),
arrayBody(t, pool, name - 1),
0, // super
0, // interfaces
0, // vtable
@ -1405,10 +1483,13 @@ parseClass(Thread* t, const uint8_t* data, unsigned size)
0); // initializer
PROTECT(t, class_);
object super = resolveClass(t, arrayBody(t, pool, s.read2()));
if (UNLIKELY(t->exception)) return 0;
unsigned super = s.read2();
if (super) {
object sc = resolveClass(t, arrayBody(t, pool, super - 1));
if (UNLIKELY(t->exception)) return 0;
set(t, classSuper(t, class_), super);
set(t, classSuper(t, class_), sc);
}
parseInterfaceTable(t, s, class_, pool);
if (UNLIKELY(t->exception)) return 0;
@ -1435,10 +1516,16 @@ resolveClass(Thread* t, object spec)
(reinterpret_cast<const char*>(&byteArrayBody(t, spec, 0)));
if (data) {
fprintf(stderr, "parsing %s\n", &byteArrayBody
(t, spec, 0));
// parse class file
class_ = parseClass(t, data->start(), data->length());
data->dispose();
fprintf(stderr, "done parsing %s\n", &byteArrayBody
(t, className(t, class_), 0));
PROTECT(t, class_);
hashMapInsert(t, t->vm->classMap, spec, class_, byteArrayHash);
@ -1614,7 +1701,7 @@ run(Thread* t)
uint8_t index2 = codeBody(t, code, ip++);
uint16_t index = (index1 << 8) | index2;
object class_ = resolveClass(t, codePool(t, code), index);
object class_ = resolveClass(t, codePool(t, code), index - 1);
if (UNLIKELY(exception)) goto throw_;
object array = makeObjectArray(t, class_, c);
@ -1804,7 +1891,7 @@ run(Thread* t)
if (stack[sp - 1]) {
uint16_t index = (index1 << 8) | index2;
object class_ = resolveClass(t, codePool(t, code), index);
object class_ = resolveClass(t, codePool(t, code), index - 1);
if (UNLIKELY(exception)) goto throw_;
if (not instanceOf(t, class_, stack[sp - 1])) {
@ -1915,7 +2002,7 @@ run(Thread* t)
uint8_t index2 = codeBody(t, code, ip++);
uint16_t index = (index1 << 8) | index2;
object field = resolveField(t, codePool(t, code), index);
object field = resolveField(t, codePool(t, code), index - 1);
if (UNLIKELY(exception)) goto throw_;
push(t, getField(t, instance, field));
@ -1930,7 +2017,7 @@ run(Thread* t)
uint8_t index2 = codeBody(t, code, ip++);
uint16_t index = (index1 << 8) | index2;
object field = resolveField(t, codePool(t, code), index);
object field = resolveField(t, codePool(t, code), index - 1);
if (UNLIKELY(exception)) goto throw_;
object clinit = classInitializer(t, fieldClass(t, field));
@ -2288,7 +2375,7 @@ run(Thread* t)
if (stack[sp - 1]) {
uint16_t index = (index1 << 8) | index2;
object class_ = resolveClass(t, codePool(t, code), index);
object class_ = resolveClass(t, codePool(t, code), index - 1);
if (UNLIKELY(exception)) goto throw_;
if (instanceOf(t, class_, stack[sp - 1])) {
@ -2308,7 +2395,7 @@ run(Thread* t)
ip += 2;
object method = resolveMethod(t, codePool(t, code), index);
object method = resolveMethod(t, codePool(t, code), index - 1);
if (UNLIKELY(exception)) goto throw_;
parameterCount = methodParameterCount(t, method);
@ -2328,7 +2415,7 @@ run(Thread* t)
uint8_t index2 = codeBody(t, code, ip++);
uint16_t index = (index1 << 8) | index2;
object method = resolveMethod(t, codePool(t, code), index);
object method = resolveMethod(t, codePool(t, code), index - 1);
if (UNLIKELY(exception)) goto throw_;
parameterCount = methodParameterCount(t, method);
@ -2353,7 +2440,7 @@ run(Thread* t)
uint8_t index2 = codeBody(t, code, ip++);
uint16_t index = (index1 << 8) | index2;
object method = resolveMethod(t, codePool(t, code), index);
object method = resolveMethod(t, codePool(t, code), index - 1);
if (UNLIKELY(exception)) goto throw_;
object clinit = classInitializer(t, methodClass(t, method));
@ -2374,7 +2461,7 @@ run(Thread* t)
uint8_t index2 = codeBody(t, code, ip++);
uint16_t index = (index1 << 8) | index2;
object method = resolveMethod(t, codePool(t, code), index);
object method = resolveMethod(t, codePool(t, code), index - 1);
if (UNLIKELY(exception)) goto throw_;
parameterCount = methodParameterCount(t, method);
@ -2539,7 +2626,7 @@ run(Thread* t)
} goto loop;
case ldc: {
push(t, arrayBody(t, codePool(t, code), codeBody(t, code, ip++)));
push(t, arrayBody(t, codePool(t, code), codeBody(t, code, ip++) - 1));
} goto loop;
case ldc_w:
@ -2547,7 +2634,7 @@ run(Thread* t)
uint8_t index1 = codeBody(t, code, ip++);
uint8_t index2 = codeBody(t, code, ip++);
push(t, arrayBody(t, codePool(t, code), (index1 << 8) | index2));
push(t, arrayBody(t, codePool(t, code), ((index1 << 8) | index2) - 1));
} goto loop;
case vm::ldiv: {
@ -2625,7 +2712,7 @@ run(Thread* t)
uint8_t index2 = codeBody(t, code, ip++);
uint16_t index = (index1 << 8) | index2;
object class_ = resolveClass(t, codePool(t, code), index);
object class_ = resolveClass(t, codePool(t, code), index - 1);
if (UNLIKELY(exception)) goto throw_;
object clinit = classInitializer(t, class_);
@ -2727,7 +2814,7 @@ run(Thread* t)
uint8_t index2 = codeBody(t, code, ip++);
uint16_t index = (index1 << 8) | index2;
object field = resolveField(t, codePool(t, code), index);
object field = resolveField(t, codePool(t, code), index - 1);
if (UNLIKELY(exception)) goto throw_;
object value = pop(t);
@ -2743,7 +2830,7 @@ run(Thread* t)
uint8_t index2 = codeBody(t, code, ip++);
uint16_t index = (index1 << 8) | index2;
object field = resolveField(t, codePool(t, code), index);
object field = resolveField(t, codePool(t, code), index - 1);
if (UNLIKELY(exception)) goto throw_;
object clinit = classInitializer(t, fieldClass(t, field));
@ -2906,11 +2993,13 @@ run(Thread* t)
if (eht) {
for (unsigned i = 0; i < exceptionHandlerTableLength(t, eht); ++i) {
ExceptionHandler* eh = exceptionHandlerTableBody(t, eht, i);
uint16_t catchType = exceptionHandlerCatchType(eh);
object catchType =
arrayBody(t, codePool(t, code), exceptionHandlerCatchType(eh) - 1);
if (catchType == 0 or
instanceOf(t,
arrayBody(t, codePool(t, code), catchType),
exception))
(objectClass(catchType)
== arrayBody(t, t->vm->types, Machine::ClassType) and
instanceOf(t, catchType, exception)))
{
sp = frameStackBase(t, frame);
ip = exceptionHandlerIp(eh);
@ -2922,14 +3011,34 @@ run(Thread* t)
}
}
object method = threadExceptionHandler(t, t->thread);
code = methodCode(t, method);
frame = makeFrame(t, method, 0, 0, 0, codeMaxLocals(t, code));
sp = 0;
ip = 0;
push(t, exception);
exception = 0;
goto loop;
if (t->thread) {
object method = threadExceptionHandler(t, t->thread);
code = methodCode(t, method);
frame = makeFrame(t, method, 0, 0, 0, codeMaxLocals(t, code));
sp = 0;
ip = 0;
push(t, exception);
exception = 0;
goto loop;
} else if (Debug) {
object p = 0;
object n = 0;
for (object trace = throwableTrace(t, exception); trace; trace = n) {
n = traceNext(t, trace);
set(t, traceNext(t, trace), p);
p = trace;
}
fprintf(stderr, "uncaught exception: %s\n", &byteArrayBody
(t, stringBytes(t, throwableMessage(t, exception)), 0));
for (; p; p = traceNext(t, p)) {
fprintf(stderr, " at %s\n", &byteArrayBody
(t, methodName(t, traceMethod(t, p)), 0));
}
}
abort(t);
}
void