Merge branch 'master' of dice:git/vm

Conflicts:

	src/run.cpp
This commit is contained in:
Joel Dice 2007-07-24 08:42:44 -06:00
commit f56fda9af6
15 changed files with 694 additions and 182 deletions

View File

@ -1,5 +1,8 @@
package java.lang; package java.lang;
import java.lang.reflect.Method;
import java.lang.reflect.Field;
public final class Class <T> { public final class Class <T> {
private short flags; private short flags;
private byte vmFlags; private byte vmFlags;
@ -9,16 +12,57 @@ public final class Class <T> {
private int[] objectMask; private int[] objectMask;
private byte[] name; private byte[] name;
private Class super_; private Class super_;
private Object interfaceTable; private Object[] interfaceTable;
private Object virtualTable; private Method[] virtualTable;
private Object fieldTable; private Field[] fieldTable;
private Object methodTable; private Method[] methodTable;
private Object staticTable; private Object[] staticTable;
private Object initializer; private Method initializer;
private Class() { } private Class() { }
public String getName() { public String getName() {
return new String(name, 0, name.length - 1, false); return new String(name, 0, name.length - 1, false);
} }
public static native Class forName(String name);
public native boolean isAssignableFrom(Class c);
public Field getDeclaredField(String name) throws NoSuchFieldException {
for (int i = 0; i < fieldTable.length; ++i) {
if (fieldTable[i].getName().equals(name)) {
return fieldTable[i];
}
}
throw new NoSuchFieldException(name);
}
private static boolean match(Class[] a, Class[] b) {
if (a.length == b.length) {
for (int i = 0; i < a.length; ++i) {
if (! a[i].isAssignableFrom(b[i])) {
return false;
}
}
return true;
} else {
return false;
}
}
public Method getDeclaredMethod(String name, Class ... parameterTypes)
throws NoSuchMethodException
{
for (int i = 0; i < methodTable.length; ++i) {
if (methodTable[i].getName().equals(name)
&& match(parameterTypes, methodTable[i].getParameterTypes()))
{
return methodTable[i];
}
}
throw new NoSuchMethodException(name);
}
} }

View File

@ -0,0 +1,19 @@
package java.lang;
public class NullPointerException extends RuntimeException {
public NullPointerException(String message, Throwable cause) {
super(message, cause);
}
public NullPointerException(String message) {
this(message, null);
}
public NullPointerException(Throwable cause) {
this(null, cause);
}
public NullPointerException() {
this(null, null);
}
}

View File

@ -101,7 +101,8 @@ public final class String implements Comparable<String> {
return new String(data, offset + start, end - start, false); return new String(data, offset + start, end - start, false);
} }
} else { } else {
throw new IndexOutOfBoundsException(); throw new IndexOutOfBoundsException
(start + " not in (0, " + end + ") or " + end + " > " + length);
} }
} }

View File

@ -30,4 +30,6 @@ public class Field<T> extends AccessibleObject {
public String getName() { public String getName() {
return new String(name, 0, name.length - 1, false); return new String(name, 0, name.length - 1, false);
} }
public native Object get(Object instance);
} }

View File

@ -32,4 +32,54 @@ public class Method<T> extends AccessibleObject implements Member {
public String getName() { public String getName() {
return new String(name, 0, name.length - 1, false); return new String(name, 0, name.length - 1, false);
} }
private static int next(char c, String s, int start) {
for (int i = start; i < s.length(); ++i) {
if (s.charAt(i) == c) return i;
}
throw new RuntimeException();
}
public Class[] getParameterTypes() {
int count = parameterCount;
if ((flags & Modifier.STATIC) == 0) {
-- count;
}
Class[] types = new Class[count];
int index = 0;
String spec = new String(this.spec, 1, this.spec.length - 1, false);
for (int i = 0; i < spec.length(); ++i) {
char c = spec.charAt(i);
if (c == ')') {
break;
} else if (c == 'L') {
int start = i + 1;
i = next(';', spec, start);
String name = spec.substring(start, i);
types[index++] = Class.forName(name);
} else if (c == '[') {
int start = i;
while (spec.charAt(i) == '[') ++i;
if (spec.charAt(i) == 'L') {
i = next(';', spec, i + 1);
String name = spec.substring(start, i);
types[index++] = Class.forName(name);
} else {
String name = spec.substring(start, i + 1);
types[index++] = Class.forName(name);
}
} else {
String name = spec.substring(i, i + 1);
types[index++] = Class.forName(name);
}
}
return types;
}
public native Object invoke(Object instance, Object ... arguments);
} }

