2013-07-02 20:52:38 -06:00
|
|
|
/* Copyright (c) 2008-2013, Avian Contributors
|
2010-09-10 15:05:29 -06: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. */
|
|
|
|
|
|
|
|
#ifndef CLASSPATH_COMMON_H
|
|
|
|
#define CLASSPATH_COMMON_H
|
|
|
|
|
2013-02-21 16:18:20 -07:00
|
|
|
#include <avian/util/string.h>
|
2013-02-19 22:56:05 -07:00
|
|
|
#include <avian/util/runtime-array.h>
|
2010-09-20 17:31:23 -06:00
|
|
|
|
2013-02-21 16:18:20 -07:00
|
|
|
using namespace avian::util;
|
2013-02-10 17:51:59 -07:00
|
|
|
|
2010-09-10 15:05:29 -06:00
|
|
|
namespace vm {
|
|
|
|
|
2010-09-20 17:31:23 -06:00
|
|
|
object
|
2010-09-10 15:05:29 -06:00
|
|
|
getTrace(Thread* t, unsigned skipCount)
|
|
|
|
{
|
|
|
|
class Visitor: public Processor::StackVisitor {
|
|
|
|
public:
|
|
|
|
Visitor(Thread* t, int skipCount):
|
|
|
|
t(t), trace(0), skipCount(skipCount)
|
|
|
|
{ }
|
|
|
|
|
|
|
|
virtual bool visit(Processor::StackWalker* walker) {
|
|
|
|
if (skipCount == 0) {
|
|
|
|
object method = walker->method();
|
|
|
|
if (isAssignableFrom
|
2010-09-14 10:49:41 -06:00
|
|
|
(t, type(t, Machine::ThrowableType), methodClass(t, method))
|
2010-09-10 15:05:29 -06:00
|
|
|
and vm::strcmp(reinterpret_cast<const int8_t*>("<init>"),
|
|
|
|
&byteArrayBody(t, methodName(t, method), 0))
|
|
|
|
== 0)
|
|
|
|
{
|
|
|
|
return true;
|
|
|
|
} else {
|
|
|
|
trace = makeTrace(t, walker);
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
-- skipCount;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Thread* t;
|
|
|
|
object trace;
|
|
|
|
unsigned skipCount;
|
|
|
|
} v(t, skipCount);
|
|
|
|
|
|
|
|
t->m->processor->walkStack(t, &v);
|
|
|
|
|
2010-09-14 10:49:41 -06:00
|
|
|
if (v.trace == 0) v.trace = makeObjectArray(t, 0);
|
2010-09-10 15:05:29 -06:00
|
|
|
|
|
|
|
return v.trace;
|
|
|
|
}
|
|
|
|
|
2010-09-20 17:31:23 -06:00
|
|
|
bool
|
2010-09-10 15:05:29 -06:00
|
|
|
compatibleArrayTypes(Thread* t, object a, object b)
|
|
|
|
{
|
|
|
|
return classArrayElementSize(t, a)
|
|
|
|
and classArrayElementSize(t, b)
|
|
|
|
and (a == b
|
|
|
|
or (not ((classVmFlags(t, a) & PrimitiveFlag)
|
|
|
|
or (classVmFlags(t, b) & PrimitiveFlag))));
|
|
|
|
}
|
|
|
|
|
2010-09-20 17:31:23 -06:00
|
|
|
void
|
2010-09-10 15:05:29 -06:00
|
|
|
arrayCopy(Thread* t, object src, int32_t srcOffset, object dst,
|
|
|
|
int32_t dstOffset, int32_t length)
|
|
|
|
{
|
|
|
|
if (LIKELY(src and dst)) {
|
|
|
|
if (LIKELY(compatibleArrayTypes
|
|
|
|
(t, objectClass(t, src), objectClass(t, dst))))
|
|
|
|
{
|
|
|
|
unsigned elementSize = classArrayElementSize(t, objectClass(t, src));
|
|
|
|
|
|
|
|
if (LIKELY(elementSize)) {
|
2013-02-10 17:38:51 -07:00
|
|
|
intptr_t sl = fieldAtOffset<uintptr_t>(src, BytesPerWord);
|
|
|
|
intptr_t dl = fieldAtOffset<uintptr_t>(dst, BytesPerWord);
|
2010-09-10 15:05:29 -06:00
|
|
|
if (LIKELY(length > 0)) {
|
|
|
|
if (LIKELY(srcOffset >= 0 and srcOffset + length <= sl and
|
|
|
|
dstOffset >= 0 and dstOffset + length <= dl))
|
|
|
|
{
|
2013-02-10 17:38:51 -07:00
|
|
|
uint8_t* sbody = &fieldAtOffset<uint8_t>(src, ArrayBody);
|
|
|
|
uint8_t* dbody = &fieldAtOffset<uint8_t>(dst, ArrayBody);
|
2010-09-10 15:05:29 -06:00
|
|
|
if (src == dst) {
|
|
|
|
memmove(dbody + (dstOffset * elementSize),
|
|
|
|
sbody + (srcOffset * elementSize),
|
|
|
|
length * elementSize);
|
|
|
|
} else {
|
|
|
|
memcpy(dbody + (dstOffset * elementSize),
|
|
|
|
sbody + (srcOffset * elementSize),
|
|
|
|
length * elementSize);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (classObjectMask(t, objectClass(t, dst))) {
|
|
|
|
mark(t, dst, ArrayBody + (dstOffset * BytesPerWord), length);
|
|
|
|
}
|
|
|
|
|
|
|
|
return;
|
|
|
|
} else {
|
2010-12-27 15:55:23 -07:00
|
|
|
throwNew(t, Machine::IndexOutOfBoundsExceptionType);
|
2010-09-10 15:05:29 -06:00
|
|
|
}
|
|
|
|
} else {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
2010-12-27 15:55:23 -07:00
|
|
|
throwNew(t, Machine::NullPointerExceptionType);
|
2010-09-10 15:05:29 -06:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2010-12-27 15:55:23 -07:00
|
|
|
throwNew(t, Machine::ArrayStoreExceptionType);
|
2010-09-10 15:05:29 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
void
|
|
|
|
runOnLoadIfFound(Thread* t, System::Library* library)
|
|
|
|
{
|
|
|
|
void* p = library->resolve("JNI_OnLoad");
|
2012-03-18 20:10:42 -06:00
|
|
|
|
|
|
|
#ifdef PLATFORM_WINDOWS
|
|
|
|
if (p == 0) {
|
|
|
|
p = library->resolve("_JNI_OnLoad@8");
|
|
|
|
if (p == 0) {
|
|
|
|
p = library->resolve("JNI_OnLoad@8");
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
|
2010-09-10 15:05:29 -06:00
|
|
|
if (p) {
|
|
|
|
jint (JNICALL * JNI_OnLoad)(Machine*, void*);
|
|
|
|
memcpy(&JNI_OnLoad, &p, sizeof(void*));
|
|
|
|
JNI_OnLoad(t->m, 0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
System::Library*
|
2010-09-20 17:31:23 -06:00
|
|
|
loadLibrary(Thread* t, const char* name)
|
2010-09-10 15:05:29 -06:00
|
|
|
{
|
|
|
|
ACQUIRE(t, t->m->classLock);
|
|
|
|
|
|
|
|
System::Library* last = t->m->libraries;
|
|
|
|
for (System::Library* lib = t->m->libraries; lib; lib = lib->next()) {
|
2010-09-20 17:31:23 -06:00
|
|
|
if (lib->name() and ::strcmp(lib->name(), name) == 0) {
|
2010-09-10 15:05:29 -06:00
|
|
|
// already loaded
|
|
|
|
return lib;
|
|
|
|
}
|
|
|
|
last = lib;
|
|
|
|
}
|
|
|
|
|
|
|
|
System::Library* lib;
|
2010-09-22 13:58:46 -06:00
|
|
|
if (t->m->system->success(t->m->system->load(&lib, name))) {
|
2010-09-10 15:05:29 -06:00
|
|
|
last->setNext(lib);
|
2010-09-20 17:31:23 -06:00
|
|
|
return lib;
|
|
|
|
} else {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
System::Library*
|
|
|
|
loadLibrary(Thread* t, const char* path, const char* name, bool mapName,
|
2013-02-27 11:34:43 -07:00
|
|
|
bool runOnLoad, bool throw_ = true)
|
2010-09-20 17:31:23 -06:00
|
|
|
{
|
|
|
|
ACQUIRE(t, t->m->classLock);
|
|
|
|
|
2010-12-27 15:55:23 -07:00
|
|
|
char* mappedName;
|
2010-09-20 17:31:23 -06:00
|
|
|
unsigned nameLength = strlen(name);
|
|
|
|
if (mapName) {
|
|
|
|
const char* builtins = findProperty(t, "avian.builtins");
|
|
|
|
if (builtins) {
|
|
|
|
const char* s = builtins;
|
|
|
|
while (*s) {
|
|
|
|
if (::strncmp(s, name, nameLength) == 0
|
|
|
|
and (s[nameLength] == ',' or s[nameLength] == 0))
|
|
|
|
{
|
|
|
|
// library is built in to this executable
|
|
|
|
if (runOnLoad and not t->m->triedBuiltinOnLoad) {
|
|
|
|
t->m->triedBuiltinOnLoad = true;
|
2010-12-08 21:36:02 -07:00
|
|
|
// todo: release the classLock before calling this to
|
|
|
|
// avoid the possibility of deadlock:
|
2010-09-20 17:31:23 -06:00
|
|
|
runOnLoadIfFound(t, t->m->libraries);
|
|
|
|
}
|
|
|
|
return t->m->libraries;
|
|
|
|
} else {
|
|
|
|
while (*s and *s != ',') ++ s;
|
|
|
|
if (*s) ++ s;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
const char* prefix = t->m->system->libraryPrefix();
|
|
|
|
const char* suffix = t->m->system->librarySuffix();
|
|
|
|
unsigned mappedNameLength = nameLength + strlen(prefix) + strlen(suffix);
|
|
|
|
|
2010-12-27 15:55:23 -07:00
|
|
|
mappedName = static_cast<char*>
|
2010-09-20 17:31:23 -06:00
|
|
|
(t->m->heap->allocate(mappedNameLength + 1));
|
|
|
|
|
|
|
|
snprintf(mappedName, mappedNameLength + 1, "%s%s%s", prefix, name, suffix);
|
|
|
|
|
|
|
|
name = mappedName;
|
|
|
|
nameLength = mappedNameLength;
|
2010-12-27 15:55:23 -07:00
|
|
|
} else {
|
|
|
|
mappedName = 0;
|
2010-09-20 17:31:23 -06:00
|
|
|
}
|
|
|
|
|
2010-12-27 15:55:23 -07:00
|
|
|
THREAD_RESOURCE2
|
|
|
|
(t, char*, mappedName, unsigned, nameLength, if (mappedName) {
|
|
|
|
t->m->heap->free(mappedName, nameLength + 1);
|
|
|
|
});
|
|
|
|
|
2010-09-20 17:31:23 -06:00
|
|
|
System::Library* lib = 0;
|
|
|
|
for (Tokenizer tokenizer(path, t->m->system->pathSeparator());
|
|
|
|
tokenizer.hasMore();)
|
|
|
|
{
|
2013-02-21 16:18:20 -07:00
|
|
|
String token(tokenizer.next());
|
2010-09-20 17:31:23 -06:00
|
|
|
|
|
|
|
unsigned fullNameLength = token.length + 1 + nameLength;
|
2010-12-27 15:55:23 -07:00
|
|
|
THREAD_RUNTIME_ARRAY(t, char, fullName, fullNameLength + 1);
|
2010-09-20 17:31:23 -06:00
|
|
|
|
|
|
|
snprintf(RUNTIME_ARRAY_BODY(fullName), fullNameLength + 1,
|
2013-02-21 16:18:20 -07:00
|
|
|
"%.*s/%s", token.length, token.text, name);
|
2010-09-20 17:31:23 -06:00
|
|
|
|
|
|
|
lib = loadLibrary(t, RUNTIME_ARRAY_BODY(fullName));
|
|
|
|
if (lib) break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (lib == 0) {
|
|
|
|
lib = loadLibrary(t, name);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (lib) {
|
2010-09-10 15:05:29 -06:00
|
|
|
if (runOnLoad) {
|
|
|
|
runOnLoadIfFound(t, lib);
|
|
|
|
}
|
2013-02-27 11:34:43 -07:00
|
|
|
} else if (throw_) {
|
2010-12-27 15:55:23 -07:00
|
|
|
throwNew(t, Machine::UnsatisfiedLinkErrorType, "library not found: %s",
|
|
|
|
name);
|
2010-09-20 17:31:23 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
return lib;
|
2010-09-10 15:05:29 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
object
|
|
|
|
clone(Thread* t, object o)
|
|
|
|
{
|
|
|
|
PROTECT(t, o);
|
|
|
|
|
|
|
|
object class_ = objectClass(t, o);
|
|
|
|
unsigned size = baseSize(t, o, class_) * BytesPerWord;
|
|
|
|
object clone;
|
|
|
|
|
|
|
|
if (classArrayElementSize(t, class_)) {
|
|
|
|
clone = static_cast<object>(allocate(t, size, classObjectMask(t, class_)));
|
|
|
|
memcpy(clone, o, size);
|
|
|
|
// clear any object header flags:
|
|
|
|
setObjectClass(t, o, objectClass(t, o));
|
2013-04-22 21:19:01 -06:00
|
|
|
} else if (instanceOf(t, type(t, Machine::CloneableType), o)) {
|
2010-09-10 15:05:29 -06:00
|
|
|
clone = make(t, class_);
|
|
|
|
memcpy(reinterpret_cast<void**>(clone) + 1,
|
|
|
|
reinterpret_cast<void**>(o) + 1,
|
|
|
|
size - BytesPerWord);
|
2013-04-22 21:19:01 -06:00
|
|
|
} else {
|
2013-05-14 01:23:03 +02:00
|
|
|
object classNameSlash = className(t, objectClass(t, o));
|
|
|
|
THREAD_RUNTIME_ARRAY(t, char, classNameDot, byteArrayLength(t, classNameSlash));
|
|
|
|
replace('/', '.', RUNTIME_ARRAY_BODY(classNameDot),
|
|
|
|
reinterpret_cast<char*>(&byteArrayBody(t, classNameSlash, 0)));
|
2013-04-22 21:19:01 -06:00
|
|
|
throwNew(t, Machine::CloneNotSupportedExceptionType, "%s",
|
2013-05-14 01:23:03 +02:00
|
|
|
RUNTIME_ARRAY_BODY(classNameDot));
|
2010-09-10 15:05:29 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
return clone;
|
|
|
|
}
|
|
|
|
|
|
|
|
object
|
|
|
|
makeStackTraceElement(Thread* t, object e)
|
|
|
|
{
|
|
|
|
PROTECT(t, e);
|
|
|
|
|
|
|
|
object class_ = className(t, methodClass(t, traceElementMethod(t, e)));
|
|
|
|
PROTECT(t, class_);
|
|
|
|
|
2010-12-27 15:55:23 -07:00
|
|
|
THREAD_RUNTIME_ARRAY(t, char, s, byteArrayLength(t, class_));
|
2010-09-10 15:05:29 -06:00
|
|
|
replace('/', '.', RUNTIME_ARRAY_BODY(s),
|
|
|
|
reinterpret_cast<char*>(&byteArrayBody(t, class_, 0)));
|
2012-08-26 17:27:22 +02:00
|
|
|
class_ = makeString(t, "%s", RUNTIME_ARRAY_BODY(s));
|
2010-09-10 15:05:29 -06:00
|
|
|
|
|
|
|
object method = methodName(t, traceElementMethod(t, e));
|
|
|
|
PROTECT(t, method);
|
|
|
|
|
|
|
|
method = t->m->classpath->makeString
|
|
|
|
(t, method, 0, byteArrayLength(t, method) - 1);
|
|
|
|
|
|
|
|
unsigned line = t->m->processor->lineNumber
|
|
|
|
(t, traceElementMethod(t, e), traceElementIp(t, e));
|
|
|
|
|
|
|
|
object file = classSourceFile(t, methodClass(t, traceElementMethod(t, e)));
|
|
|
|
file = file ? t->m->classpath->makeString
|
|
|
|
(t, file, 0, byteArrayLength(t, file) - 1) : 0;
|
|
|
|
|
|
|
|
return makeStackTraceElement(t, class_, method, file, line);
|
|
|
|
}
|
|
|
|
|
2011-02-22 17:54:56 -07:00
|
|
|
object
|
|
|
|
translateInvokeResult(Thread* t, unsigned returnCode, object o)
|
|
|
|
{
|
|
|
|
switch (returnCode) {
|
|
|
|
case ByteField:
|
|
|
|
return makeByte(t, intValue(t, o));
|
|
|
|
|
|
|
|
case BooleanField:
|
|
|
|
return makeBoolean(t, intValue(t, o) != 0);
|
|
|
|
|
|
|
|
case CharField:
|
|
|
|
return makeChar(t, intValue(t, o));
|
|
|
|
|
|
|
|
case ShortField:
|
|
|
|
return makeShort(t, intValue(t, o));
|
|
|
|
|
|
|
|
case FloatField:
|
|
|
|
return makeFloat(t, intValue(t, o));
|
|
|
|
|
|
|
|
case IntField:
|
|
|
|
case LongField:
|
|
|
|
case ObjectField:
|
|
|
|
case VoidField:
|
|
|
|
return o;
|
|
|
|
|
|
|
|
case DoubleField:
|
|
|
|
return makeDouble(t, longValue(t, o));
|
|
|
|
|
|
|
|
default:
|
|
|
|
abort(t);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-02-20 10:22:40 -07:00
|
|
|
object
|
|
|
|
resolveClassBySpec(Thread* t, object loader, const char* spec,
|
|
|
|
unsigned specLength)
|
|
|
|
{
|
|
|
|
switch (*spec) {
|
|
|
|
case 'L': {
|
|
|
|
THREAD_RUNTIME_ARRAY(t, char, s, specLength - 1);
|
|
|
|
memcpy(RUNTIME_ARRAY_BODY(s), spec + 1, specLength - 2);
|
|
|
|
RUNTIME_ARRAY_BODY(s)[specLength - 2] = 0;
|
2013-02-22 14:50:15 -07:00
|
|
|
return resolveClass(t, loader, RUNTIME_ARRAY_BODY(s));
|
2013-02-20 10:22:40 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
case '[': {
|
|
|
|
THREAD_RUNTIME_ARRAY(t, char, s, specLength + 1);
|
|
|
|
memcpy(RUNTIME_ARRAY_BODY(s), spec, specLength);
|
|
|
|
RUNTIME_ARRAY_BODY(s)[specLength] = 0;
|
2013-02-22 14:50:15 -07:00
|
|
|
return resolveClass(t, loader, RUNTIME_ARRAY_BODY(s));
|
2013-02-20 10:22:40 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
default:
|
|
|
|
return primitiveClass(t, *spec);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-02-21 15:37:17 -07:00
|
|
|
object
|
|
|
|
resolveJType(Thread* t, object loader, const char* spec, unsigned specLength)
|
|
|
|
{
|
|
|
|
return getJClass(t, resolveClassBySpec(t, loader, spec, specLength));
|
|
|
|
}
|
|
|
|
|
|
|
|
object
|
|
|
|
resolveParameterTypes(Thread* t, object loader, object spec,
|
|
|
|
unsigned* parameterCount, unsigned* returnTypeSpec)
|
|
|
|
{
|
|
|
|
PROTECT(t, loader);
|
|
|
|
PROTECT(t, spec);
|
|
|
|
|
|
|
|
object list = 0;
|
|
|
|
PROTECT(t, list);
|
|
|
|
|
|
|
|
unsigned offset = 1;
|
|
|
|
unsigned count = 0;
|
|
|
|
while (byteArrayBody(t, spec, offset) != ')') {
|
|
|
|
switch (byteArrayBody(t, spec, offset)) {
|
|
|
|
case 'L': {
|
|
|
|
unsigned start = offset;
|
|
|
|
++ offset;
|
|
|
|
while (byteArrayBody(t, spec, offset) != ';') ++ offset;
|
|
|
|
++ offset;
|
|
|
|
|
|
|
|
object type = resolveClassBySpec
|
|
|
|
(t, loader, reinterpret_cast<char*>(&byteArrayBody(t, spec, start)),
|
|
|
|
offset - start);
|
|
|
|
|
|
|
|
list = makePair(t, type, list);
|
|
|
|
|
|
|
|
++ count;
|
|
|
|
} break;
|
|
|
|
|
|
|
|
case '[': {
|
|
|
|
unsigned start = offset;
|
|
|
|
while (byteArrayBody(t, spec, offset) == '[') ++ offset;
|
|
|
|
switch (byteArrayBody(t, spec, offset)) {
|
|
|
|
case 'L':
|
|
|
|
++ offset;
|
|
|
|
while (byteArrayBody(t, spec, offset) != ';') ++ offset;
|
|
|
|
++ offset;
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
++ offset;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
object type = resolveClassBySpec
|
|
|
|
(t, loader, reinterpret_cast<char*>(&byteArrayBody(t, spec, start)),
|
|
|
|
offset - start);
|
|
|
|
|
|
|
|
list = makePair(t, type, list);
|
|
|
|
++ count;
|
|
|
|
} break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
list = makePair
|
|
|
|
(t, primitiveClass(t, byteArrayBody(t, spec, offset)), list);
|
|
|
|
++ offset;
|
|
|
|
++ count;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
*parameterCount = count;
|
|
|
|
*returnTypeSpec = offset + 1;
|
|
|
|
return list;
|
|
|
|
}
|
|
|
|
|
|
|
|
object
|
|
|
|
resolveParameterJTypes(Thread* t, object loader, object spec,
|
|
|
|
unsigned* parameterCount, unsigned* returnTypeSpec)
|
|
|
|
{
|
|
|
|
object list = resolveParameterTypes
|
|
|
|
(t, loader, spec, parameterCount, returnTypeSpec);
|
|
|
|
|
|
|
|
PROTECT(t, list);
|
|
|
|
|
|
|
|
object array = makeObjectArray
|
|
|
|
(t, type(t, Machine::JclassType), *parameterCount);
|
|
|
|
PROTECT(t, array);
|
|
|
|
|
|
|
|
for (int i = *parameterCount - 1; i >= 0; --i) {
|
|
|
|
object c = getJClass(t, pairFirst(t, list));
|
|
|
|
set(t, array, ArrayBody + (i * BytesPerWord), c);
|
|
|
|
list = pairSecond(t, list);
|
|
|
|
}
|
|
|
|
|
|
|
|
return array;
|
|
|
|
}
|
|
|
|
|
|
|
|
object
|
|
|
|
resolveExceptionJTypes(Thread* t, object loader, object addendum)
|
|
|
|
{
|
|
|
|
if (addendum == 0 or methodAddendumExceptionTable(t, addendum) == 0) {
|
|
|
|
return makeObjectArray(t, type(t, Machine::JclassType), 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
PROTECT(t, loader);
|
|
|
|
PROTECT(t, addendum);
|
|
|
|
|
|
|
|
object array = makeObjectArray
|
|
|
|
(t, type(t, Machine::JclassType),
|
|
|
|
shortArrayLength(t, methodAddendumExceptionTable(t, addendum)));
|
|
|
|
PROTECT(t, array);
|
|
|
|
|
|
|
|
for (unsigned i = 0; i < shortArrayLength
|
|
|
|
(t, methodAddendumExceptionTable(t, addendum)); ++i)
|
|
|
|
{
|
|
|
|
uint16_t index = shortArrayBody
|
|
|
|
(t, methodAddendumExceptionTable(t, addendum), i) - 1;
|
|
|
|
|
|
|
|
object o = singletonObject(t, addendumPool(t, addendum), index);
|
|
|
|
|
|
|
|
if (objectClass(t, o) == type(t, Machine::ReferenceType)) {
|
|
|
|
o = resolveClass(t, loader, referenceName(t, o));
|
|
|
|
|
|
|
|
set(t, addendumPool(t, addendum), SingletonBody + (index * BytesPerWord),
|
|
|
|
o);
|
|
|
|
}
|
|
|
|
|
|
|
|
o = getJClass(t, o);
|
|
|
|
|
|
|
|
set(t, array, ArrayBody + (i * BytesPerWord), o);
|
|
|
|
}
|
|
|
|
|
|
|
|
return array;
|
|
|
|
}
|
|
|
|
|
|
|
|
object
|
|
|
|
invoke(Thread* t, object method, object instance, object args)
|
|
|
|
{
|
|
|
|
PROTECT(t, method);
|
|
|
|
PROTECT(t, instance);
|
|
|
|
PROTECT(t, args);
|
|
|
|
|
|
|
|
if (methodFlags(t, method) & ACC_STATIC) {
|
|
|
|
instance = 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((args == 0 ? 0 : objectArrayLength(t, args))
|
|
|
|
!= methodParameterCount(t, method))
|
|
|
|
{
|
|
|
|
throwNew(t, Machine::IllegalArgumentExceptionType);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (methodParameterCount(t, method)) {
|
|
|
|
PROTECT(t, method);
|
|
|
|
|
|
|
|
unsigned specLength = byteArrayLength(t, methodSpec(t, method));
|
|
|
|
THREAD_RUNTIME_ARRAY(t, char, spec, specLength);
|
2013-02-22 14:50:15 -07:00
|
|
|
memcpy(RUNTIME_ARRAY_BODY(spec),
|
|
|
|
&byteArrayBody(t, methodSpec(t, method), 0), specLength);
|
2013-02-21 15:37:17 -07:00
|
|
|
unsigned i = 0;
|
2013-02-22 14:50:15 -07:00
|
|
|
for (MethodSpecIterator it(t, RUNTIME_ARRAY_BODY(spec)); it.hasNext();) {
|
2013-02-21 15:37:17 -07:00
|
|
|
object type;
|
|
|
|
bool objectType = false;
|
|
|
|
const char* p = it.next();
|
|
|
|
switch (*p) {
|
|
|
|
case 'Z': type = vm::type(t, Machine::BooleanType); break;
|
|
|
|
case 'B': type = vm::type(t, Machine::ByteType); break;
|
|
|
|
case 'S': type = vm::type(t, Machine::ShortType); break;
|
|
|
|
case 'C': type = vm::type(t, Machine::CharType); break;
|
|
|
|
case 'I': type = vm::type(t, Machine::IntType); break;
|
|
|
|
case 'F': type = vm::type(t, Machine::FloatType); break;
|
|
|
|
case 'J': type = vm::type(t, Machine::LongType); break;
|
|
|
|
case 'D': type = vm::type(t, Machine::DoubleType); break;
|
|
|
|
|
2013-04-22 18:57:26 -06:00
|
|
|
case 'L':
|
2013-02-21 15:37:17 -07:00
|
|
|
case '[': {
|
|
|
|
objectType = true;
|
2013-04-22 18:57:26 -06:00
|
|
|
unsigned nameLength;
|
|
|
|
if (*p == 'L') {
|
|
|
|
++ p;
|
|
|
|
nameLength = it.s - p;
|
|
|
|
} else {
|
|
|
|
nameLength = (it.s - p) + 1;
|
|
|
|
}
|
2013-02-21 15:37:17 -07:00
|
|
|
THREAD_RUNTIME_ARRAY(t, char, name, nameLength);
|
2013-02-22 14:50:15 -07:00
|
|
|
memcpy(RUNTIME_ARRAY_BODY(name), p, nameLength - 1);
|
|
|
|
RUNTIME_ARRAY_BODY(name)[nameLength - 1] = 0;
|
2013-02-21 15:37:17 -07:00
|
|
|
type = resolveClass
|
2013-02-22 14:50:15 -07:00
|
|
|
(t, classLoader(t, methodClass(t, method)),
|
|
|
|
RUNTIME_ARRAY_BODY(name));
|
2013-02-21 15:37:17 -07:00
|
|
|
} break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
abort();
|
|
|
|
}
|
|
|
|
|
|
|
|
object arg = objectArrayBody(t, args, i++);
|
|
|
|
if ((arg == 0 and (not objectType))
|
|
|
|
or (arg and (not instanceOf(t, type, arg))))
|
|
|
|
{
|
|
|
|
// fprintf(stderr, "%s is not a %s\n", arg ? &byteArrayBody(t, className(t, objectClass(t, arg)), 0) : reinterpret_cast<const int8_t*>("<null>"), &byteArrayBody(t, className(t, type), 0));
|
|
|
|
|
|
|
|
throwNew(t, Machine::IllegalArgumentExceptionType);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-12-05 22:28:13 -07:00
|
|
|
initClass(t, methodClass(t, method));
|
|
|
|
|
2013-02-21 15:37:17 -07:00
|
|
|
unsigned returnCode = methodReturnCode(t, method);
|
|
|
|
|
|
|
|
THREAD_RESOURCE0(t, {
|
|
|
|
if (t->exception) {
|
|
|
|
t->exception = makeThrowable
|
2013-12-05 22:28:13 -07:00
|
|
|
(t, Machine::InvocationTargetExceptionType, 0, 0, t->exception);
|
2013-04-18 11:23:59 -06:00
|
|
|
|
|
|
|
set(t, t->exception, InvocationTargetExceptionTarget,
|
|
|
|
throwableCause(t, t->exception));
|
2013-02-21 15:37:17 -07:00
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
object result;
|
|
|
|
if (args) {
|
|
|
|
result = t->m->processor->invokeArray(t, method, instance, args);
|
|
|
|
} else {
|
|
|
|
result = t->m->processor->invoke(t, method, instance);
|
|
|
|
}
|
|
|
|
|
|
|
|
return translateInvokeResult(t, returnCode, result);
|
|
|
|
}
|
|
|
|
|
|
|
|
// only safe to call during bootstrap when there's only one thread
|
|
|
|
// running:
|
|
|
|
void
|
|
|
|
intercept(Thread* t, object c, const char* name, const char* spec,
|
2013-03-15 13:28:01 -06:00
|
|
|
void* function, bool updateRuntimeData)
|
2013-02-21 15:37:17 -07:00
|
|
|
{
|
|
|
|
object m = findMethodOrNull(t, c, name, spec);
|
|
|
|
if (m) {
|
|
|
|
PROTECT(t, m);
|
|
|
|
|
2013-06-04 13:32:22 -06:00
|
|
|
methodFlags(t, m) |= ACC_NATIVE;
|
|
|
|
|
2013-03-15 13:28:01 -06:00
|
|
|
if (updateRuntimeData) {
|
2013-06-04 13:32:22 -06:00
|
|
|
object clone = methodClone(t, m);
|
2013-02-21 15:37:17 -07:00
|
|
|
|
2013-03-15 13:28:01 -06:00
|
|
|
// make clone private to prevent vtable updates at compilation
|
|
|
|
// time. Otherwise, our interception might be bypassed by calls
|
|
|
|
// through the vtable.
|
|
|
|
methodFlags(t, clone) |= ACC_PRIVATE;
|
2013-02-21 15:37:17 -07:00
|
|
|
|
2013-03-15 13:28:01 -06:00
|
|
|
object native = makeNativeIntercept(t, function, true, clone);
|
|
|
|
|
|
|
|
PROTECT(t, native);
|
|
|
|
|
|
|
|
object runtimeData = getMethodRuntimeData(t, m);
|
|
|
|
|
|
|
|
set(t, runtimeData, MethodRuntimeDataNative, native);
|
|
|
|
}
|
2013-02-21 15:37:17 -07:00
|
|
|
} else {
|
|
|
|
// If we can't find the method, just ignore it, since ProGuard may
|
|
|
|
// have stripped it out as unused. Otherwise, the code below can
|
|
|
|
// be uncommented for debugging purposes.
|
|
|
|
|
|
|
|
// fprintf(stderr, "unable to find %s%s in %s\n",
|
|
|
|
// name, spec, &byteArrayBody(t, className(t, c), 0));
|
|
|
|
|
|
|
|
// abort(t);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-02-22 17:23:59 -07:00
|
|
|
Finder*
|
|
|
|
getFinder(Thread* t, const char* name, unsigned nameLength)
|
|
|
|
{
|
|
|
|
ACQUIRE(t, t->m->referenceLock);
|
|
|
|
|
|
|
|
for (object p = root(t, Machine::VirtualFileFinders);
|
|
|
|
p; p = finderNext(t, p))
|
|
|
|
{
|
|
|
|
if (byteArrayLength(t, finderName(t, p)) == nameLength
|
|
|
|
and strncmp(reinterpret_cast<const char*>
|
|
|
|
(&byteArrayBody(t, finderName(t, p), 0)),
|
|
|
|
name, nameLength))
|
|
|
|
{
|
|
|
|
return static_cast<Finder*>(finderFinder(t, p));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
object n = makeByteArray(t, nameLength + 1);
|
|
|
|
memcpy(&byteArrayBody(t, n, 0), name, nameLength);
|
|
|
|
|
|
|
|
void* p = t->m->libraries->resolve
|
|
|
|
(reinterpret_cast<const char*>(&byteArrayBody(t, n, 0)));
|
|
|
|
|
|
|
|
if (p) {
|
|
|
|
uint8_t* (*function)(unsigned*);
|
|
|
|
memcpy(&function, &p, BytesPerWord);
|
|
|
|
|
|
|
|
unsigned size;
|
|
|
|
uint8_t* data = function(&size);
|
|
|
|
if (data) {
|
|
|
|
Finder* f = makeFinder(t->m->system, t->m->heap, data, size);
|
|
|
|
object finder = makeFinder
|
|
|
|
(t, f, n, root(t, Machine::VirtualFileFinders));
|
|
|
|
|
|
|
|
setRoot(t, Machine::VirtualFileFinders, finder);
|
|
|
|
|
|
|
|
return f;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-12-06 15:45:46 -07:00
|
|
|
object
|
|
|
|
getDeclaredClasses(Thread* t, object c, bool publicOnly)
|
|
|
|
{
|
|
|
|
object addendum = classAddendum(t, c);
|
|
|
|
if (addendum) {
|
|
|
|
object table = classAddendumInnerClassTable(t, addendum);
|
|
|
|
if (table) {
|
|
|
|
PROTECT(t, table);
|
|
|
|
|
|
|
|
unsigned count = 0;
|
|
|
|
for (unsigned i = 0; i < arrayLength(t, table); ++i) {
|
|
|
|
object reference = arrayBody(t, table, i);
|
|
|
|
object outer = innerClassReferenceOuter(t, reference);
|
|
|
|
if (outer and byteArrayEqual(t, outer, className(t, c))
|
|
|
|
and ((not publicOnly)
|
|
|
|
or (innerClassReferenceFlags(t, reference) & ACC_PUBLIC)))
|
|
|
|
{
|
|
|
|
++ count;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
object result = makeObjectArray(t, type(t, Machine::JclassType), count);
|
|
|
|
PROTECT(t, result);
|
|
|
|
|
|
|
|
for (unsigned i = 0; i < arrayLength(t, table); ++i) {
|
|
|
|
object reference = arrayBody(t, table, i);
|
|
|
|
object outer = innerClassReferenceOuter(t, reference);
|
|
|
|
if (outer and byteArrayEqual(t, outer, className(t, c))
|
|
|
|
and ((not publicOnly)
|
|
|
|
or (innerClassReferenceFlags(t, reference) & ACC_PUBLIC)))
|
|
|
|
{
|
|
|
|
object inner = getJClass
|
|
|
|
(t, resolveClass
|
|
|
|
(t, classLoader(t, c),
|
|
|
|
innerClassReferenceInner(t, arrayBody(t, table, i))));
|
|
|
|
|
|
|
|
-- count;
|
|
|
|
set(t, result, ArrayBody + (count * BytesPerWord), inner);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return makeObjectArray(t, type(t, Machine::JclassType), 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
object
|
|
|
|
getDeclaringClass(Thread* t, object c)
|
|
|
|
{
|
|
|
|
object addendum = classAddendum(t, c);
|
|
|
|
if (addendum) {
|
|
|
|
object table = classAddendumInnerClassTable(t, addendum);
|
|
|
|
if (table) {
|
|
|
|
for (unsigned i = 0; i < arrayLength(t, table); ++i) {
|
|
|
|
object reference = arrayBody(t, table, i);
|
2014-03-14 11:08:08 -06:00
|
|
|
if (innerClassReferenceOuter(t, reference) and strcmp
|
2013-12-06 15:45:46 -07:00
|
|
|
(&byteArrayBody(t, innerClassReferenceInner(t, reference), 0),
|
|
|
|
&byteArrayBody(t, className(t, c), 0)) == 0)
|
|
|
|
{
|
|
|
|
return getJClass
|
|
|
|
(t, resolveClass
|
|
|
|
(t, classLoader(t, c), innerClassReferenceOuter(t, reference)));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-03-06 15:01:44 -07:00
|
|
|
unsigned
|
|
|
|
classModifiers(Thread* t, object c)
|
|
|
|
{
|
|
|
|
object addendum = classAddendum(t, c);
|
|
|
|
if (addendum) {
|
|
|
|
object table = classAddendumInnerClassTable(t, addendum);
|
|
|
|
if (table) {
|
|
|
|
for (unsigned i = 0; i < arrayLength(t, table); ++i) {
|
|
|
|
object reference = arrayBody(t, table, i);
|
|
|
|
if (0 == strcmp
|
|
|
|
(&byteArrayBody(t, className(t, c), 0),
|
|
|
|
&byteArrayBody(t, innerClassReferenceInner(t, reference), 0)))
|
|
|
|
{
|
|
|
|
return innerClassReferenceFlags(t, reference);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return classFlags(t, c);
|
|
|
|
}
|
|
|
|
|
2010-09-10 15:05:29 -06:00
|
|
|
} // namespace vm
|
|
|
|
|
|
|
|
#endif//CLASSPATH_COMMON_H
|