2009-03-15 18:02:36 +00:00
|
|
|
/* Copyright (c) 2008-2009, Avian Contributors
|
2008-02-19 18:06:52 +00:00
|
|
|
|
|
|
|
Permission to use, copy, modify, and/or distribute this software
|
|
|
|
for any purpose with or without fee is hereby granted, provided
|
|
|
|
that the above copyright notice and this permission notice appear
|
|
|
|
in all copies.
|
|
|
|
|
|
|
|
There is NO WARRANTY for this software. See license.txt for
|
|
|
|
details. */
|
|
|
|
|
2007-09-26 23:23:03 +00:00
|
|
|
#ifndef PROCESS_H
|
|
|
|
#define PROCESS_H
|
|
|
|
|
|
|
|
#include "common.h"
|
|
|
|
#include "system.h"
|
|
|
|
#include "machine.h"
|
|
|
|
#include "constants.h"
|
|
|
|
|
|
|
|
namespace vm {
|
|
|
|
|
|
|
|
inline int16_t
|
|
|
|
codeReadInt16(Thread* t, object code, unsigned& ip)
|
|
|
|
{
|
|
|
|
uint8_t v1 = codeBody(t, code, ip++);
|
|
|
|
uint8_t v2 = codeBody(t, code, ip++);
|
|
|
|
return ((v1 << 8) | v2);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline int32_t
|
|
|
|
codeReadInt32(Thread* t, object code, unsigned& ip)
|
|
|
|
{
|
|
|
|
uint8_t v1 = codeBody(t, code, ip++);
|
|
|
|
uint8_t v2 = codeBody(t, code, ip++);
|
|
|
|
uint8_t v3 = codeBody(t, code, ip++);
|
|
|
|
uint8_t v4 = codeBody(t, code, ip++);
|
|
|
|
return ((v1 << 24) | (v2 << 16) | (v3 << 8) | v4);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline object
|
2009-08-10 13:56:16 +00:00
|
|
|
resolveClassInObject(Thread* t, object loader, object container,
|
|
|
|
unsigned classOffset)
|
2007-09-26 23:23:03 +00:00
|
|
|
{
|
2007-10-22 17:22:30 +00:00
|
|
|
object o = cast<object>(container, classOffset);
|
2008-04-10 23:48:28 +00:00
|
|
|
if (objectClass(t, o) == arrayBody(t, t->m->types, Machine::ByteArrayType)) {
|
2007-09-26 23:23:03 +00:00
|
|
|
PROTECT(t, container);
|
|
|
|
|
2009-08-10 13:56:16 +00:00
|
|
|
o = resolveClass(t, loader, o);
|
2007-09-26 23:23:03 +00:00
|
|
|
if (UNLIKELY(t->exception)) return 0;
|
|
|
|
|
2007-10-22 17:22:30 +00:00
|
|
|
set(t, container, classOffset, o);
|
2007-09-26 23:23:03 +00:00
|
|
|
}
|
|
|
|
return o;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline object
|
2009-08-10 13:56:16 +00:00
|
|
|
resolveClassInPool(Thread* t, object method, unsigned index)
|
2007-09-26 23:23:03 +00:00
|
|
|
{
|
2009-08-10 13:56:16 +00:00
|
|
|
object o = singletonObject(t, codePool(t, methodCode(t, method)), index);
|
2008-04-10 23:48:28 +00:00
|
|
|
if (objectClass(t, o) == arrayBody(t, t->m->types, Machine::ByteArrayType)) {
|
2009-08-10 13:56:16 +00:00
|
|
|
PROTECT(t, method);
|
2007-09-26 23:23:03 +00:00
|
|
|
|
2009-08-10 13:56:16 +00:00
|
|
|
o = resolveClass(t, classLoader(t, methodClass(t, method)), o);
|
2007-09-26 23:23:03 +00:00
|
|
|
if (UNLIKELY(t->exception)) return 0;
|
|
|
|
|
2009-08-10 13:56:16 +00:00
|
|
|
set(t, codePool(t, methodCode(t, method)),
|
|
|
|
SingletonBody + (index * BytesPerWord), o);
|
2007-09-26 23:23:03 +00:00
|
|
|
}
|
|
|
|
return o;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline object
|
2009-08-10 13:56:16 +00:00
|
|
|
resolve(Thread* t, object method, unsigned index,
|
2007-09-26 23:23:03 +00:00
|
|
|
object (*find)(vm::Thread*, object, object, object),
|
|
|
|
object (*makeError)(vm::Thread*, object))
|
|
|
|
{
|
2009-08-10 13:56:16 +00:00
|
|
|
object o = singletonObject(t, codePool(t, methodCode(t, method)), index);
|
2007-09-26 23:23:03 +00:00
|
|
|
if (objectClass(t, o) == arrayBody(t, t->m->types, Machine::ReferenceType))
|
|
|
|
{
|
2009-08-10 13:56:16 +00:00
|
|
|
PROTECT(t, method);
|
2007-09-26 23:23:03 +00:00
|
|
|
|
|
|
|
object reference = o;
|
|
|
|
PROTECT(t, reference);
|
|
|
|
|
2009-08-10 13:56:16 +00:00
|
|
|
object class_ = resolveClassInObject
|
|
|
|
(t, classLoader(t, methodClass(t, method)), o, ReferenceClass);
|
2007-09-26 23:23:03 +00:00
|
|
|
if (UNLIKELY(t->exception)) return 0;
|
|
|
|
|
|
|
|
o = findInHierarchy
|
|
|
|
(t, class_, referenceName(t, reference), referenceSpec(t, reference),
|
|
|
|
find, makeError);
|
|
|
|
if (UNLIKELY(t->exception)) return 0;
|
|
|
|
|
2009-08-10 13:56:16 +00:00
|
|
|
set(t, codePool(t, methodCode(t, method)),
|
|
|
|
SingletonBody + (index * BytesPerWord), o);
|
2007-09-26 23:23:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return o;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline object
|
2009-08-10 13:56:16 +00:00
|
|
|
resolveField(Thread* t, object method, unsigned index)
|
2007-09-26 23:23:03 +00:00
|
|
|
{
|
2009-08-10 13:56:16 +00:00
|
|
|
return resolve(t, method, index, findFieldInClass, makeNoSuchFieldError);
|
2007-09-26 23:23:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
inline object
|
2009-08-10 13:56:16 +00:00
|
|
|
resolveMethod(Thread* t, object method, unsigned index)
|
2007-09-26 23:23:03 +00:00
|
|
|
{
|
2009-08-10 13:56:16 +00:00
|
|
|
return resolve(t, method, index, findMethodInClass, makeNoSuchMethodError);
|
2007-09-26 23:23:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
inline 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<const int8_t*>("<init>"),
|
|
|
|
&byteArrayBody(t, methodName(t, method), 0)) != 0
|
|
|
|
and isSuperclass(t, methodClass(t, method), class_);
|
|
|
|
}
|
|
|
|
|
2008-01-28 17:27:02 +00:00
|
|
|
void*
|
2009-05-05 01:04:17 +00:00
|
|
|
resolveNativeMethod(Thread* t, object method);
|
2007-09-28 23:41:03 +00:00
|
|
|
|
2007-10-16 17:21:26 +00:00
|
|
|
inline object
|
|
|
|
findInterfaceMethod(Thread* t, object method, object class_)
|
|
|
|
{
|
2007-11-05 14:28:46 +00:00
|
|
|
assert(t, (classVmFlags(t, class_) & BootstrapFlag) == 0);
|
|
|
|
|
2007-10-16 17:21:26 +00:00
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
2007-11-20 16:20:26 +00:00
|
|
|
inline void
|
|
|
|
populateMultiArray(Thread* t, object array, int32_t* counts,
|
|
|
|
unsigned index, unsigned dimensions)
|
|
|
|
{
|
|
|
|
if (index + 1 == dimensions or counts[index] == 0) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
PROTECT(t, array);
|
|
|
|
|
|
|
|
object spec = className(t, objectClass(t, array));
|
|
|
|
PROTECT(t, spec);
|
|
|
|
|
2009-03-04 03:05:48 +00:00
|
|
|
object elementSpec = makeByteArray(t, byteArrayLength(t, spec) - 1);
|
2007-11-20 16:20:26 +00:00
|
|
|
memcpy(&byteArrayBody(t, elementSpec, 0),
|
|
|
|
&byteArrayBody(t, spec, 1),
|
|
|
|
byteArrayLength(t, spec) - 1);
|
|
|
|
|
2009-08-10 13:56:16 +00:00
|
|
|
object class_ = resolveSystemClass(t, elementSpec);
|
2007-11-20 16:20:26 +00:00
|
|
|
PROTECT(t, class_);
|
|
|
|
|
|
|
|
for (int32_t i = 0; i < counts[index]; ++i) {
|
2009-03-04 03:05:48 +00:00
|
|
|
object a = makeArray(t, counts[index + 1]);
|
2007-11-20 16:20:26 +00:00
|
|
|
setObjectClass(t, a, class_);
|
|
|
|
set(t, array, ArrayBody + (i * BytesPerWord), a);
|
|
|
|
|
|
|
|
populateMultiArray(t, a, counts, index + 1, dimensions);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-01-28 17:27:02 +00:00
|
|
|
int
|
|
|
|
findLineNumber(Thread* t, object method, unsigned ip);
|
2007-11-26 23:15:53 +00:00
|
|
|
|
2007-09-26 23:23:03 +00:00
|
|
|
} // namespace vm
|
|
|
|
|
|
|
|
#endif//PROCESS_H
|