View File

@ -16,7 +16,7 @@ src = src
classpath = classpath classpath = classpath
test = test test = test
input = $(cls)/References.class input = $(cls)/Threads.class
cxx = g++ cxx = g++
cc = gcc cc = gcc

View File

@ -1,7 +1,30 @@
#include "builtin.h" #include "builtin.h"
#include "machine.h" #include "machine.h"
#include "constants.h"
#include "run.h" #include "run.h"
using namespace vm;
namespace {
object
doInvoke(Thread* t, object this_, object instance, object arguments)
{
object v = pushReference(t, run2(t, this_, instance, arguments));
if (t->exception) {
t->exception = makeInvocationTargetException(t, t->exception);
}
return v;
}
inline void
replace(char a, char b, char* c)
{
for (; *c; ++c) if (*c == a) *c = b;
}
} // namespace
namespace vm { namespace vm {
namespace builtin { namespace builtin {
@ -41,6 +64,148 @@ notifyAll(Thread* t, jobject this_)
vm::notifyAll(t, *this_); vm::notifyAll(t, *this_);
} }
jclass
forName(Thread* t, jstring name)
{
if (LIKELY(name)) {
object n = makeByteArray(t, stringLength(t, *name) + 1, false);
char* s = reinterpret_cast<char*>(&byteArrayBody(t, n, 0));
stringChars(t, *name, s);
replace('.', '/', s);
object c = resolveClass(t, n);
if (t->exception) {
return 0;
}
object clinit = classInitializer(t, c);
if (clinit) {
PROTECT(t, c);
set(t, classInitializer(t, c), 0);
run(t, clinit, 0);
}
return pushReference(t, c);
} else {
t->exception = makeNullPointerException(t);
return 0;
}
}
jboolean
isAssignableFrom(Thread* t, jobject this_, jclass that)
{
if (LIKELY(that)) {
return vm::isAssignableFrom(t, *this_, *that);
} else {
t->exception = makeNullPointerException(t);
return 0;
}
}
jobject
get(Thread* t, jobject this_, jobject instancep)
{
object field = *this_;
if (fieldFlags(t, field) & ACC_STATIC) {
return pushReference
(t, arrayBody(t, classStaticTable(t, fieldClass(t, field)),
fieldOffset(t, field)));
} else if (instancep) {
object instance = *instancep;
if (instanceOf(t, fieldClass(t, this_), instance)) {
switch (fieldCode(t, field)) {
case ByteField:
return pushReference
(t, makeByte(t, cast<int8_t>(instance, fieldOffset(t, field))));
case BooleanField:
return pushReference
(t, makeBoolean(t, cast<uint8_t>(instance, fieldOffset(t, field))));
case CharField:
return pushReference
(t, makeChar(t, cast<uint16_t>(instance, fieldOffset(t, field))));
case ShortField:
return pushReference
(t, makeShort(t, cast<int16_t>(instance, fieldOffset(t, field))));
case FloatField:
return pushReference
(t, makeFloat(t, cast<uint32_t>(instance, fieldOffset(t, field))));
case IntField:
return pushReference
(t, makeInt(t, cast<int32_t>(instance, fieldOffset(t, field))));
case DoubleField:
return pushReference
(t, makeDouble(t, cast<uint64_t>(instance, fieldOffset(t, field))));
case LongField:
return pushReference
(t, makeLong(t, cast<int64_t>(instance, fieldOffset(t, field))));
case ObjectField:
return pushReference
(t, cast<object>(instance, fieldOffset(t, field)));
default:
abort(t);
}
} else {
t->exception = makeIllegalArgumentException(t);
return 0;
}
} else {
t->exception = makeNullPointerException(t);
return 0;
}
}
jobject
invoke(Thread* t, jobject this_, jobject instancep, jobjectArray argumentsp)
{
object method = *this_;
if (argumentsp) {
object arguments = *argumentsp;
if (methodFlags(t, method) & ACC_STATIC) {
if (objectArrayLength(t, arguments)
== methodParameterCount(t, method))
{
return pushReference(t, doInvoke(t, method, 0, arguments));
} else {
t->exception = makeArrayIndexOutOfBoundsException(t, 0);
}
} else if (instancep) {
object instance = *instancep;
if (instanceOf(t, methodClass(t, method), instance)) {
if (objectArrayLength(t, arguments)
== static_cast<unsigned>(methodParameterCount(t, method) - 1))
{
return pushReference(t, doInvoke(t, method, instance, arguments));
} else {
t->exception = makeArrayIndexOutOfBoundsException(t, 0);
}
}
} else {
t->exception = makeNullPointerException(t);
}
} else {
t->exception = makeNullPointerException(t);
}
return 0;
}
jobject jobject
currentThread(Thread* t) currentThread(Thread* t)
{ {
@ -104,15 +269,14 @@ currentTimeMillis(Thread* t)
} }
void void
loadLibrary(Thread* t, jobject, jstring nameString) loadLibrary(Thread* t, jobject, jstring name)
{ {
if (LIKELY(nameString)) { if (LIKELY(name)) {
object n = *nameString; char n[stringLength(t, *name) + 1];
char name[stringLength(t, n) + 1]; stringChars(t, *name, n);
stringChars(t, n, name);
for (System::Library* lib = t->vm->libraries; lib; lib = lib->next()) { for (System::Library* lib = t->vm->libraries; lib; lib = lib->next()) {
if (::strcmp(lib->name(), name) == 0) { if (::strcmp(lib->name(), n) == 0) {
// already loaded // already loaded
return; return;
} }
@ -120,11 +284,11 @@ loadLibrary(Thread* t, jobject, jstring nameString)
System::Library* lib; System::Library* lib;
if (LIKELY(t->vm->system->success if (LIKELY(t->vm->system->success
(t->vm->system->load(&lib, name, t->vm->libraries)))) (t->vm->system->load(&lib, n, t->vm->libraries))))
{ {
t->vm->libraries = lib; t->vm->libraries = lib;
} else { } else {
object message = makeString(t, "library not found: %s", name); object message = makeString(t, "library not found: %s", n);
t->exception = makeRuntimeException(t, message); t->exception = makeRuntimeException(t, message);
} }
} else { } else {
@ -224,6 +388,10 @@ start(Thread* t, jobject this_)
vm::run(t, "java/lang/Thread", "run", "()V", t->javaThread); vm::run(t, "java/lang/Thread", "run", "()V", t->javaThread);
if (t->exception) {
printTrace(t, t->exception);
}
t->exit(); t->exit();
} }
@ -252,6 +420,11 @@ populate(Thread* t, object map)
const char* key; const char* key;
void* value; void* value;
} builtins[] = { } builtins[] = {
{ "Java_java_lang_Class_forName",
reinterpret_cast<void*>(forName) },
{ "Java_java_lang_Class_isAssignableFrom",
reinterpret_cast<void*>(isAssignableFrom) },
{ "Java_java_lang_System_arraycopy", { "Java_java_lang_System_arraycopy",
reinterpret_cast<void*>(arraycopy) }, reinterpret_cast<void*>(arraycopy) },
@ -285,6 +458,12 @@ populate(Thread* t, object map)
{ "Java_java_lang_Object_wait", { "Java_java_lang_Object_wait",
reinterpret_cast<void*>(wait) }, reinterpret_cast<void*>(wait) },
{ "Java_java_lang_reflect_Field_get",
reinterpret_cast<void*>(get) },
{ "Java_java_lang_reflect_Method_invoke",
reinterpret_cast<void*>(invoke) },
{ 0, 0 } { 0, 0 }
}; };

View File

@ -99,7 +99,7 @@ enum OpCode {
i2s = 0x93, i2s = 0x93,
iadd = 0x60, iadd = 0x60,
iaload = 0x2e, iaload = 0x2e,
iand = 0x73, iand = 0x7e,
iastore = 0x4f, iastore = 0x4f,
iconst_m1 = 0x02, iconst_m1 = 0x02,
iconst_0 = 0x03, iconst_0 = 0x03,
@ -113,9 +113,9 @@ enum OpCode {
if_acmpne = 0xa6, if_acmpne = 0xa6,
if_icmpeq = 0x9f, if_icmpeq = 0x9f,
if_icmpne = 0xa0, if_icmpne = 0xa0,
if_icmpgt = 0xa1, if_icmplt = 0xa1,
if_icmpge = 0xa2, if_icmpge = 0xa2,
if_icmplt = 0xa3, if_icmpgt = 0xa3,
if_icmple = 0xa4, if_icmple = 0xa4,
ifeq = 0x99, ifeq = 0x99,
ifge = 0x9c, ifge = 0x9c,

View File

@ -1376,7 +1376,7 @@ Thread::Thread(Machine* m, object javaThread, Thread* parent):
builtin::populate(t, m->builtinMap); builtin::populate(t, m->builtinMap);
javaThread = makeThread(t, 0, 0, reinterpret_cast<int64_t>(t)); t->javaThread = makeThread(t, 0, 0, reinterpret_cast<int64_t>(t));
} else { } else {
threadPeer(this, javaThread) = reinterpret_cast<jlong>(this); threadPeer(this, javaThread) = reinterpret_cast<jlong>(this);
parent->child = this; parent->child = this;
@ -1625,6 +1625,39 @@ stringChars(Thread* t, object string, char* chars)
chars[stringLength(t, string)] = 0; chars[stringLength(t, string)] = 0;
} }
bool
isAssignableFrom(Thread* t, object a, object b)
{
if (classFlags(t, a) & ACC_INTERFACE) {
for (; b; b = classSuper(t, b)) {
object itable = classInterfaceTable(t, b);
for (unsigned i = 0; i < arrayLength(t, itable); i += 2) {
if (arrayBody(t, itable, i) == a) {
return true;
}
}
}
} else {
for (; b; b = classSuper(t, b)) {
if (b == a) {
return true;
}
}
}
return false;
}
bool
instanceOf(Thread* t, object class_, object o)
{
if (o == 0) {
return false;
} else {
return isAssignableFrom(t, class_, objectClass(t, o));
}
}
unsigned unsigned
parameterFootprint(const char* s) parameterFootprint(const char* s)
{ {
@ -2015,17 +2048,16 @@ resolveClass(Thread* t, object spec)
if (data) { if (data) {
if (Verbose) { if (Verbose) {
fprintf(stderr, "parsing %s\n", &byteArrayBody fprintf(stderr, "parsing %s\n", &byteArrayBody(t, spec, 0));
(t, spec, 0));
} }
// parse class file // parse class file
class_ = parseClass(t, data->start(), data->length()); class_ = parseClass(t, data->start(), data->length());
data->dispose(); data->dispose();
if (LIKELY(t->exception == 0)) {
if (Verbose) { if (Verbose) {
fprintf(stderr, "done parsing %s\n", &byteArrayBody fprintf(stderr, "done parsing %s\n", &byteArrayBody(t, spec, 0));
(t, className(t, class_), 0));
} }
object bootstrapClass = hashMapFind object bootstrapClass = hashMapFind
@ -2039,6 +2071,7 @@ resolveClass(Thread* t, object spec)
} }
} }
} }
}
if (class_) { if (class_) {
PROTECT(t, class_); PROTECT(t, class_);
@ -2313,6 +2346,51 @@ collect(Thread* t, Heap::CollectionType type)
killZombies(t, m->rootThread); killZombies(t, m->rootThread);
} }
void
printTrace(Thread* t, object exception)
{
for (object e = exception; e; e = throwableCauseUnsafe(t, e)) {
if (e != exception) {
fprintf(stderr, "caused by: ");
}
fprintf(stderr, "%s", &byteArrayBody
(t, className(t, objectClass(t, e)), 0));
if (throwableMessageUnsafe(t, e)) {
object m = throwableMessageUnsafe(t, e);
char message[stringLength(t, m) + 1];
stringChars(t, m, message);
fprintf(stderr, ": %s\n", message);
} else {
fprintf(stderr, "\n");
}
object trace = throwableTraceUnsafe(t, e);
for (unsigned i = 0; i < arrayLength(t, trace); ++i) {
object e = arrayBody(t, trace, i);
const int8_t* class_ = &byteArrayBody
(t, className(t, methodClass(t, traceElementMethod(t, e))), 0);
const int8_t* method = &byteArrayBody
(t, methodName(t, traceElementMethod(t, e)), 0);
int line = lineNumber(t, traceElementMethod(t, e), traceElementIp(t, e));
fprintf(stderr, " at %s.%s ", class_, method);
switch (line) {
case NativeLine:
fprintf(stderr, "(native)\n");
break;
case UnknownLine:
fprintf(stderr, "(unknown line)\n");
break;
default:
fprintf(stderr, "(line %d)\n", line);
}
}
}
}
void void
noop() noop()
{ } { }

View File

@ -1377,6 +1377,12 @@ makeIllegalStateException(Thread* t, object message)
return makeIllegalStateException(t, message, trace, 0); return makeIllegalStateException(t, message, trace, 0);
} }
inline object
makeIllegalArgumentException(Thread* t)
{
return makeIllegalArgumentException(t, 0, makeTrace(t), 0);
}
inline object inline object
makeIllegalMonitorStateException(Thread* t) makeIllegalMonitorStateException(Thread* t)
{ {
@ -1427,6 +1433,14 @@ makeNullPointerException(Thread* t)
return makeNullPointerException(t, 0, makeTrace(t), 0); return makeNullPointerException(t, 0, makeTrace(t), 0);
} }
inline object
makeInvocationTargetException(Thread* t, object targetException)
{
PROTECT(t, targetException);
object trace = makeTrace(t);
return makeRuntimeException(t, 0, trace, targetException);
}
inline object inline object
makeStackOverflowError(Thread* t) makeStackOverflowError(Thread* t)
{ {
@ -1466,6 +1480,12 @@ makeString(Thread* t, const char* format, ...);
void void
stringChars(Thread* t, object string, char* chars); stringChars(Thread* t, object string, char* chars);
bool
isAssignableFrom(Thread* t, object a, object b);
bool
instanceOf(Thread* t, object class_, object o);
inline void inline void
pushObject(Thread* t, object o) pushObject(Thread* t, object o)
{ {
@ -1622,9 +1642,13 @@ pokeLong(Thread* t, unsigned index, uint64_t value)
inline object* inline object*
pushReference(Thread* t, object o) pushReference(Thread* t, object o)
{ {
if (o) {
expect(t, t->sp + 1 < Thread::StackSizeInWords / 2); expect(t, t->sp + 1 < Thread::StackSizeInWords / 2);
pushObject(t, o); pushObject(t, o);
return reinterpret_cast<object*>(t->stack + ((t->sp - 1) * 2) + 1); return reinterpret_cast<object*>(t->stack + ((t->sp - 1) * 2) + 1);
} else {
return 0;
}
} }
inline int inline int
@ -2055,6 +2079,9 @@ vmNotifyAll(Thread* t, object o)
notifyAll(t, o); notifyAll(t, o);
} }
void
printTrace(Thread* t, object exception);
void void
exit(Thread* t); exit(Thread* t);

View File

@ -91,8 +91,7 @@ make(Thread* t, object class_)
ACQUIRE(t, t->vm->referenceLock); ACQUIRE(t, t->vm->referenceLock);
// jreferenceNext(t, instance) jreferenceNextUnsafe(t, instance) = t->vm->weakReferences;
cast<object>(instance, BytesPerWord) = t->vm->weakReferences;
t->vm->weakReferences = instance; t->vm->weakReferences = instance;
} }
@ -106,33 +105,6 @@ setStatic(Thread* t, object field, object value)
fieldOffset(t, field)), value); fieldOffset(t, field)), value);
} }
bool
instanceOf(Thread* t, object class_, object o)
{
if (o == 0) {
return false;
}
if (classFlags(t, class_) & ACC_INTERFACE) {
for (object oc = objectClass(t, o); oc; oc = classSuper(t, oc)) {
object itable = classInterfaceTable(t, oc);
for (unsigned i = 0; i < arrayLength(t, itable); i += 2) {
if (arrayBody(t, itable, i) == class_) {
return true;
}
}
}
} else {
for (object oc = objectClass(t, o); oc; oc = classSuper(t, oc)) {
if (oc == class_) {
return true;
}
}
}
return false;
}
object object
findInterfaceMethod(Thread* t, object method, object o) findInterfaceMethod(Thread* t, object method, object o)
{ {
@ -328,6 +300,16 @@ makeNativeMethodData(Thread* t, object method, void* function, bool builtin)
case '[': case '[':
argumentTableSize += BytesPerWord; argumentTableSize += BytesPerWord;
while (*s == '[') ++ s; while (*s == '[') ++ s;
switch (*s) {
case 'L':
while (*s and *s != ';') ++ s;
++ s;
break;
default:
++ s;
break;
}
break; break;
default: default:
@ -535,6 +517,8 @@ invokeNative(Thread* t, object method)
object object
run(Thread* t) run(Thread* t)
{ {
const int base = t->frame;
unsigned instruction = nop; unsigned instruction = nop;
unsigned& ip = t->ip; unsigned& ip = t->ip;
unsigned& sp = t->sp; unsigned& sp = t->sp;
@ -659,8 +643,8 @@ run(Thread* t)
case areturn: { case areturn: {
object result = popObject(t); object result = popObject(t);
if (frame > base) {
popFrame(t); popFrame(t);
if (frame >= 0) {
pushObject(t, result); pushObject(t, result);
goto loop; goto loop;
} else { } else {
@ -1451,8 +1435,8 @@ run(Thread* t)
case ireturn: { case ireturn: {
int32_t result = popInt(t); int32_t result = popInt(t);
if (frame > base) {
popFrame(t); popFrame(t);
if (frame >= 0) {
pushInt(t, result); pushInt(t, result);
goto loop; goto loop;
} else { } else {
@ -1626,16 +1610,16 @@ run(Thread* t)
if (objectClass(t, v) == arrayBody(t, t->vm->types, Machine::IntType)) { if (objectClass(t, v) == arrayBody(t, t->vm->types, Machine::IntType)) {
pushInt(t, intValue(t, v)); pushInt(t, intValue(t, v));
} else if (objectClass(t, v)
== arrayBody(t, t->vm->types, Machine::StringType))
{
pushObject(t, v);
} else if (objectClass(t, v) } else if (objectClass(t, v)
== arrayBody(t, t->vm->types, Machine::FloatType)) == arrayBody(t, t->vm->types, Machine::FloatType))
{ {
pushInt(t, floatValue(t, v)); pushInt(t, floatValue(t, v));
} else if (objectClass(t, v)
== arrayBody(t, t->vm->types, Machine::StringType))
{
pushObject(t, v);
} else { } else {
abort(t); pushObject(t, resolveClass(t, v));
} }
} goto loop; } goto loop;
@ -1710,8 +1694,8 @@ run(Thread* t)
case lreturn: { case lreturn: {
int64_t result = popLong(t); int64_t result = popLong(t);
if (frame > base) {
popFrame(t); popFrame(t);
if (frame >= 0) {
pushLong(t, result); pushLong(t, result);
goto loop; goto loop;
} else { } else {
@ -1993,8 +1977,8 @@ run(Thread* t)
} goto loop; } goto loop;
case return_: { case return_: {
if (frame > base) {
popFrame(t); popFrame(t);
if (frame >= 0) {
goto loop; goto loop;
} else { } else {
return 0; return 0;
@ -2148,7 +2132,7 @@ run(Thread* t)
pokeInt(t, t->frame + FrameIpOffset, t->ip); pokeInt(t, t->frame + FrameIpOffset, t->ip);
for (; frame >= 0; frame = frameNext(t, frame)) { for (; frame >= 0; frame = frameNext(t, frame)) {
if (methodFlags(t, frameMethod(t, frame)) & ACC_NATIVE) { if (frame < base) {
return 0; return 0;
} }
@ -2185,49 +2169,6 @@ run(Thread* t)
} }
} }
for (object e = exception; e; e = throwableCause(t, e)) {
if (e == exception) {
fprintf(stderr, "uncaught exception: ");
} else {
fprintf(stderr, "caused by: ");
}
fprintf(stderr, "%s", &byteArrayBody
(t, className(t, objectClass(t, exception)), 0));
if (throwableMessage(t, exception)) {
object m = throwableMessage(t, exception);
char message[stringLength(t, m) + 1];
stringChars(t, m, message);
fprintf(stderr, ": %s\n", message);
} else {
fprintf(stderr, "\n");
}
object trace = throwableTrace(t, e);
for (unsigned i = 0; i < arrayLength(t, trace); ++i) {
object e = arrayBody(t, trace, i);
const int8_t* class_ = &byteArrayBody
(t, className(t, methodClass(t, traceElementMethod(t, e))), 0);
const int8_t* method = &byteArrayBody
(t, methodName(t, traceElementMethod(t, e)), 0);
int line = lineNumber(t, traceElementMethod(t, e), traceElementIp(t, e));
fprintf(stderr, " at %s.%s ", class_, method);
switch (line) {
case NativeLine:
fprintf(stderr, "(native)\n");
break;
case UnknownLine:
fprintf(stderr, "(unknown line)\n");
break;
default:
fprintf(stderr, "(line %d)\n", line);
}
}
}
return 0; return 0;
} }
@ -2249,32 +2190,14 @@ run(Thread* t, const char* className, int argc, const char** argv)
run(t, className, "main", "([Ljava/lang/String;)V", 0, args); run(t, className, "main", "([Ljava/lang/String;)V", 0, args);
} }
} // namespace void
pushArguments(Thread* t, object this_, const char* spec, va_list a)
namespace vm {
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;
}
if (this_) { if (this_) {
pushObject(t, this_); pushObject(t, this_);
} }
va_list a; const char* s = spec;
va_start(a, this_);
const char* s = methodSpec;
++ s; // skip '(' ++ s; // skip '('
while (*s and *s != ')') { while (*s and *s != ')') {
switch (*s) { switch (*s) {
@ -2311,22 +2234,59 @@ run(Thread* t, const char* className, const char* methodName,
break; break;
} }
} }
}
va_end(a); void
pushArguments(Thread* t, object this_, const char* spec, object a)
{
if (this_) {
pushObject(t, this_);
}
object class_ = resolveClass(t, makeByteArray(t, "%s", className)); unsigned index = 0;
if (LIKELY(t->exception == 0)) { const char* s = spec;
PROTECT(t, class_); ++ s; // skip '('
while (*s and *s != ')') {
switch (*s) {
case 'L':
while (*s and *s != ';') ++ s;
++ s;
pushObject(t, objectArrayBody(t, a, index++));
break;
object name = makeByteArray(t, methodName); case '[':
PROTECT(t, name); while (*s == '[') ++ s;
switch (*s) {
case 'L':
while (*s and *s != ';') ++ s;
++ s;
break;
object spec = makeByteArray(t, methodSpec); default:
object reference = makeReference(t, class_, name, spec); ++ s;
break;
}
pushObject(t, objectArrayBody(t, a, index++));
break;
object method = findMethodInClass(t, class_, reference); case 'J':
if (LIKELY(t->exception == 0)) { case 'D':
assert(t, ((methodFlags(t, method) & ACC_STATIC) == 0) xor (this_ == 0)); ++ s;
pushLong(t, cast<int64_t>(objectArrayBody(t, a, index++), BytesPerWord));
break;
default:
++ s;
pushInt(t, cast<int32_t>(objectArrayBody(t, a, index++), BytesPerWord));
break;
}
}
}
object
invoke(Thread* t, object method)
{
object result = 0;
if (methodFlags(t, method) & ACC_NATIVE) { if (methodFlags(t, method) & ACC_NATIVE) {
unsigned returnCode = invokeNative(t, method); unsigned returnCode = invokeNative(t, method);
@ -2359,12 +2319,109 @@ run(Thread* t, const char* className, const char* methodName,
checkStack(t, method); checkStack(t, method);
if (LIKELY(t->exception == 0)) { if (LIKELY(t->exception == 0)) {
pushFrame(t, method); pushFrame(t, method);
} result = ::run(t);
if (LIKELY(t->exception == 0)) {
popFrame(t);
} }
} }
} }
return ::run(t); return result;
}
} // 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<char*>
(&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<char*>
(&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 int
@ -2379,7 +2436,10 @@ run(System* system, Heap* heap, ClassFinder* classFinder,
::run(t, className, argc, argv); ::run(t, className, argc, argv);
int exitCode = 0; int exitCode = 0;
if (t->exception) exitCode = -1; if (t->exception) {
exitCode = -1;
printTrace(t, t->exception);
}
exit(t); exit(t);

View File

@ -8,6 +8,12 @@
namespace vm { namespace vm {
object
run(Thread* t, object method, object this_, ...);
object
run2(Thread* t, object method, object this_, object arguments);
object object
run(Thread* t, const char* className, const char* methodName, run(Thread* t, const char* className, const char* methodName,
const char* methodSpec, object this_, ...); const char* methodSpec, object this_, ...);

View File

@ -212,7 +212,7 @@ class MySystem: public System {
} }
virtual void dispose() { virtual void dispose() {
assert(s, context == 0); //assert(s, context == 0);
pthread_mutex_destroy(&mutex); pthread_mutex_destroy(&mutex);
pthread_cond_destroy(&condition); pthread_cond_destroy(&condition);
s->free(this); s->free(this);

View File

@ -139,9 +139,9 @@
(type throwable java/lang/Throwable (type throwable java/lang/Throwable
(extends jobject) (extends jobject)
(object message) (noassert object message)
(object trace) (noassert object trace)
(object cause)) (noassert object cause))
(type exception java/lang/Exception (type exception java/lang/Exception
(extends throwable)) (extends throwable))
@ -155,6 +155,9 @@
(type illegalStateException java/lang/IllegalStateException (type illegalStateException java/lang/IllegalStateException
(extends runtimeException)) (extends runtimeException))
(type illegalArgumentException java/lang/IllegalArgumentException
(extends runtimeException))
(type illegalMonitorStateException java/lang/IllegalMonitorStateException (type illegalMonitorStateException java/lang/IllegalMonitorStateException
(extends runtimeException)) (extends runtimeException))
@ -174,6 +177,9 @@
(type classNotFoundException java/lang/ClassNotFoundException (type classNotFoundException java/lang/ClassNotFoundException
(extends runtimeException)) (extends runtimeException))
(type invocationTargetException java/lang/InvocationTargetException
(extends exception))
(type error java/lang/Error (type error java/lang/Error
(extends throwable)) (extends throwable))
@ -231,7 +237,7 @@
(type jreference java/lang/ref/Reference (type jreference java/lang/ref/Reference
(extends jobject) (extends jobject)
(void* next) (noassert void* next)
(void* target) (void* target)
(void* queue) (void* queue)
(object jnext)) (object jnext))
@ -273,3 +279,30 @@
(type doubleArray [D (type doubleArray [D
(extends jobject) (extends jobject)
(array uint64_t body)) (array uint64_t body))
(type jbyte B
(extends jobject))
(type jboolean Z
(extends jobject))
(type jshort S
(extends jobject))
(type jchar C
(extends jobject))
(type jint I
(extends jobject))
(type jlong L
(extends jobject))
(type jfloat F
(extends jobject))
(type jdouble D
(extends jobject))
(type jvoid V
(extends jobject))

13
test/Reflection.java Normal file
View File

@ -0,0 +1,13 @@
import java.lang.reflect.Method;
import java.lang.reflect.Field;
public class Reflection {
public static void main(String[] args) throws Exception {
Class system = Class.forName("java.lang.System");
Field out = system.getDeclaredField("out");
Class output = Class.forName("java.lang.System$Output");
Method println = output.getDeclaredMethod("println", String.class);
println.invoke(out.get(null), "Hello, World!");
}
}