2014-04-21 02:14:48 +00:00
|
|
|
/* Copyright (c) 2008-2014, Avian Contributors
|
2013-02-19 16:36:19 +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. */
|
|
|
|
|
2013-02-19 23:48:33 +00:00
|
|
|
struct JavaVM;
|
2013-07-05 20:34:16 +00:00
|
|
|
struct _JNIEnv;
|
|
|
|
|
|
|
|
struct JniConstants {
|
|
|
|
static void init(_JNIEnv* env);
|
|
|
|
};
|
2013-02-19 23:48:33 +00:00
|
|
|
|
|
|
|
extern "C" int JNI_OnLoad(JavaVM*, void*);
|
|
|
|
|
|
|
|
#define _POSIX_C_SOURCE 200112L
|
|
|
|
#undef _GNU_SOURCE
|
2013-02-27 20:25:50 +00:00
|
|
|
#include "avian/machine.h"
|
|
|
|
#include "avian/classpath-common.h"
|
|
|
|
#include "avian/process.h"
|
2014-03-19 17:21:26 +00:00
|
|
|
#include "avian/util.h"
|
2013-02-19 16:36:19 +00:00
|
|
|
|
2014-02-26 22:56:17 +00:00
|
|
|
#ifdef PLATFORM_WINDOWS
|
2014-07-11 15:47:57 +00:00
|
|
|
const char* getErrnoDescription(
|
|
|
|
int err); // This function is defined in mingw-extensions.cpp
|
2014-02-26 22:56:17 +00:00
|
|
|
#endif
|
|
|
|
|
2013-02-19 16:36:19 +00:00
|
|
|
using namespace vm;
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_avian_Classes_defineVMClass(Thread*, object, uintptr_t*);
|
2013-07-05 20:34:16 +00:00
|
|
|
|
2013-02-19 16:36:19 +00:00
|
|
|
namespace {
|
|
|
|
|
|
|
|
namespace local {
|
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
void* getDirectBufferAddress(Thread* t, object b)
|
2013-03-14 21:33:05 +00:00
|
|
|
{
|
|
|
|
PROTECT(t, b);
|
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
GcField* field
|
|
|
|
= resolveField(t, objectClass(t, b), "effectiveDirectAddress", "J");
|
2013-03-14 21:33:05 +00:00
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
return reinterpret_cast<void*>(fieldAtOffset<int64_t>(b, field->offset()));
|
2013-03-14 21:33:05 +00:00
|
|
|
}
|
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
void JNICALL loadLibrary(Thread* t, object, uintptr_t* arguments)
|
2013-02-21 22:37:17 +00:00
|
|
|
{
|
2014-06-28 23:24:24 +00:00
|
|
|
GcString* name = cast<GcString>(t, reinterpret_cast<object>(arguments[1]));
|
2013-02-21 22:37:17 +00:00
|
|
|
|
2014-06-21 04:16:33 +00:00
|
|
|
Thread::LibraryLoadStack stack(
|
|
|
|
t, cast<GcClassLoader>(t, reinterpret_cast<object>(arguments[2])));
|
2014-07-01 16:18:45 +00:00
|
|
|
|
2014-06-28 23:24:24 +00:00
|
|
|
unsigned length = name->length(t);
|
2013-02-21 22:37:17 +00:00
|
|
|
THREAD_RUNTIME_ARRAY(t, char, n, length + 1);
|
|
|
|
stringChars(t, name, RUNTIME_ARRAY_BODY(n));
|
|
|
|
|
2014-04-07 19:25:57 +00:00
|
|
|
/* org_conscrypt_NativeCrypto.o is linked statically, and in Avian build
|
|
|
|
the package is named org.conscrypt.NativeCrypto. When Android code sees
|
|
|
|
that name it thinks the library isn't linked as a part of Android, so it
|
|
|
|
tries to load in dynamically, but there's actually no need to, so we
|
|
|
|
just ignore this request. */
|
|
|
|
if (strcmp(RUNTIME_ARRAY_BODY(n), "conscrypt_jni") != 0) {
|
|
|
|
loadLibrary(t, "", RUNTIME_ARRAY_BODY(n), true, true);
|
2014-04-07 19:29:23 +00:00
|
|
|
}
|
2013-02-21 22:37:17 +00:00
|
|
|
}
|
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
void JNICALL finalizeAllEnqueued(Thread*, object, uintptr_t*)
|
2013-04-23 19:47:15 +00:00
|
|
|
{
|
|
|
|
// ignore
|
|
|
|
}
|
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
int64_t JNICALL appLoader(Thread* t, object, uintptr_t*)
|
2013-02-21 22:37:17 +00:00
|
|
|
{
|
2014-06-30 01:44:41 +00:00
|
|
|
return reinterpret_cast<uintptr_t>(roots(t)->appLoader());
|
2013-02-21 22:37:17 +00:00
|
|
|
}
|
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
int64_t JNICALL defineClass(Thread* t, object method, uintptr_t* arguments)
|
2013-07-05 20:34:16 +00:00
|
|
|
{
|
2014-07-11 15:50:18 +00:00
|
|
|
uintptr_t args[] = {arguments[0], arguments[2], arguments[3], arguments[4]};
|
2013-07-05 20:34:16 +00:00
|
|
|
|
|
|
|
int64_t v = Avian_avian_Classes_defineVMClass(t, method, args);
|
|
|
|
|
|
|
|
if (v) {
|
2014-07-11 15:47:57 +00:00
|
|
|
return reinterpret_cast<uintptr_t>(
|
|
|
|
getJClass(t, cast<GcClass>(t, reinterpret_cast<object>(v))));
|
2013-07-05 20:34:16 +00:00
|
|
|
} else {
|
|
|
|
return 0;
|
2014-07-11 15:50:18 +00:00
|
|
|
}
|
2013-07-05 20:34:16 +00:00
|
|
|
}
|
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
int64_t JNICALL mapData(Thread*, object, uintptr_t*);
|
2013-02-23 00:23:59 +00:00
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
void JNICALL closeMemoryMappedFile(Thread*, GcMethod*, uintptr_t*);
|
2013-02-23 00:23:59 +00:00
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
object makeMethodOrConstructor(Thread* t, GcJclass* c, unsigned index)
|
2013-02-22 18:06:49 +00:00
|
|
|
{
|
|
|
|
PROTECT(t, c);
|
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
GcMethod* method = cast<GcMethod>(
|
|
|
|
t, cast<GcArray>(t, c->vmClass()->methodTable())->body()[index]);
|
2013-02-22 18:06:49 +00:00
|
|
|
PROTECT(t, method);
|
|
|
|
|
|
|
|
unsigned parameterCount;
|
|
|
|
unsigned returnTypeSpec;
|
2014-07-11 15:47:57 +00:00
|
|
|
object parameterTypes = resolveParameterJTypes(t,
|
|
|
|
method->class_()->loader(),
|
|
|
|
method->spec(),
|
|
|
|
¶meterCount,
|
|
|
|
&returnTypeSpec);
|
2013-02-22 18:06:49 +00:00
|
|
|
PROTECT(t, parameterTypes);
|
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
GcJclass* returnType = resolveJType(
|
|
|
|
t,
|
|
|
|
method->class_()->loader(),
|
|
|
|
reinterpret_cast<char*>(&method->spec()->body()[returnTypeSpec]),
|
|
|
|
method->spec()->length() - 1 - returnTypeSpec);
|
2013-02-22 18:06:49 +00:00
|
|
|
PROTECT(t, returnType);
|
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
object exceptionTypes = resolveExceptionJTypes(
|
|
|
|
t, method->class_()->loader(), method->addendum());
|
2013-02-22 18:06:49 +00:00
|
|
|
|
2014-06-28 21:11:31 +00:00
|
|
|
if (method->name()->body()[0] == '<') {
|
2014-07-11 15:50:18 +00:00
|
|
|
return makeJconstructor(
|
|
|
|
t, 0, c, parameterTypes, exceptionTypes, 0, 0, 0, 0, index);
|
2013-02-22 18:06:49 +00:00
|
|
|
} else {
|
|
|
|
PROTECT(t, exceptionTypes);
|
2014-07-11 15:47:57 +00:00
|
|
|
|
|
|
|
GcString* name = t->m->classpath->makeString(
|
|
|
|
t, method->name(), 0, method->name()->length() - 1);
|
2013-02-22 18:06:49 +00:00
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
return makeJmethod(t,
|
|
|
|
0,
|
|
|
|
index,
|
|
|
|
c,
|
|
|
|
name,
|
|
|
|
parameterTypes,
|
|
|
|
exceptionTypes,
|
|
|
|
returnType,
|
|
|
|
0,
|
|
|
|
0,
|
|
|
|
0,
|
|
|
|
0,
|
|
|
|
0);
|
2013-02-22 18:06:49 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
object makeField(Thread* t, GcJclass* c, unsigned index)
|
2013-02-22 18:06:49 +00:00
|
|
|
{
|
|
|
|
PROTECT(t, c);
|
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
GcField* field = cast<GcField>(
|
|
|
|
t, cast<GcArray>(t, c->vmClass()->fieldTable())->body()[index]);
|
2013-02-22 18:06:49 +00:00
|
|
|
|
|
|
|
PROTECT(t, field);
|
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
GcJclass* type = getJClass(
|
|
|
|
t,
|
|
|
|
resolveClassBySpec(t,
|
|
|
|
field->class_()->loader(),
|
|
|
|
reinterpret_cast<char*>(field->spec()->body().begin()),
|
|
|
|
field->spec()->length() - 1));
|
2013-02-22 18:06:49 +00:00
|
|
|
PROTECT(t, type);
|
2014-07-11 15:47:57 +00:00
|
|
|
|
|
|
|
GcString* name = t->m->classpath->makeString(
|
|
|
|
t, field->name(), 0, field->name()->length() - 1);
|
2013-02-22 18:06:49 +00:00
|
|
|
|
2014-07-02 21:11:27 +00:00
|
|
|
return makeJfield(t, 0, c, type, 0, 0, name, index);
|
2013-02-22 18:06:49 +00:00
|
|
|
}
|
|
|
|
|
2014-06-28 19:16:26 +00:00
|
|
|
void initVmThread(Thread* t, GcThread* thread, unsigned offset)
|
2013-02-22 18:06:49 +00:00
|
|
|
{
|
|
|
|
PROTECT(t, thread);
|
|
|
|
|
2014-02-25 19:33:12 +00:00
|
|
|
if (fieldAtOffset<object>(thread, offset) == 0) {
|
2014-07-11 15:47:57 +00:00
|
|
|
GcClass* c = resolveClass(t, roots(t)->bootLoader(), "java/lang/VMThread");
|
2013-02-22 18:06:49 +00:00
|
|
|
PROTECT(t, c);
|
|
|
|
|
|
|
|
object instance = makeNew(t, c);
|
|
|
|
PROTECT(t, instance);
|
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
GcMethod* constructor
|
|
|
|
= resolveMethod(t, c, "<init>", "(Ljava/lang/Thread;)V");
|
2013-02-22 18:06:49 +00:00
|
|
|
|
|
|
|
t->m->processor->invoke(t, constructor, instance, thread);
|
|
|
|
|
2014-07-02 21:11:27 +00:00
|
|
|
setField(t, thread, offset, instance);
|
2013-02-22 18:06:49 +00:00
|
|
|
}
|
2013-04-23 19:47:15 +00:00
|
|
|
|
2014-06-28 19:16:26 +00:00
|
|
|
if (thread->group() == 0) {
|
2014-07-10 23:51:38 +00:00
|
|
|
thread->setGroup(t, t->javaThread->group());
|
2014-06-28 19:16:26 +00:00
|
|
|
expect(t, thread->group());
|
2013-04-23 19:47:15 +00:00
|
|
|
}
|
2013-02-22 18:06:49 +00:00
|
|
|
}
|
|
|
|
|
2014-06-28 19:16:26 +00:00
|
|
|
void initVmThread(Thread* t, GcThread* thread)
|
2014-02-25 19:33:12 +00:00
|
|
|
{
|
2014-07-11 15:47:57 +00:00
|
|
|
initVmThread(t,
|
|
|
|
thread,
|
|
|
|
resolveField(t,
|
|
|
|
objectClass(t, thread),
|
|
|
|
"vmThread",
|
|
|
|
"Ljava/lang/VMThread;")->offset());
|
2014-02-25 19:33:12 +00:00
|
|
|
}
|
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
object translateStackTrace(Thread* t, object raw)
|
2013-02-22 18:06:49 +00:00
|
|
|
{
|
|
|
|
PROTECT(t, raw);
|
2014-07-11 15:47:57 +00:00
|
|
|
|
|
|
|
object array = makeObjectArray(
|
|
|
|
t,
|
|
|
|
resolveClass(t, roots(t)->bootLoader(), "java/lang/StackTraceElement"),
|
|
|
|
objectArrayLength(t, raw));
|
2013-02-22 18:06:49 +00:00
|
|
|
PROTECT(t, array);
|
|
|
|
|
|
|
|
for (unsigned i = 0; i < objectArrayLength(t, array); ++i) {
|
2014-07-11 15:47:57 +00:00
|
|
|
GcStackTraceElement* e = makeStackTraceElement(
|
|
|
|
t, cast<GcTraceElement>(t, objectArrayBody(t, raw, i)));
|
2013-02-22 18:06:49 +00:00
|
|
|
|
2014-07-02 21:11:27 +00:00
|
|
|
setField(t, array, ArrayBody + (i * BytesPerWord), e);
|
2013-02-22 18:06:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return array;
|
|
|
|
}
|
|
|
|
|
2013-02-19 16:36:19 +00:00
|
|
|
class MyClasspath : public Classpath {
|
|
|
|
public:
|
2014-04-12 01:38:11 +00:00
|
|
|
MyClasspath(Allocator* allocator)
|
|
|
|
: allocator(allocator), tzdata(0), mayInitClasses_(false)
|
2014-07-11 15:50:18 +00:00
|
|
|
{
|
|
|
|
}
|
2013-02-19 16:36:19 +00:00
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
virtual GcJclass* makeJclass(Thread* t, GcClass* class_)
|
2013-02-19 16:36:19 +00:00
|
|
|
{
|
|
|
|
PROTECT(t, class_);
|
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
GcJclass* c
|
|
|
|
= reinterpret_cast<GcJclass*>(allocate(t, GcJclass::FixedSize, true));
|
2014-07-02 21:11:27 +00:00
|
|
|
setObjectClass(t, c, type(t, GcJclass::Type));
|
2014-07-10 23:51:38 +00:00
|
|
|
c->setVmClass(t, class_);
|
2013-02-19 16:36:19 +00:00
|
|
|
|
2014-06-26 02:17:27 +00:00
|
|
|
return c;
|
2013-02-19 16:36:19 +00:00
|
|
|
}
|
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
virtual GcString* makeString(Thread* t,
|
|
|
|
object array,
|
|
|
|
int32_t offset,
|
|
|
|
int32_t length)
|
2013-02-19 16:36:19 +00:00
|
|
|
{
|
2014-05-29 04:17:25 +00:00
|
|
|
if (objectClass(t, array) == type(t, GcByteArray::Type)) {
|
2014-06-27 00:17:46 +00:00
|
|
|
GcByteArray* byteArray = cast<GcByteArray>(t, array);
|
2013-02-19 16:36:19 +00:00
|
|
|
PROTECT(t, array);
|
2014-06-27 00:17:46 +00:00
|
|
|
PROTECT(t, byteArray);
|
2014-07-11 15:47:57 +00:00
|
|
|
|
2014-06-27 00:17:46 +00:00
|
|
|
GcCharArray* charArray = makeCharArray(t, length);
|
2013-02-19 16:36:19 +00:00
|
|
|
for (int i = 0; i < length; ++i) {
|
2014-06-27 00:17:46 +00:00
|
|
|
expect(t, (byteArray->body()[offset + i] & 0x80) == 0);
|
2013-02-19 16:36:19 +00:00
|
|
|
|
2014-06-27 00:17:46 +00:00
|
|
|
charArray->body()[i] = byteArray->body()[offset + i];
|
2013-02-19 16:36:19 +00:00
|
|
|
}
|
|
|
|
|
2014-07-02 21:11:27 +00:00
|
|
|
array = charArray;
|
2013-02-19 16:36:19 +00:00
|
|
|
} else {
|
2014-05-29 04:17:25 +00:00
|
|
|
expect(t, objectClass(t, array) == type(t, GcCharArray::Type));
|
2013-02-19 16:36:19 +00:00
|
|
|
}
|
|
|
|
|
2014-06-28 23:24:24 +00:00
|
|
|
return vm::makeString(t, array, offset, length, 0);
|
2013-02-19 16:36:19 +00:00
|
|
|
}
|
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
virtual GcThread* makeThread(Thread* t, Thread* parent)
|
2013-02-19 16:36:19 +00:00
|
|
|
{
|
|
|
|
const unsigned NormalPriority = 5;
|
|
|
|
|
2014-06-28 23:24:24 +00:00
|
|
|
GcThreadGroup* group = 0;
|
2013-02-22 18:06:49 +00:00
|
|
|
PROTECT(t, group);
|
2013-02-19 16:36:19 +00:00
|
|
|
if (parent) {
|
2014-06-28 23:24:24 +00:00
|
|
|
group = parent->javaThread->group();
|
2013-02-19 16:36:19 +00:00
|
|
|
} else {
|
2014-07-11 15:47:57 +00:00
|
|
|
resolveSystemClass(t,
|
|
|
|
roots(t)->bootLoader(),
|
|
|
|
type(t, GcThreadGroup::Type)->name(),
|
|
|
|
false);
|
2013-02-19 16:36:19 +00:00
|
|
|
|
2014-06-28 23:24:24 +00:00
|
|
|
group = cast<GcThreadGroup>(t, makeNew(t, type(t, GcThreadGroup::Type)));
|
2013-02-22 18:06:49 +00:00
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
GcMethod* constructor
|
|
|
|
= resolveMethod(t, type(t, GcThreadGroup::Type), "<init>", "()V");
|
2013-02-21 22:37:17 +00:00
|
|
|
|
2014-07-02 21:11:27 +00:00
|
|
|
t->m->processor->invoke(t, constructor, group);
|
2013-02-22 18:06:49 +00:00
|
|
|
}
|
2013-02-21 22:37:17 +00:00
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
resolveSystemClass(
|
|
|
|
t, roots(t)->bootLoader(), type(t, GcThread::Type)->name(), false);
|
|
|
|
|
2014-06-28 19:16:26 +00:00
|
|
|
GcThread* thread = cast<GcThread>(t, makeNew(t, type(t, GcThread::Type)));
|
2013-02-22 18:06:49 +00:00
|
|
|
PROTECT(t, thread);
|
2013-02-21 22:37:17 +00:00
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
GcMethod* constructor
|
|
|
|
= resolveMethod(t,
|
|
|
|
type(t, GcThread::Type),
|
|
|
|
"<init>",
|
|
|
|
"(Ljava/lang/ThreadGroup;Ljava/lang/String;IZ)V");
|
2013-02-21 22:37:17 +00:00
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
t->m->processor->invoke(
|
|
|
|
t, constructor, thread, group, 0, NormalPriority, false);
|
2013-02-21 22:37:17 +00:00
|
|
|
|
2014-07-10 23:51:38 +00:00
|
|
|
thread->setContextClassLoader(t, roots(t)->appLoader());
|
2013-02-26 23:24:02 +00:00
|
|
|
|
2013-02-22 18:06:49 +00:00
|
|
|
initVmThread(t, thread);
|
2013-02-19 16:36:19 +00:00
|
|
|
|
2014-06-28 23:24:24 +00:00
|
|
|
return thread;
|
2013-02-19 16:36:19 +00:00
|
|
|
}
|
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
virtual object makeJMethod(Thread* t, GcMethod* vmMethod)
|
2013-02-19 16:36:19 +00:00
|
|
|
{
|
2014-06-27 00:17:46 +00:00
|
|
|
GcArray* table = cast<GcArray>(t, vmMethod->class_()->methodTable());
|
|
|
|
for (unsigned i = 0; i < table->length(); ++i) {
|
2014-07-02 21:11:27 +00:00
|
|
|
if (vmMethod == table->body()[i]) {
|
2014-07-11 15:47:57 +00:00
|
|
|
return makeMethodOrConstructor(t, getJClass(t, vmMethod->class_()), i);
|
2013-02-22 18:06:49 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
abort(t);
|
2013-02-19 16:36:19 +00:00
|
|
|
}
|
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
virtual GcMethod* getVMMethod(Thread* t, object jmethod)
|
2013-02-19 16:36:19 +00:00
|
|
|
{
|
2014-07-11 15:47:57 +00:00
|
|
|
return cast<GcMethod>(
|
|
|
|
t,
|
|
|
|
objectClass(t, jmethod) == type(t, GcJmethod::Type)
|
|
|
|
? cast<GcArray>(t,
|
|
|
|
cast<GcJmethod>(t, jmethod)
|
|
|
|
->declaringClass()
|
|
|
|
->vmClass()
|
|
|
|
->methodTable())
|
|
|
|
->body()[cast<GcJmethod>(t, jmethod)->slot()]
|
|
|
|
: cast<GcArray>(t,
|
|
|
|
cast<GcJconstructor>(t, jmethod)
|
|
|
|
->declaringClass()
|
|
|
|
->vmClass()
|
|
|
|
->methodTable())
|
|
|
|
->body()[cast<GcJconstructor>(t, jmethod)->slot()]);
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual object makeJField(Thread* t, GcField* vmField)
|
2013-02-19 16:36:19 +00:00
|
|
|
{
|
2014-06-27 00:17:46 +00:00
|
|
|
GcArray* table = cast<GcArray>(t, vmField->class_()->fieldTable());
|
|
|
|
for (unsigned i = 0; i < table->length(); ++i) {
|
2014-07-02 21:11:27 +00:00
|
|
|
if (vmField == table->body()[i]) {
|
2014-06-28 23:24:24 +00:00
|
|
|
return makeField(t, getJClass(t, vmField->class_()), i);
|
2013-02-22 18:06:49 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
abort(t);
|
2013-02-19 16:36:19 +00:00
|
|
|
}
|
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
virtual GcField* getVMField(Thread* t, GcJfield* jfield)
|
2013-02-19 16:36:19 +00:00
|
|
|
{
|
2014-07-11 15:47:57 +00:00
|
|
|
return cast<GcField>(
|
|
|
|
t,
|
|
|
|
cast<GcArray>(t, jfield->declaringClass()->vmClass()->fieldTable())
|
|
|
|
->body()[jfield->slot()]);
|
2013-02-19 16:36:19 +00:00
|
|
|
}
|
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
virtual void clearInterrupted(Thread*)
|
2013-02-19 16:36:19 +00:00
|
|
|
{
|
|
|
|
// ignore
|
|
|
|
}
|
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
virtual void runThread(Thread* t)
|
2013-02-19 16:36:19 +00:00
|
|
|
{
|
2013-02-22 18:06:49 +00:00
|
|
|
// force monitor creation so we don't get an OutOfMemory error
|
|
|
|
// later when we try to acquire it:
|
2014-07-02 21:11:27 +00:00
|
|
|
objectMonitor(t, t->javaThread, true);
|
2013-02-22 18:06:49 +00:00
|
|
|
|
2014-06-28 23:24:24 +00:00
|
|
|
GcField* field = resolveField(
|
|
|
|
t, objectClass(t, t->javaThread), "vmThread", "Ljava/lang/VMThread;");
|
2014-02-25 19:33:12 +00:00
|
|
|
|
2014-06-28 21:11:31 +00:00
|
|
|
unsigned offset = field->offset();
|
2014-02-25 19:33:12 +00:00
|
|
|
|
|
|
|
THREAD_RESOURCE(t, unsigned, offset, {
|
|
|
|
object vmt = fieldAtOffset<object>(t->javaThread, offset);
|
|
|
|
if (vmt) {
|
|
|
|
PROTECT(t, vmt);
|
|
|
|
vm::acquire(t, vmt);
|
|
|
|
fieldAtOffset<object>(t->javaThread, offset) = 0;
|
|
|
|
vm::notifyAll(t, vmt);
|
|
|
|
vm::release(t, vmt);
|
|
|
|
}
|
|
|
|
|
2014-07-02 21:11:27 +00:00
|
|
|
vm::acquire(t, t->javaThread);
|
2014-02-25 19:33:12 +00:00
|
|
|
t->flags &= ~Thread::ActiveFlag;
|
2014-07-02 21:11:27 +00:00
|
|
|
vm::notifyAll(t, t->javaThread);
|
|
|
|
vm::release(t, t->javaThread);
|
2013-02-22 18:06:49 +00:00
|
|
|
});
|
|
|
|
|
2014-06-28 23:24:24 +00:00
|
|
|
initVmThread(t, t->javaThread, offset);
|
2013-02-22 18:06:49 +00:00
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
GcMethod* method = resolveMethod(
|
|
|
|
t, roots(t)->bootLoader(), "java/lang/Thread", "run", "()V");
|
2013-02-19 16:36:19 +00:00
|
|
|
|
2014-07-02 21:11:27 +00:00
|
|
|
t->m->processor->invoke(t, method, t->javaThread);
|
2013-02-19 16:36:19 +00:00
|
|
|
}
|
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
virtual void resolveNative(Thread* t, GcMethod* method)
|
2013-02-19 16:36:19 +00:00
|
|
|
{
|
|
|
|
vm::resolveNative(t, method);
|
|
|
|
}
|
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
void interceptMethods(Thread* t, bool updateRuntimeData)
|
2013-02-19 16:36:19 +00:00
|
|
|
{
|
2014-07-11 15:47:57 +00:00
|
|
|
{
|
|
|
|
GcClass* c
|
|
|
|
= resolveClass(t, roots(t)->bootLoader(), "java/lang/Runtime", false);
|
2013-02-21 22:37:17 +00:00
|
|
|
|
2013-02-23 00:23:59 +00:00
|
|
|
if (c) {
|
|
|
|
PROTECT(t, c);
|
2013-02-21 22:37:17 +00:00
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
intercept(t,
|
|
|
|
c,
|
|
|
|
"loadLibrary",
|
2013-02-21 22:37:17 +00:00
|
|
|
"(Ljava/lang/String;Ljava/lang/ClassLoader;)V",
|
2014-07-11 15:50:18 +00:00
|
|
|
voidPointer(loadLibrary),
|
|
|
|
updateRuntimeData);
|
2013-02-21 22:37:17 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
{
|
|
|
|
GcClass* c = resolveClass(
|
|
|
|
t, roots(t)->bootLoader(), "java/lang/ref/FinalizerReference", false);
|
2013-04-23 19:47:15 +00:00
|
|
|
|
|
|
|
if (c) {
|
|
|
|
PROTECT(t, c);
|
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
intercept(t,
|
|
|
|
c,
|
|
|
|
"finalizeAllEnqueued",
|
|
|
|
"()V",
|
|
|
|
voidPointer(finalizeAllEnqueued),
|
|
|
|
updateRuntimeData);
|
2013-04-23 19:47:15 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
{
|
|
|
|
GcClass* c = resolveClass(
|
|
|
|
t, roots(t)->bootLoader(), "java/lang/ClassLoader", false);
|
2013-02-21 22:37:17 +00:00
|
|
|
|
2013-02-23 00:23:59 +00:00
|
|
|
if (c) {
|
|
|
|
PROTECT(t, c);
|
2013-02-21 22:37:17 +00:00
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
intercept(t,
|
|
|
|
c,
|
|
|
|
"createSystemClassLoader",
|
|
|
|
"()Ljava/lang/ClassLoader;",
|
|
|
|
voidPointer(appLoader),
|
|
|
|
updateRuntimeData);
|
2013-07-05 20:34:16 +00:00
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
intercept(t,
|
|
|
|
c,
|
|
|
|
"defineClass",
|
2013-07-05 20:34:16 +00:00
|
|
|
"(Ljava/lang/String;[BII)Ljava/lang/Class;",
|
2014-07-11 15:50:18 +00:00
|
|
|
voidPointer(defineClass),
|
|
|
|
updateRuntimeData);
|
2013-02-21 22:37:17 +00:00
|
|
|
}
|
|
|
|
}
|
2013-02-23 00:23:59 +00:00
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
{
|
|
|
|
GcClass* c = resolveClass(
|
|
|
|
t, roots(t)->bootLoader(), "libcore/util/ZoneInfoDB", false);
|
2013-02-23 00:23:59 +00:00
|
|
|
|
|
|
|
if (c) {
|
|
|
|
PROTECT(t, c);
|
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
intercept(t,
|
|
|
|
c,
|
|
|
|
"mapData",
|
|
|
|
"()Llibcore/io/MemoryMappedFile;",
|
|
|
|
voidPointer(mapData),
|
|
|
|
updateRuntimeData);
|
2013-02-23 00:23:59 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
{
|
|
|
|
GcClass* c = resolveClass(
|
|
|
|
t, roots(t)->bootLoader(), "libcore/io/MemoryMappedFile", false);
|
2013-02-23 00:23:59 +00:00
|
|
|
|
|
|
|
if (c) {
|
|
|
|
PROTECT(t, c);
|
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
intercept(t,
|
|
|
|
c,
|
|
|
|
"close",
|
|
|
|
"()V",
|
|
|
|
voidPointer(closeMemoryMappedFile),
|
2013-03-15 19:28:01 +00:00
|
|
|
updateRuntimeData);
|
2013-02-23 00:23:59 +00:00
|
|
|
}
|
|
|
|
}
|
2013-03-15 19:28:01 +00:00
|
|
|
}
|
2013-03-14 21:33:05 +00:00
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
virtual void interceptMethods(Thread* t)
|
2013-03-15 19:28:01 +00:00
|
|
|
{
|
|
|
|
interceptMethods(t, false);
|
|
|
|
}
|
2013-03-14 21:33:05 +00:00
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
virtual void preBoot(Thread* t)
|
2013-03-15 19:28:01 +00:00
|
|
|
{
|
2014-07-11 15:50:18 +00:00
|
|
|
// Android's System.initSystemProperties throws an NPE if
|
|
|
|
// LD_LIBRARY_PATH is not set as of this writing:
|
2013-03-15 19:28:01 +00:00
|
|
|
#ifdef PLATFORM_WINDOWS
|
|
|
|
_wputenv(L"LD_LIBRARY_PATH=(dummy)");
|
2014-07-11 15:50:18 +00:00
|
|
|
#elif(!defined AVIAN_IOS)
|
2013-03-15 19:28:01 +00:00
|
|
|
setenv("LD_LIBRARY_PATH", "", false);
|
|
|
|
#endif
|
2014-07-11 15:50:18 +00:00
|
|
|
|
2013-03-15 19:28:01 +00:00
|
|
|
interceptMethods(t, true);
|
|
|
|
|
2013-07-05 20:34:16 +00:00
|
|
|
JniConstants::init(reinterpret_cast<_JNIEnv*>(t));
|
|
|
|
|
2013-02-19 23:48:33 +00:00
|
|
|
JNI_OnLoad(reinterpret_cast< ::JavaVM*>(t->m), 0);
|
2014-04-12 01:38:11 +00:00
|
|
|
|
|
|
|
mayInitClasses_ = true;
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual bool mayInitClasses()
|
|
|
|
{
|
|
|
|
return mayInitClasses_;
|
2013-02-19 16:36:19 +00:00
|
|
|
}
|
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
virtual void boot(Thread* t)
|
2013-02-22 21:41:24 +00:00
|
|
|
{
|
2014-07-11 15:47:57 +00:00
|
|
|
GcClass* c
|
|
|
|
= resolveClass(t, roots(t)->bootLoader(), "java/lang/ClassLoader");
|
2014-03-19 17:21:26 +00:00
|
|
|
PROTECT(t, c);
|
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
GcMethod* constructor
|
|
|
|
= resolveMethod(t, c, "<init>", "(Ljava/lang/ClassLoader;Z)V");
|
2014-03-19 17:21:26 +00:00
|
|
|
PROTECT(t, constructor);
|
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
t->m->processor->invoke(t, constructor, roots(t)->bootLoader(), 0, true);
|
2014-03-19 17:21:26 +00:00
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
t->m->processor->invoke(
|
|
|
|
t, constructor, roots(t)->appLoader(), roots(t)->bootLoader(), false);
|
2013-02-22 21:41:24 +00:00
|
|
|
}
|
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
virtual const char* bootClasspath()
|
2013-02-19 16:36:19 +00:00
|
|
|
{
|
|
|
|
return AVIAN_CLASSPATH;
|
|
|
|
}
|
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
virtual object makeDirectByteBuffer(Thread* t, void* p, jlong capacity)
|
2013-02-21 22:37:17 +00:00
|
|
|
{
|
2014-07-11 15:47:57 +00:00
|
|
|
GcClass* c
|
|
|
|
= resolveClass(t, roots(t)->bootLoader(), "java/nio/DirectByteBuffer");
|
2013-02-21 22:37:17 +00:00
|
|
|
PROTECT(t, c);
|
|
|
|
|
|
|
|
object instance = makeNew(t, c);
|
|
|
|
PROTECT(t, instance);
|
|
|
|
|
2014-05-29 04:17:25 +00:00
|
|
|
GcMethod* constructor = resolveMethod(t, c, "<init>", "(JI)V");
|
2013-02-21 22:37:17 +00:00
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
t->m->processor->invoke(t,
|
|
|
|
constructor,
|
|
|
|
instance,
|
|
|
|
reinterpret_cast<int64_t>(p),
|
|
|
|
static_cast<int>(capacity));
|
2013-02-21 22:37:17 +00:00
|
|
|
|
|
|
|
return instance;
|
|
|
|
}
|
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
virtual void* getDirectBufferAddress(Thread* t, object b)
|
2013-02-21 22:37:17 +00:00
|
|
|
{
|
2013-03-14 21:33:05 +00:00
|
|
|
return local::getDirectBufferAddress(t, b);
|
2013-02-21 22:37:17 +00:00
|
|
|
}
|
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
virtual int64_t getDirectBufferCapacity(Thread* t, object b)
|
2013-02-21 22:37:17 +00:00
|
|
|
{
|
|
|
|
PROTECT(t, b);
|
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
GcField* field = resolveField(t, objectClass(t, b), "capacity", "I");
|
2013-02-21 22:37:17 +00:00
|
|
|
|
2014-06-28 21:11:31 +00:00
|
|
|
return fieldAtOffset<int32_t>(b, field->offset());
|
2013-02-21 22:37:17 +00:00
|
|
|
}
|
|
|
|
|
2014-07-12 22:03:11 +00:00
|
|
|
virtual bool canTailCall(Thread* t UNUSED,
|
2014-07-11 15:47:57 +00:00
|
|
|
GcMethod*,
|
2014-07-12 22:03:11 +00:00
|
|
|
GcByteArray* calleeClassName,
|
|
|
|
GcByteArray* calleeMethodName,
|
2014-07-11 15:47:57 +00:00
|
|
|
GcByteArray*)
|
2013-03-05 22:43:49 +00:00
|
|
|
{
|
2014-07-12 22:03:11 +00:00
|
|
|
// we can't tail call System.load[Library] or
|
|
|
|
// Runtime.load[Library] due to their use of
|
|
|
|
// ClassLoader.getCaller, which gets confused if we elide stack
|
|
|
|
// frames.
|
|
|
|
|
|
|
|
return (
|
|
|
|
(strcmp("loadLibrary",
|
|
|
|
reinterpret_cast<char*>(calleeMethodName->body().begin()))
|
|
|
|
and strcmp("load",
|
|
|
|
reinterpret_cast<char*>(calleeMethodName->body().begin())))
|
|
|
|
or (strcmp("java/lang/System",
|
|
|
|
reinterpret_cast<char*>(calleeClassName->body().begin()))
|
|
|
|
and strcmp(
|
|
|
|
"java/lang/Runtime",
|
|
|
|
reinterpret_cast<char*>(calleeClassName->body().begin()))));
|
2013-03-05 22:43:49 +00:00
|
|
|
}
|
|
|
|
|
2014-06-21 04:16:33 +00:00
|
|
|
virtual GcClassLoader* libraryClassLoader(Thread* t, GcMethod* caller)
|
2014-07-01 16:18:45 +00:00
|
|
|
{
|
2014-07-11 15:47:57 +00:00
|
|
|
return strcmp("java/lang/Runtime",
|
|
|
|
reinterpret_cast<char*>(
|
|
|
|
caller->class_()->name()->body().begin())) == 0
|
2014-07-01 16:18:45 +00:00
|
|
|
? t->libraryLoadStack->classLoader
|
2014-06-21 04:16:33 +00:00
|
|
|
: caller->class_()->loader();
|
2014-07-01 16:18:45 +00:00
|
|
|
}
|
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
virtual void shutDown(Thread*)
|
2013-04-19 19:00:47 +00:00
|
|
|
{
|
|
|
|
// ignore
|
|
|
|
}
|
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
virtual void dispose()
|
2013-02-19 16:36:19 +00:00
|
|
|
{
|
2013-02-23 00:23:59 +00:00
|
|
|
if (tzdata) {
|
|
|
|
tzdata->dispose();
|
|
|
|
}
|
2013-02-19 16:36:19 +00:00
|
|
|
allocator->free(this, sizeof(*this));
|
|
|
|
}
|
|
|
|
|
|
|
|
Allocator* allocator;
|
2013-02-23 00:23:59 +00:00
|
|
|
System::Region* tzdata;
|
2014-04-12 01:38:11 +00:00
|
|
|
bool mayInitClasses_;
|
2013-02-19 16:36:19 +00:00
|
|
|
};
|
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
int64_t JNICALL mapData(Thread* t, object, uintptr_t*)
|
2013-02-23 00:23:59 +00:00
|
|
|
{
|
2014-07-11 15:47:57 +00:00
|
|
|
GcClass* c
|
|
|
|
= resolveClass(t, roots(t)->bootLoader(), "libcore/io/MemoryMappedFile");
|
2013-02-23 00:23:59 +00:00
|
|
|
PROTECT(t, c);
|
2014-07-11 15:50:18 +00:00
|
|
|
|
2013-02-23 00:23:59 +00:00
|
|
|
object instance = makeNew(t, c);
|
|
|
|
PROTECT(t, instance);
|
2014-07-11 15:47:57 +00:00
|
|
|
|
2014-05-29 04:17:25 +00:00
|
|
|
GcMethod* constructor = resolveMethod(t, c, "<init>", "(JJ)V");
|
2014-07-11 15:47:57 +00:00
|
|
|
|
2013-02-23 00:23:59 +00:00
|
|
|
const char* jar = "javahomeJar";
|
|
|
|
Finder* finder = getFinder(t, jar, strlen(jar));
|
|
|
|
if (finder) {
|
|
|
|
System::Region* r = finder->find("tzdata");
|
|
|
|
if (r) {
|
|
|
|
MyClasspath* cp = static_cast<MyClasspath*>(t->m->classpath);
|
|
|
|
|
|
|
|
expect(t, cp->tzdata == 0);
|
|
|
|
|
|
|
|
cp->tzdata = r;
|
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
t->m->processor->invoke(t,
|
|
|
|
constructor,
|
|
|
|
instance,
|
|
|
|
reinterpret_cast<int64_t>(r->start()),
|
|
|
|
static_cast<int64_t>(r->length()));
|
2013-02-23 00:23:59 +00:00
|
|
|
|
|
|
|
return reinterpret_cast<uintptr_t>(instance);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-05-29 04:17:25 +00:00
|
|
|
throwNew(t, GcRuntimeException::Type);
|
2013-02-23 00:23:59 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void JNICALL
|
2014-07-11 15:47:57 +00:00
|
|
|
closeMemoryMappedFile(Thread* t, GcMethod* method, uintptr_t* arguments)
|
2013-02-23 00:23:59 +00:00
|
|
|
{
|
|
|
|
object file = reinterpret_cast<object>(arguments[0]);
|
|
|
|
PROTECT(t, file);
|
|
|
|
|
|
|
|
MyClasspath* cp = static_cast<MyClasspath*>(t->m->classpath);
|
|
|
|
|
|
|
|
if (cp->tzdata) {
|
2014-06-28 23:24:24 +00:00
|
|
|
GcField* field = resolveField(t, objectClass(t, file), "address", "J");
|
2014-07-11 15:47:57 +00:00
|
|
|
|
2014-06-28 21:11:31 +00:00
|
|
|
if (fieldAtOffset<int64_t>(file, field->offset())
|
2014-07-11 15:47:57 +00:00
|
|
|
== reinterpret_cast<int64_t>(cp->tzdata->start())) {
|
2013-02-23 00:23:59 +00:00
|
|
|
cp->tzdata->dispose();
|
|
|
|
cp->tzdata = 0;
|
|
|
|
|
2014-06-28 21:11:31 +00:00
|
|
|
fieldAtOffset<int64_t>(file, field->offset()) = 0;
|
2013-02-23 00:23:59 +00:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
t->m->processor->invoke(t,
|
|
|
|
cast<GcMethod>(t,
|
|
|
|
getMethodRuntimeData(t, method)
|
|
|
|
->native()
|
|
|
|
->as<GcNativeIntercept>(t)
|
|
|
|
->original()),
|
|
|
|
file);
|
2013-02-23 00:23:59 +00:00
|
|
|
}
|
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
bool matchType(Thread* t, GcField* field, object o)
|
2013-12-06 22:45:46 +00:00
|
|
|
{
|
2014-06-28 21:11:31 +00:00
|
|
|
switch (field->code()) {
|
2013-12-06 22:45:46 +00:00
|
|
|
case ByteField:
|
2014-05-29 04:17:25 +00:00
|
|
|
return objectClass(t, o) == type(t, GcByte::Type);
|
2013-12-06 22:45:46 +00:00
|
|
|
|
|
|
|
case BooleanField:
|
2014-05-29 04:17:25 +00:00
|
|
|
return objectClass(t, o) == type(t, GcBoolean::Type);
|
2013-12-06 22:45:46 +00:00
|
|
|
|
|
|
|
case CharField:
|
2014-05-29 04:17:25 +00:00
|
|
|
return objectClass(t, o) == type(t, GcChar::Type);
|
2013-12-06 22:45:46 +00:00
|
|
|
|
|
|
|
case ShortField:
|
2014-05-29 04:17:25 +00:00
|
|
|
return objectClass(t, o) == type(t, GcShort::Type);
|
2013-12-06 22:45:46 +00:00
|
|
|
|
|
|
|
case IntField:
|
2014-05-29 04:17:25 +00:00
|
|
|
return objectClass(t, o) == type(t, GcInt::Type);
|
2013-12-06 22:45:46 +00:00
|
|
|
|
|
|
|
case LongField:
|
2014-05-29 04:17:25 +00:00
|
|
|
return objectClass(t, o) == type(t, GcLong::Type);
|
2013-12-06 22:45:46 +00:00
|
|
|
|
|
|
|
case FloatField:
|
2014-05-29 04:17:25 +00:00
|
|
|
return objectClass(t, o) == type(t, GcFloat::Type);
|
2013-12-06 22:45:46 +00:00
|
|
|
|
|
|
|
case DoubleField:
|
2014-05-29 04:17:25 +00:00
|
|
|
return objectClass(t, o) == type(t, GcDouble::Type);
|
2013-12-06 22:45:46 +00:00
|
|
|
|
|
|
|
case ObjectField:
|
|
|
|
if (o == 0) {
|
|
|
|
return true;
|
|
|
|
} else {
|
|
|
|
PROTECT(t, o);
|
|
|
|
|
2014-06-28 21:11:31 +00:00
|
|
|
GcByteArray* spec;
|
|
|
|
if (field->spec()->body()[0] == '[') {
|
|
|
|
spec = field->spec();
|
2013-12-06 22:45:46 +00:00
|
|
|
} else {
|
2014-06-28 21:11:31 +00:00
|
|
|
spec = makeByteArray(t, field->spec()->length() - 2);
|
2014-07-11 15:47:57 +00:00
|
|
|
|
2014-06-28 21:11:31 +00:00
|
|
|
memcpy(spec->body().begin(),
|
|
|
|
&field->spec()->body()[1],
|
|
|
|
field->spec()->length() - 3);
|
2013-12-06 22:45:46 +00:00
|
|
|
|
2014-06-28 21:11:31 +00:00
|
|
|
spec->body()[field->spec()->length() - 3] = 0;
|
2013-12-06 22:45:46 +00:00
|
|
|
}
|
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
return instanceOf(t, resolveClass(t, field->class_()->loader(), spec), o);
|
2013-12-06 22:45:46 +00:00
|
|
|
}
|
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
default:
|
|
|
|
abort(t);
|
2013-12-06 22:45:46 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
object getField(Thread* t, GcField* field, object instance)
|
2013-12-06 22:45:46 +00:00
|
|
|
{
|
|
|
|
PROTECT(t, field);
|
|
|
|
PROTECT(t, instance);
|
|
|
|
|
2014-06-28 21:11:31 +00:00
|
|
|
initClass(t, field->class_());
|
2013-12-06 22:45:46 +00:00
|
|
|
|
|
|
|
object target;
|
2014-06-28 21:11:31 +00:00
|
|
|
if (field->flags() & ACC_STATIC) {
|
2014-07-02 21:11:27 +00:00
|
|
|
target = field->class_()->staticTable();
|
2014-07-11 15:47:57 +00:00
|
|
|
} else if (instanceOf(t, field->class_(), instance)) {
|
2013-12-06 22:45:46 +00:00
|
|
|
target = instance;
|
|
|
|
} else {
|
2014-05-29 04:17:25 +00:00
|
|
|
throwNew(t, GcIllegalArgumentException::Type);
|
2013-12-06 22:45:46 +00:00
|
|
|
}
|
|
|
|
|
2014-06-28 21:11:31 +00:00
|
|
|
unsigned offset = field->offset();
|
|
|
|
switch (field->code()) {
|
2013-12-06 22:45:46 +00:00
|
|
|
case ByteField:
|
2014-07-02 21:11:27 +00:00
|
|
|
return makeByte(t, fieldAtOffset<int8_t>(target, offset));
|
2013-12-06 22:45:46 +00:00
|
|
|
|
|
|
|
case BooleanField:
|
2014-07-02 21:11:27 +00:00
|
|
|
return makeBoolean(t, fieldAtOffset<int8_t>(target, offset));
|
2013-12-06 22:45:46 +00:00
|
|
|
|
|
|
|
case CharField:
|
2014-07-02 21:11:27 +00:00
|
|
|
return makeChar(t, fieldAtOffset<int16_t>(target, offset));
|
2013-12-06 22:45:46 +00:00
|
|
|
|
|
|
|
case ShortField:
|
2014-07-02 21:11:27 +00:00
|
|
|
return makeShort(t, fieldAtOffset<int16_t>(target, offset));
|
2013-12-06 22:45:46 +00:00
|
|
|
|
|
|
|
case IntField:
|
2014-07-02 21:11:27 +00:00
|
|
|
return makeInt(t, fieldAtOffset<int32_t>(target, offset));
|
2013-12-06 22:45:46 +00:00
|
|
|
|
|
|
|
case LongField:
|
2014-07-02 21:11:27 +00:00
|
|
|
return makeLong(t, fieldAtOffset<int64_t>(target, offset));
|
2013-12-06 22:45:46 +00:00
|
|
|
|
|
|
|
case FloatField:
|
2014-07-02 21:11:27 +00:00
|
|
|
return makeFloat(t, fieldAtOffset<int32_t>(target, offset));
|
2013-12-06 22:45:46 +00:00
|
|
|
|
|
|
|
case DoubleField:
|
2014-07-02 21:11:27 +00:00
|
|
|
return makeDouble(t, fieldAtOffset<int64_t>(target, offset));
|
2013-12-06 22:45:46 +00:00
|
|
|
|
|
|
|
case ObjectField:
|
|
|
|
return fieldAtOffset<object>(target, offset);
|
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
default:
|
|
|
|
abort(t);
|
2013-12-06 22:45:46 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
void setField(Thread* t, GcField* field, object instance, object value)
|
2013-12-06 22:45:46 +00:00
|
|
|
{
|
|
|
|
PROTECT(t, field);
|
|
|
|
PROTECT(t, instance);
|
|
|
|
PROTECT(t, value);
|
|
|
|
|
|
|
|
if (not matchType(t, field, value)) {
|
2014-05-29 04:17:25 +00:00
|
|
|
throwNew(t, GcIllegalArgumentException::Type);
|
2013-12-06 22:45:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
object target;
|
2014-06-28 21:11:31 +00:00
|
|
|
if ((field->flags() & ACC_STATIC) != 0) {
|
2014-07-02 21:11:27 +00:00
|
|
|
target = field->class_()->staticTable();
|
2014-07-11 15:47:57 +00:00
|
|
|
} else if (instanceOf(t, field->class_(), instance)) {
|
2013-12-06 22:45:46 +00:00
|
|
|
target = instance;
|
|
|
|
} else {
|
2014-05-29 04:17:25 +00:00
|
|
|
throwNew(t, GcIllegalArgumentException::Type);
|
2013-12-06 22:45:46 +00:00
|
|
|
}
|
|
|
|
PROTECT(t, target);
|
|
|
|
|
2014-06-28 21:11:31 +00:00
|
|
|
initClass(t, field->class_());
|
2013-12-06 22:45:46 +00:00
|
|
|
|
2014-06-28 21:11:31 +00:00
|
|
|
unsigned offset = field->offset();
|
|
|
|
switch (field->code()) {
|
2013-12-06 22:45:46 +00:00
|
|
|
case ByteField:
|
2014-06-27 00:17:46 +00:00
|
|
|
fieldAtOffset<int8_t>(target, offset) = cast<GcByte>(t, value)->value();
|
2013-12-06 22:45:46 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case BooleanField:
|
2014-06-27 00:17:46 +00:00
|
|
|
fieldAtOffset<int8_t>(target, offset) = cast<GcBoolean>(t, value)->value();
|
2013-12-06 22:45:46 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case CharField:
|
2014-06-27 00:17:46 +00:00
|
|
|
fieldAtOffset<int16_t>(target, offset) = cast<GcChar>(t, value)->value();
|
2013-12-06 22:45:46 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case ShortField:
|
2014-06-27 00:17:46 +00:00
|
|
|
fieldAtOffset<int16_t>(target, offset) = cast<GcShort>(t, value)->value();
|
2013-12-06 22:45:46 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case IntField:
|
2014-06-27 00:17:46 +00:00
|
|
|
fieldAtOffset<int32_t>(target, offset) = cast<GcInt>(t, value)->value();
|
2013-12-06 22:45:46 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case LongField:
|
2014-06-27 00:17:46 +00:00
|
|
|
fieldAtOffset<int64_t>(target, offset) = cast<GcLong>(t, value)->value();
|
2013-12-06 22:45:46 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case FloatField:
|
2014-06-27 00:17:46 +00:00
|
|
|
fieldAtOffset<int32_t>(target, offset) = cast<GcFloat>(t, value)->value();
|
2013-12-06 22:45:46 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case DoubleField:
|
2014-06-27 00:17:46 +00:00
|
|
|
fieldAtOffset<int64_t>(target, offset) = cast<GcDouble>(t, value)->value();
|
2013-12-06 22:45:46 +00:00
|
|
|
break;
|
|
|
|
|
|
|
|
case ObjectField:
|
2014-07-10 23:51:38 +00:00
|
|
|
setField(t, target, offset, value);
|
2013-12-06 22:45:46 +00:00
|
|
|
break;
|
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
default:
|
|
|
|
abort(t);
|
2013-12-06 22:45:46 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
} // namespace local
|
2013-02-19 16:36:19 +00:00
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
} // namespace
|
2013-02-19 16:36:19 +00:00
|
|
|
|
|
|
|
namespace vm {
|
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
Classpath* makeClasspath(System*,
|
|
|
|
Allocator* allocator,
|
|
|
|
const char*,
|
|
|
|
const char*)
|
2013-02-19 16:36:19 +00:00
|
|
|
{
|
|
|
|
return new (allocator->allocate(sizeof(local::MyClasspath)))
|
2014-07-11 15:50:18 +00:00
|
|
|
local::MyClasspath(allocator);
|
2013-02-19 16:36:19 +00:00
|
|
|
}
|
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
} // namespace vm
|
2013-02-19 23:48:33 +00:00
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
extern "C" int jniRegisterNativeMethods(JNIEnv* e,
|
|
|
|
const char* className,
|
|
|
|
const JNINativeMethod* methods,
|
|
|
|
int methodCount)
|
2013-02-19 23:48:33 +00:00
|
|
|
{
|
|
|
|
jclass c = e->vtable->FindClass(e, className);
|
|
|
|
|
|
|
|
if (c) {
|
|
|
|
e->vtable->RegisterNatives(e, c, methods, methodCount);
|
2013-02-26 23:24:02 +00:00
|
|
|
} else {
|
|
|
|
e->vtable->ExceptionClear(e);
|
2013-02-19 23:48:33 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
extern "C" void jniLogException(JNIEnv*, int, const char*, jthrowable)
|
2013-02-19 23:48:33 +00:00
|
|
|
{
|
|
|
|
// ignore
|
|
|
|
}
|
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
extern "C" int jniThrowException(JNIEnv* e,
|
|
|
|
const char* className,
|
|
|
|
const char* message)
|
2013-02-19 23:48:33 +00:00
|
|
|
{
|
|
|
|
jclass c = e->vtable->FindClass(e, className);
|
|
|
|
|
|
|
|
if (c) {
|
|
|
|
e->vtable->ThrowNew(e, c, message);
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
extern "C" int jniThrowExceptionFmt(JNIEnv* e,
|
|
|
|
const char* className,
|
|
|
|
const char* format,
|
|
|
|
va_list args)
|
2013-02-19 23:48:33 +00:00
|
|
|
{
|
|
|
|
const unsigned size = 4096;
|
|
|
|
char buffer[size];
|
|
|
|
::vsnprintf(buffer, size, format, args);
|
|
|
|
return jniThrowException(e, className, buffer);
|
|
|
|
}
|
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
extern "C" int jniThrowNullPointerException(JNIEnv* e, const char* message)
|
2013-02-19 23:48:33 +00:00
|
|
|
{
|
|
|
|
return jniThrowException(e, "java/lang/NullPointerException", message);
|
|
|
|
}
|
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
extern "C" int jniThrowRuntimeException(JNIEnv* e, const char* message)
|
2013-02-19 23:48:33 +00:00
|
|
|
{
|
|
|
|
return jniThrowException(e, "java/lang/RuntimeException", message);
|
|
|
|
}
|
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
extern "C" int jniThrowIOException(JNIEnv* e, const char* message)
|
2013-02-19 23:48:33 +00:00
|
|
|
{
|
|
|
|
return jniThrowException(e, "java/lang/IOException", message);
|
|
|
|
}
|
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
extern "C" const char* jniStrError(int error, char* buffer, size_t length)
|
2013-02-19 23:48:33 +00:00
|
|
|
{
|
2013-03-14 21:33:05 +00:00
|
|
|
#ifdef PLATFORM_WINDOWS
|
2014-02-26 22:56:17 +00:00
|
|
|
const char* s = getErrnoDescription(error);
|
2013-03-14 21:33:05 +00:00
|
|
|
if (strlen(s) < length) {
|
|
|
|
strncpy(buffer, s, length);
|
|
|
|
return buffer;
|
|
|
|
} else {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
#else
|
2013-02-19 23:48:33 +00:00
|
|
|
if (static_cast<int>(strerror_r(error, buffer, length)) == 0) {
|
|
|
|
return buffer;
|
|
|
|
} else {
|
|
|
|
return 0;
|
|
|
|
}
|
2013-03-14 21:33:05 +00:00
|
|
|
#endif
|
2013-02-19 23:48:33 +00:00
|
|
|
}
|
|
|
|
|
2014-03-15 21:52:58 +00:00
|
|
|
/*
|
|
|
|
* Android log priority values (as text)
|
|
|
|
*/
|
2014-07-11 15:50:18 +00:00
|
|
|
const char* const androidLogPriorityTitles[] = {"UNKNOWN",
|
|
|
|
"DEFAULT",
|
|
|
|
"VERBOSE",
|
|
|
|
"DEBUG",
|
|
|
|
"INFO",
|
|
|
|
"WARNING",
|
|
|
|
"ERROR",
|
|
|
|
"FATAL",
|
|
|
|
"SILENT"};
|
|
|
|
|
|
|
|
extern "C" int __android_log_print(int priority,
|
|
|
|
const char* tag,
|
|
|
|
const char* format,
|
|
|
|
...)
|
2013-02-19 23:48:33 +00:00
|
|
|
{
|
|
|
|
va_list a;
|
|
|
|
const unsigned size = 4096;
|
|
|
|
char buffer[size];
|
|
|
|
|
|
|
|
va_start(a, format);
|
|
|
|
::vsnprintf(buffer, size, format, a);
|
|
|
|
va_end(a);
|
|
|
|
|
2014-03-15 21:52:58 +00:00
|
|
|
#ifndef PLATFORM_WINDOWS
|
2014-07-11 15:50:18 +00:00
|
|
|
return printf(
|
|
|
|
"[%s] %s: %s\n", androidLogPriorityTitles[priority], tag, buffer);
|
2014-03-15 21:52:58 +00:00
|
|
|
#else
|
2014-07-11 15:50:18 +00:00
|
|
|
return __mingw_fprintf(
|
|
|
|
stderr, "[%s] %s: %s\n", androidLogPriorityTitles[priority], tag, buffer);
|
2014-03-15 21:52:58 +00:00
|
|
|
#endif
|
2013-02-19 23:48:33 +00:00
|
|
|
}
|
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
extern "C" int jniGetFDFromFileDescriptor(JNIEnv* e, jobject descriptor)
|
2013-02-19 23:48:33 +00:00
|
|
|
{
|
2014-07-11 15:50:18 +00:00
|
|
|
return e->vtable->GetIntField(
|
|
|
|
e,
|
|
|
|
descriptor,
|
|
|
|
e->vtable->GetFieldID(e,
|
|
|
|
e->vtable->FindClass(e, "java/io/FileDescriptor"),
|
|
|
|
"descriptor",
|
|
|
|
"I"));
|
2013-02-19 23:48:33 +00:00
|
|
|
}
|
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
extern "C" void jniSetFileDescriptorOfFD(JNIEnv* e,
|
|
|
|
jobject descriptor,
|
|
|
|
int value)
|
2013-02-19 23:48:33 +00:00
|
|
|
{
|
2014-07-11 15:50:18 +00:00
|
|
|
e->vtable->SetIntField(
|
|
|
|
e,
|
|
|
|
descriptor,
|
|
|
|
e->vtable->GetFieldID(e,
|
|
|
|
e->vtable->FindClass(e, "java/io/FileDescriptor"),
|
|
|
|
"descriptor",
|
|
|
|
"I"),
|
|
|
|
value);
|
2013-02-19 23:48:33 +00:00
|
|
|
}
|
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
extern "C" jobject jniCreateFileDescriptor(JNIEnv* e, int fd)
|
2013-02-19 23:48:33 +00:00
|
|
|
{
|
2014-07-11 15:50:18 +00:00
|
|
|
jobject descriptor = e->vtable->NewObject(
|
|
|
|
e,
|
|
|
|
e->vtable->FindClass(e, "java/io/FileDescriptor"),
|
|
|
|
e->vtable->GetMethodID(e,
|
|
|
|
e->vtable->FindClass(e, "java/io/FileDescriptor"),
|
|
|
|
"<init>",
|
|
|
|
"()V"));
|
2013-02-19 23:48:33 +00:00
|
|
|
|
|
|
|
jniSetFileDescriptorOfFD(e, descriptor, fd);
|
|
|
|
|
|
|
|
return descriptor;
|
|
|
|
}
|
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
int register_org_apache_harmony_dalvik_NativeTestTarget(_JNIEnv*)
|
2013-02-19 23:48:33 +00:00
|
|
|
{
|
|
|
|
// ignore
|
|
|
|
return 0;
|
|
|
|
}
|
2013-02-20 17:22:40 +00:00
|
|
|
|
2014-07-01 16:18:45 +00:00
|
|
|
int register_java_math_NativeBN(_JNIEnv*)
|
|
|
|
{
|
|
|
|
// ignore
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_String_compareTo(Thread* t, object, uintptr_t* arguments)
|
2013-02-22 18:06:49 +00:00
|
|
|
{
|
2014-06-28 23:24:24 +00:00
|
|
|
GcString* a = cast<GcString>(t, reinterpret_cast<object>(arguments[0]));
|
|
|
|
GcString* b = cast<GcString>(t, reinterpret_cast<object>(arguments[1]));
|
2013-02-22 18:06:49 +00:00
|
|
|
|
2014-06-28 23:24:24 +00:00
|
|
|
unsigned length = a->length(t);
|
|
|
|
if (length > b->length(t)) {
|
|
|
|
length = b->length(t);
|
2013-02-22 18:06:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
for (unsigned i = 0; i < length; ++i) {
|
|
|
|
int d = stringCharAt(t, a, i) - stringCharAt(t, b, i);
|
|
|
|
if (d) {
|
|
|
|
return d;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-06-28 23:24:24 +00:00
|
|
|
return a->length(t) - b->length(t);
|
2013-02-22 18:06:49 +00:00
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_String_isEmpty(Thread* t, object, uintptr_t* arguments)
|
2013-02-20 17:22:40 +00:00
|
|
|
{
|
2014-07-11 15:47:57 +00:00
|
|
|
return cast<GcString>(t, reinterpret_cast<object>(arguments[0]))->length(t)
|
|
|
|
== 0;
|
2013-02-20 17:22:40 +00:00
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_String_length(Thread* t, object, uintptr_t* arguments)
|
2013-02-20 17:22:40 +00:00
|
|
|
{
|
2014-06-27 00:17:46 +00:00
|
|
|
return cast<GcString>(t, reinterpret_cast<object>(arguments[0]))->length(t);
|
2013-02-20 17:22:40 +00:00
|
|
|
}
|
|
|
|
|
2013-12-06 22:45:46 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_String_intern(Thread* t, object, uintptr_t* arguments)
|
2013-12-06 22:45:46 +00:00
|
|
|
{
|
2014-07-11 15:50:18 +00:00
|
|
|
return reinterpret_cast<intptr_t>(
|
|
|
|
intern(t, reinterpret_cast<object>(arguments[0])));
|
2013-12-06 22:45:46 +00:00
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_String_charAt(Thread* t, object, uintptr_t* arguments)
|
2013-02-20 17:22:40 +00:00
|
|
|
{
|
2014-07-11 15:47:57 +00:00
|
|
|
return stringCharAt(t,
|
|
|
|
cast<GcString>(t, reinterpret_cast<object>(arguments[0])),
|
|
|
|
arguments[1]);
|
2013-02-20 17:22:40 +00:00
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_String_equals(Thread* t, object, uintptr_t* arguments)
|
2013-02-20 17:22:40 +00:00
|
|
|
{
|
2014-07-11 15:50:18 +00:00
|
|
|
return arguments[1] and stringEqual(t,
|
|
|
|
reinterpret_cast<object>(arguments[0]),
|
|
|
|
reinterpret_cast<object>(arguments[1]));
|
2013-02-20 17:22:40 +00:00
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_String_fastIndexOf(Thread* t, object, uintptr_t* arguments)
|
2013-02-20 17:22:40 +00:00
|
|
|
{
|
2014-06-28 23:24:24 +00:00
|
|
|
GcString* s = cast<GcString>(t, reinterpret_cast<object>(arguments[0]));
|
2013-02-20 17:22:40 +00:00
|
|
|
unsigned c = arguments[1];
|
|
|
|
unsigned start = arguments[2];
|
|
|
|
|
2014-06-28 23:24:24 +00:00
|
|
|
for (unsigned i = start; i < s->length(t); ++i) {
|
2013-02-20 17:22:40 +00:00
|
|
|
if (stringCharAt(t, s, i) == c) {
|
|
|
|
return i;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_Class_getInterfaces(Thread* t, object, uintptr_t* arguments)
|
2013-02-22 18:06:49 +00:00
|
|
|
{
|
2014-06-21 04:16:33 +00:00
|
|
|
GcJclass* c = cast<GcJclass>(t, reinterpret_cast<object>(arguments[0]));
|
2013-02-22 18:06:49 +00:00
|
|
|
|
2014-06-29 02:44:36 +00:00
|
|
|
GcClassAddendum* addendum = c->vmClass()->addendum();
|
2013-02-22 18:06:49 +00:00
|
|
|
if (addendum) {
|
2014-06-27 00:17:46 +00:00
|
|
|
GcArray* table = cast<GcArray>(t, addendum->interfaceTable());
|
2013-02-22 18:06:49 +00:00
|
|
|
if (table) {
|
|
|
|
PROTECT(t, table);
|
|
|
|
|
2014-06-27 00:17:46 +00:00
|
|
|
object array = makeObjectArray(t, table->length());
|
2013-02-22 18:06:49 +00:00
|
|
|
PROTECT(t, array);
|
|
|
|
|
2014-06-27 00:17:46 +00:00
|
|
|
for (unsigned i = 0; i < table->length(); ++i) {
|
|
|
|
GcJclass* c = getJClass(t, cast<GcClass>(t, table->body()[i]));
|
2014-07-11 15:47:57 +00:00
|
|
|
setField(t,
|
|
|
|
array,
|
|
|
|
ArrayBody + (i * BytesPerWord),
|
|
|
|
reinterpret_cast<object>(c));
|
2013-02-22 18:06:49 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return reinterpret_cast<uintptr_t>(array);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
return reinterpret_cast<uintptr_t>(
|
|
|
|
makeObjectArray(t, type(t, GcJclass::Type), 0));
|
2013-02-22 18:06:49 +00:00
|
|
|
}
|
|
|
|
|
2013-12-06 22:45:46 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_Class_getDeclaredClasses(Thread* t,
|
|
|
|
object,
|
|
|
|
uintptr_t* arguments)
|
2013-12-06 22:45:46 +00:00
|
|
|
{
|
2014-07-11 15:47:57 +00:00
|
|
|
return reinterpret_cast<intptr_t>(getDeclaredClasses(
|
|
|
|
t,
|
|
|
|
cast<GcJclass>(t, reinterpret_cast<object>(arguments[0]))->vmClass(),
|
2013-12-06 22:45:46 +00:00
|
|
|
arguments[1]));
|
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_Class_newInstanceImpl(Thread* t,
|
|
|
|
object,
|
|
|
|
uintptr_t* arguments)
|
2013-02-21 22:37:17 +00:00
|
|
|
{
|
2014-07-11 15:47:57 +00:00
|
|
|
GcClass* c
|
|
|
|
= cast<GcJclass>(t, reinterpret_cast<object>(arguments[0]))->vmClass();
|
2013-02-21 22:37:17 +00:00
|
|
|
|
2014-05-29 04:17:25 +00:00
|
|
|
GcMethod* method = resolveMethod(t, c, "<init>", "()V");
|
2013-02-21 22:37:17 +00:00
|
|
|
PROTECT(t, method);
|
|
|
|
|
|
|
|
object instance = makeNew(t, c);
|
|
|
|
PROTECT(t, instance);
|
|
|
|
|
|
|
|
t->m->processor->invoke(t, method, instance);
|
|
|
|
|
|
|
|
return reinterpret_cast<uintptr_t>(instance);
|
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_Class_getComponentType(Thread* t,
|
|
|
|
object,
|
|
|
|
uintptr_t* arguments)
|
2013-02-20 17:22:40 +00:00
|
|
|
{
|
2014-06-21 04:16:33 +00:00
|
|
|
GcJclass* c = cast<GcJclass>(t, reinterpret_cast<object>(arguments[0]));
|
2013-02-20 17:22:40 +00:00
|
|
|
|
2014-06-29 02:44:36 +00:00
|
|
|
if (c->vmClass()->arrayDimensions()) {
|
|
|
|
uint8_t n = c->vmClass()->name()->body()[1];
|
2013-02-20 17:22:40 +00:00
|
|
|
if (n != 'L' and n != '[') {
|
2014-07-11 15:50:18 +00:00
|
|
|
return reinterpret_cast<uintptr_t>(getJClass(t, primitiveClass(t, n)));
|
2013-02-20 17:22:40 +00:00
|
|
|
} else {
|
2014-07-11 15:47:57 +00:00
|
|
|
return reinterpret_cast<uintptr_t>(
|
|
|
|
getJClass(t, c->vmClass()->arrayElementClass()));
|
2014-06-29 02:44:36 +00:00
|
|
|
}
|
2013-02-20 17:22:40 +00:00
|
|
|
} else {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_Class_classForName(Thread* t, object, uintptr_t* arguments)
|
2013-02-21 22:37:17 +00:00
|
|
|
{
|
|
|
|
object name = reinterpret_cast<object>(arguments[0]);
|
|
|
|
PROTECT(t, name);
|
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
GcClassLoader* loader
|
|
|
|
= cast<GcClassLoader>(t, reinterpret_cast<object>(arguments[2]));
|
2013-02-21 22:37:17 +00:00
|
|
|
PROTECT(t, loader);
|
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
GcMethod* method = resolveMethod(
|
|
|
|
t,
|
|
|
|
roots(t)->bootLoader(),
|
|
|
|
"avian/Classes",
|
|
|
|
"forName",
|
|
|
|
"(Ljava/lang/String;ZLjava/lang/ClassLoader;)Ljava/lang/Class;");
|
2013-02-21 22:37:17 +00:00
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
return reinterpret_cast<uintptr_t>(t->m->processor->invoke(
|
|
|
|
t, method, 0, name, static_cast<int>(arguments[1]), loader));
|
2013-02-21 22:37:17 +00:00
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_Class_getDeclaredField(Thread* t,
|
|
|
|
object,
|
|
|
|
uintptr_t* arguments)
|
2013-02-20 17:22:40 +00:00
|
|
|
{
|
2014-06-21 04:16:33 +00:00
|
|
|
GcJclass* c = cast<GcJclass>(t, reinterpret_cast<object>(arguments[0]));
|
2013-02-20 17:22:40 +00:00
|
|
|
PROTECT(t, c);
|
|
|
|
|
|
|
|
object name = reinterpret_cast<object>(arguments[1]);
|
|
|
|
PROTECT(t, name);
|
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
GcMethod* method = resolveMethod(t,
|
|
|
|
roots(t)->bootLoader(),
|
|
|
|
"avian/Classes",
|
|
|
|
"findField",
|
|
|
|
"(Lavian/VMClass;Ljava/lang/String;)I");
|
|
|
|
|
|
|
|
int index = cast<GcInt>(
|
|
|
|
t, t->m->processor->invoke(t, method, 0, c->vmClass(), name))
|
|
|
|
->value();
|
2013-02-21 22:37:17 +00:00
|
|
|
|
|
|
|
if (index >= 0) {
|
2013-02-22 18:06:49 +00:00
|
|
|
return reinterpret_cast<uintptr_t>(local::makeField(t, c, index));
|
2013-02-20 17:22:40 +00:00
|
|
|
} else {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_Class_getDeclaredConstructorOrMethod(Thread* t,
|
|
|
|
object,
|
|
|
|
uintptr_t* arguments)
|
2013-02-21 22:37:17 +00:00
|
|
|
{
|
2014-06-21 04:16:33 +00:00
|
|
|
GcJclass* c = cast<GcJclass>(t, reinterpret_cast<object>(arguments[0]));
|
2013-02-21 22:37:17 +00:00
|
|
|
PROTECT(t, c);
|
|
|
|
|
|
|
|
object name = reinterpret_cast<object>(arguments[1]);
|
|
|
|
PROTECT(t, name);
|
|
|
|
|
|
|
|
object parameterTypes = reinterpret_cast<object>(arguments[2]);
|
|
|
|
PROTECT(t, parameterTypes);
|
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
GcMethod* method
|
|
|
|
= resolveMethod(t,
|
|
|
|
roots(t)->bootLoader(),
|
|
|
|
"avian/Classes",
|
|
|
|
"findMethod",
|
|
|
|
"(Lavian/VMClass;Ljava/lang/String;[Ljava/lang/Class;)I");
|
|
|
|
|
|
|
|
int index = cast<GcInt>(t,
|
|
|
|
t->m->processor->invoke(
|
|
|
|
t, method, 0, c->vmClass(), name, parameterTypes))
|
|
|
|
->value();
|
2013-02-21 22:37:17 +00:00
|
|
|
|
|
|
|
if (index >= 0) {
|
2014-07-11 15:50:18 +00:00
|
|
|
return reinterpret_cast<uintptr_t>(
|
|
|
|
local::makeMethodOrConstructor(t, c, index));
|
2013-02-21 22:37:17 +00:00
|
|
|
} else {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_avian_SystemClassLoader_findLoadedVMClass(Thread*,
|
|
|
|
object,
|
|
|
|
uintptr_t*);
|
2013-02-21 22:37:17 +00:00
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_VMClassLoader_findLoadedClass(Thread* t,
|
|
|
|
object method,
|
|
|
|
uintptr_t* arguments)
|
2013-02-21 22:37:17 +00:00
|
|
|
{
|
2014-07-11 15:50:18 +00:00
|
|
|
int64_t v
|
|
|
|
= Avian_avian_SystemClassLoader_findLoadedVMClass(t, method, arguments);
|
2013-02-21 22:37:17 +00:00
|
|
|
|
|
|
|
if (v) {
|
2014-07-11 15:47:57 +00:00
|
|
|
return reinterpret_cast<uintptr_t>(
|
|
|
|
getJClass(t, cast<GcClass>(t, reinterpret_cast<object>(v))));
|
2013-02-21 22:37:17 +00:00
|
|
|
} else {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_VMClassLoader_defineClass__Ljava_lang_ClassLoader_2Ljava_lang_String_2_3BII(
|
|
|
|
Thread* t,
|
|
|
|
object method,
|
|
|
|
uintptr_t* arguments)
|
2013-02-21 22:37:17 +00:00
|
|
|
{
|
2014-07-11 15:50:18 +00:00
|
|
|
uintptr_t args[] = {arguments[0], arguments[2], arguments[3], arguments[4]};
|
2013-02-21 22:37:17 +00:00
|
|
|
|
|
|
|
int64_t v = Avian_avian_Classes_defineVMClass(t, method, args);
|
|
|
|
|
|
|
|
if (v) {
|
2014-07-11 15:47:57 +00:00
|
|
|
return reinterpret_cast<uintptr_t>(
|
|
|
|
getJClass(t, cast<GcClass>(t, reinterpret_cast<object>(v))));
|
2013-02-21 22:37:17 +00:00
|
|
|
} else {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_dalvik_system_VMRuntime_bootClassPath(Thread* t, object, uintptr_t*)
|
2013-02-20 17:22:40 +00:00
|
|
|
{
|
2014-06-30 01:44:41 +00:00
|
|
|
return reinterpret_cast<uintptr_t>(roots(t)->bootLoader());
|
2013-02-20 17:22:40 +00:00
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_dalvik_system_VMRuntime_classPath(Thread* t, object, uintptr_t*)
|
2013-02-20 17:22:40 +00:00
|
|
|
{
|
2014-06-30 01:44:41 +00:00
|
|
|
return reinterpret_cast<uintptr_t>(roots(t)->appLoader());
|
2013-02-20 17:22:40 +00:00
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_dalvik_system_VMRuntime_vmVersion(Thread* t, object, uintptr_t*)
|
2013-02-20 17:22:40 +00:00
|
|
|
{
|
|
|
|
return reinterpret_cast<uintptr_t>(makeString(t, "%s", AVIAN_VERSION));
|
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_dalvik_system_VMRuntime_properties(Thread* t, object, uintptr_t*)
|
2013-02-20 17:22:40 +00:00
|
|
|
{
|
2014-07-11 15:47:57 +00:00
|
|
|
object array
|
|
|
|
= makeObjectArray(t, type(t, GcString::Type), t->m->propertyCount + 1);
|
2013-02-26 23:24:02 +00:00
|
|
|
PROTECT(t, array);
|
|
|
|
|
2014-04-04 17:10:38 +00:00
|
|
|
unsigned i;
|
|
|
|
for (i = 0; i < t->m->propertyCount; ++i) {
|
2014-06-28 23:24:24 +00:00
|
|
|
GcString* s = makeString(t, "%s", t->m->properties[i]);
|
2014-07-11 15:47:57 +00:00
|
|
|
setField(
|
|
|
|
t, array, ArrayBody + (i * BytesPerWord), reinterpret_cast<object>(s));
|
2014-04-04 17:10:38 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
{
|
2014-06-28 23:24:24 +00:00
|
|
|
GcString* s = makeString(t, "%s", "java.protocol.handler.pkgs=avian");
|
2014-07-11 15:47:57 +00:00
|
|
|
setField(t,
|
|
|
|
array,
|
|
|
|
ArrayBody + (i++ * BytesPerWord),
|
|
|
|
reinterpret_cast<object>(s));
|
2014-04-04 17:10:38 +00:00
|
|
|
}
|
2013-02-26 23:24:02 +00:00
|
|
|
|
|
|
|
return reinterpret_cast<uintptr_t>(array);
|
2013-02-20 17:22:40 +00:00
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_Runtime_gc(Thread* t, object, uintptr_t*)
|
2013-02-22 18:06:49 +00:00
|
|
|
{
|
|
|
|
collect(t, Heap::MajorCollection);
|
|
|
|
}
|
|
|
|
|
2014-07-12 22:03:11 +00:00
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
|
|
|
Avian_java_lang_Runtime_nativeExit(Thread* t, object, uintptr_t* arguments)
|
|
|
|
{
|
|
|
|
shutDown(t);
|
|
|
|
|
|
|
|
t->m->system->exit(arguments[0]);
|
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_Runtime_nativeLoad(Thread* t, object, uintptr_t* arguments)
|
2013-02-26 23:24:02 +00:00
|
|
|
{
|
2014-06-28 23:24:24 +00:00
|
|
|
GcString* name = cast<GcString>(t, reinterpret_cast<object>(arguments[0]));
|
2013-02-26 23:24:02 +00:00
|
|
|
PROTECT(t, name);
|
|
|
|
|
2014-06-28 23:24:24 +00:00
|
|
|
unsigned length = name->length(t);
|
2013-02-26 23:24:02 +00:00
|
|
|
THREAD_RUNTIME_ARRAY(t, char, n, length + 1);
|
|
|
|
stringChars(t, name, RUNTIME_ARRAY_BODY(n));
|
|
|
|
|
|
|
|
if (loadLibrary(t, "", RUNTIME_ARRAY_BODY(n), false, true)) {
|
|
|
|
return 0;
|
|
|
|
} else {
|
|
|
|
return reinterpret_cast<uintptr_t>(name);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_System_arraycopy(Thread* t, object, uintptr_t* arguments)
|
2013-02-20 17:22:40 +00:00
|
|
|
{
|
2014-07-11 15:50:18 +00:00
|
|
|
arrayCopy(t,
|
|
|
|
reinterpret_cast<object>(arguments[0]),
|
2013-02-20 17:22:40 +00:00
|
|
|
arguments[1],
|
|
|
|
reinterpret_cast<object>(arguments[2]),
|
|
|
|
arguments[3],
|
|
|
|
arguments[4]);
|
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_sun_misc_Unsafe_objectFieldOffset(Thread* t,
|
|
|
|
object,
|
|
|
|
uintptr_t* arguments)
|
2013-02-20 17:22:40 +00:00
|
|
|
{
|
2014-06-27 00:17:46 +00:00
|
|
|
GcJfield* jfield = cast<GcJfield>(t, reinterpret_cast<object>(arguments[1]));
|
2014-07-11 15:47:57 +00:00
|
|
|
return cast<GcField>(
|
|
|
|
t,
|
|
|
|
cast<GcArray>(t, jfield->declaringClass()->vmClass()->fieldTable())
|
|
|
|
->body()[jfield->slot()])->offset();
|
2013-02-20 17:22:40 +00:00
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_VMThread_interrupt(Thread* t, object, uintptr_t* arguments)
|
2013-02-22 18:06:49 +00:00
|
|
|
{
|
|
|
|
object vmThread = reinterpret_cast<object>(arguments[0]);
|
|
|
|
PROTECT(t, vmThread);
|
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
GcField* field = resolveField(
|
|
|
|
t, objectClass(t, vmThread), "thread", "Ljava/lang/Thread;");
|
2013-02-22 18:06:49 +00:00
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
interrupt(
|
|
|
|
t,
|
|
|
|
reinterpret_cast<Thread*>(
|
|
|
|
cast<GcThread>(t, fieldAtOffset<object>(vmThread, field->offset()))
|
|
|
|
->peer()));
|
2013-02-22 18:06:49 +00:00
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_VMThread_interrupted(Thread* t, object, uintptr_t*)
|
2013-02-22 18:06:49 +00:00
|
|
|
{
|
|
|
|
return getAndClearInterrupted(t, t);
|
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_VMThread_isInterrupted(Thread* t,
|
|
|
|
object,
|
|
|
|
uintptr_t* arguments)
|
2013-02-22 18:06:49 +00:00
|
|
|
{
|
|
|
|
object vmThread = reinterpret_cast<object>(arguments[0]);
|
|
|
|
PROTECT(t, vmThread);
|
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
GcField* field = resolveField(
|
|
|
|
t, objectClass(t, vmThread), "thread", "Ljava/lang/Thread;");
|
2013-02-22 18:06:49 +00:00
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
return cast<GcThread>(t, fieldAtOffset<object>(vmThread, field->offset()))
|
|
|
|
->interrupted();
|
2013-02-22 18:06:49 +00:00
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_VMThread_getStatus(Thread*, object, uintptr_t*)
|
2013-02-22 18:06:49 +00:00
|
|
|
{
|
|
|
|
// todo
|
2013-04-23 19:47:15 +00:00
|
|
|
return 1;
|
2013-02-22 18:06:49 +00:00
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_VMThread_currentThread(Thread* t, object, uintptr_t*)
|
2013-02-20 17:22:40 +00:00
|
|
|
{
|
|
|
|
return reinterpret_cast<uintptr_t>(t->javaThread);
|
|
|
|
}
|
2013-02-21 22:37:17 +00:00
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_VMThread_create(Thread* t, object, uintptr_t* arguments)
|
2013-02-22 18:06:49 +00:00
|
|
|
{
|
2014-06-28 19:16:26 +00:00
|
|
|
GcThread* thread = cast<GcThread>(t, reinterpret_cast<object>(arguments[0]));
|
2013-03-15 19:28:01 +00:00
|
|
|
PROTECT(t, thread);
|
|
|
|
|
|
|
|
local::initVmThread(t, thread);
|
|
|
|
startThread(t, thread);
|
2013-02-22 18:06:49 +00:00
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_VMThread_sleep(Thread* t, object, uintptr_t* arguments)
|
2013-02-22 18:06:49 +00:00
|
|
|
{
|
2014-07-11 15:50:18 +00:00
|
|
|
int64_t milliseconds;
|
|
|
|
memcpy(&milliseconds, arguments, 8);
|
|
|
|
if (arguments[2] > 0)
|
|
|
|
++milliseconds;
|
|
|
|
if (milliseconds <= 0)
|
|
|
|
milliseconds = 1;
|
2013-02-22 18:06:49 +00:00
|
|
|
|
2014-06-28 23:24:24 +00:00
|
|
|
if (t->javaThread->sleepLock() == 0) {
|
2014-07-02 21:11:27 +00:00
|
|
|
object lock = makeJobject(t);
|
2014-07-10 23:51:38 +00:00
|
|
|
t->javaThread->setSleepLock(t, lock);
|
2013-02-22 18:06:49 +00:00
|
|
|
}
|
|
|
|
|
2014-06-28 23:24:24 +00:00
|
|
|
acquire(t, t->javaThread->sleepLock());
|
|
|
|
vm::wait(t, t->javaThread->sleepLock(), milliseconds);
|
|
|
|
release(t, t->javaThread->sleepLock());
|
2013-02-22 18:06:49 +00:00
|
|
|
}
|
|
|
|
|
2014-07-12 22:03:11 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
|
|
|
Avian_java_lang_VMThread_holdsLock(Thread* t, object, uintptr_t* arguments)
|
|
|
|
{
|
|
|
|
object vmThread = reinterpret_cast<object>(arguments[0]);
|
|
|
|
PROTECT(t, vmThread);
|
|
|
|
|
|
|
|
GcField* field = resolveField(
|
|
|
|
t, objectClass(t, vmThread), "thread", "Ljava/lang/Thread;");
|
|
|
|
|
|
|
|
if (cast<GcThread>(t, fieldAtOffset<object>(vmThread, field->offset()))
|
|
|
|
!= t->javaThread) {
|
|
|
|
throwNew(t,
|
|
|
|
GcIllegalStateException::Type,
|
|
|
|
"VMThread.holdsLock may only be called on current thread");
|
|
|
|
}
|
|
|
|
|
|
|
|
GcMonitor* m
|
|
|
|
= objectMonitor(t, reinterpret_cast<object>(arguments[1]), false);
|
|
|
|
|
|
|
|
return m and m->owner() == t;
|
|
|
|
}
|
|
|
|
|
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
|
|
|
Avian_java_lang_VMThread_yield(Thread* t, object, uintptr_t*)
|
|
|
|
{
|
|
|
|
t->m->system->yield();
|
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_dalvik_system_VMStack_getThreadStackTrace(Thread* t,
|
|
|
|
object,
|
|
|
|
uintptr_t* arguments)
|
2013-02-22 18:06:49 +00:00
|
|
|
{
|
2014-07-11 15:47:57 +00:00
|
|
|
Thread* p = reinterpret_cast<Thread*>(
|
|
|
|
cast<GcThread>(t, reinterpret_cast<object>(arguments[0]))->peer());
|
2013-02-22 18:06:49 +00:00
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
return reinterpret_cast<uintptr_t>(local::translateStackTrace(
|
|
|
|
t, p == t ? makeTrace(t) : t->m->processor->getStackTrace(t, p)));
|
2013-02-22 18:06:49 +00:00
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_dalvik_system_VMStack_getCallingClassLoader(Thread* t,
|
|
|
|
object,
|
|
|
|
uintptr_t*)
|
2013-02-21 22:37:17 +00:00
|
|
|
{
|
2014-07-11 15:50:18 +00:00
|
|
|
class Visitor : public Processor::StackVisitor {
|
2013-02-21 22:37:17 +00:00
|
|
|
public:
|
2014-07-11 15:50:18 +00:00
|
|
|
Visitor(Thread* t) : t(t), loader(0), counter(2)
|
|
|
|
{
|
|
|
|
}
|
2013-02-21 22:37:17 +00:00
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
virtual bool visit(Processor::StackWalker* walker)
|
|
|
|
{
|
2013-02-21 22:37:17 +00:00
|
|
|
if (counter--) {
|
|
|
|
return true;
|
|
|
|
} else {
|
2014-06-21 04:16:33 +00:00
|
|
|
this->loader = walker->method()->class_()->loader();
|
2013-02-21 22:37:17 +00:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
Thread* t;
|
2014-06-21 04:16:33 +00:00
|
|
|
GcClassLoader* loader;
|
2013-02-21 22:37:17 +00:00
|
|
|
unsigned counter;
|
|
|
|
} v(t);
|
|
|
|
|
|
|
|
t->m->processor->walkStack(t, &v);
|
|
|
|
|
|
|
|
return reinterpret_cast<uintptr_t>(v.loader);
|
|
|
|
}
|
|
|
|
|
2013-12-06 22:45:46 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_dalvik_system_VMStack_getClasses(Thread* t, object, uintptr_t*)
|
2013-12-06 22:45:46 +00:00
|
|
|
{
|
2014-07-11 15:50:18 +00:00
|
|
|
class Visitor : public Processor::StackVisitor {
|
2013-12-06 22:45:46 +00:00
|
|
|
public:
|
2014-07-11 15:50:18 +00:00
|
|
|
Visitor(Thread* t) : t(t), array(0), counter(0)
|
|
|
|
{
|
|
|
|
}
|
2013-12-06 22:45:46 +00:00
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
virtual bool visit(Processor::StackWalker* walker)
|
|
|
|
{
|
2013-12-06 22:45:46 +00:00
|
|
|
if (counter < 2) {
|
|
|
|
return true;
|
|
|
|
} else {
|
|
|
|
if (array == 0) {
|
2014-07-11 15:47:57 +00:00
|
|
|
array = makeObjectArray(t, type(t, GcJclass::Type), walker->count());
|
2013-12-06 22:45:46 +00:00
|
|
|
}
|
|
|
|
|
2014-06-21 04:16:33 +00:00
|
|
|
GcJclass* c = getJClass(t, walker->method()->class_());
|
2014-07-11 15:47:57 +00:00
|
|
|
|
2014-06-04 01:52:01 +00:00
|
|
|
assertT(t, counter - 2 < objectArrayLength(t, array));
|
2013-12-06 22:45:46 +00:00
|
|
|
|
2014-07-02 21:11:27 +00:00
|
|
|
setField(t, array, ArrayBody + ((counter - 2) * BytesPerWord), c);
|
2013-12-06 22:45:46 +00:00
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
++counter;
|
2013-12-06 22:45:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Thread* t;
|
|
|
|
object array;
|
|
|
|
unsigned counter;
|
|
|
|
} v(t);
|
|
|
|
|
|
|
|
PROTECT(t, v.array);
|
|
|
|
|
|
|
|
t->m->processor->walkStack(t, &v);
|
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
return reinterpret_cast<uintptr_t>(
|
|
|
|
v.array ? v.array : makeObjectArray(t, type(t, GcJclass::Type), 0));
|
2013-12-06 22:45:46 +00:00
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_Math_min(Thread*, object, uintptr_t* arguments)
|
2013-02-21 22:37:17 +00:00
|
|
|
{
|
|
|
|
return min(static_cast<int>(arguments[0]), static_cast<int>(arguments[1]));
|
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_Math_max(Thread*, object, uintptr_t* arguments)
|
2013-02-21 22:37:17 +00:00
|
|
|
{
|
|
|
|
return max(static_cast<int>(arguments[0]), static_cast<int>(arguments[1]));
|
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_Math_cos(Thread*, object, uintptr_t* arguments)
|
2013-02-22 18:06:49 +00:00
|
|
|
{
|
2014-07-11 15:50:18 +00:00
|
|
|
int64_t v;
|
|
|
|
memcpy(&v, arguments, 8);
|
2013-02-26 23:24:02 +00:00
|
|
|
return doubleToBits(cos(bitsToDouble(v)));
|
2013-02-22 18:06:49 +00:00
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_Math_sin(Thread*, object, uintptr_t* arguments)
|
2013-02-23 00:23:59 +00:00
|
|
|
{
|
2014-07-11 15:50:18 +00:00
|
|
|
int64_t v;
|
|
|
|
memcpy(&v, arguments, 8);
|
2013-02-26 23:24:02 +00:00
|
|
|
return doubleToBits(sin(bitsToDouble(v)));
|
2013-02-23 00:23:59 +00:00
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_Math_sqrt(Thread*, object, uintptr_t* arguments)
|
2013-03-15 19:28:01 +00:00
|
|
|
{
|
2014-07-11 15:50:18 +00:00
|
|
|
int64_t v;
|
|
|
|
memcpy(&v, arguments, 8);
|
2013-03-15 19:28:01 +00:00
|
|
|
return doubleToBits(sqrt(bitsToDouble(v)));
|
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_Math_abs__I(Thread*, object, uintptr_t* arguments)
|
2013-03-15 19:28:01 +00:00
|
|
|
{
|
|
|
|
return abs(static_cast<int32_t>(arguments[0]));
|
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_Math_abs__J(Thread*, object, uintptr_t* arguments)
|
2013-03-15 19:28:01 +00:00
|
|
|
{
|
|
|
|
return llabs(arguments[0]);
|
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_Math_abs__F(Thread*, object, uintptr_t* arguments)
|
2013-03-15 19:28:01 +00:00
|
|
|
{
|
|
|
|
return floatToBits(abs(bitsToFloat(arguments[0])));
|
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_Float_intBitsToFloat(Thread*, object, uintptr_t* arguments)
|
2013-02-22 18:06:49 +00:00
|
|
|
{
|
|
|
|
return arguments[0];
|
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_Float_floatToIntBits(Thread*, object, uintptr_t* arguments)
|
2013-02-22 18:06:49 +00:00
|
|
|
{
|
|
|
|
if (((arguments[0] & 0x7F800000) == 0x7F800000)
|
2014-07-11 15:50:18 +00:00
|
|
|
and ((arguments[0] & 0x007FFFFF) != 0)) {
|
2013-02-22 18:06:49 +00:00
|
|
|
return 0x7fc00000;
|
|
|
|
} else {
|
|
|
|
return arguments[0];
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_Double_doubleToRawLongBits(Thread*,
|
|
|
|
object,
|
|
|
|
uintptr_t* arguments)
|
2013-02-23 00:23:59 +00:00
|
|
|
{
|
2014-07-11 15:50:18 +00:00
|
|
|
int64_t v;
|
|
|
|
memcpy(&v, arguments, 8);
|
2013-02-23 00:23:59 +00:00
|
|
|
// todo: do we need to do NaN checks as in
|
|
|
|
// Avian_java_lang_Float_floatToIntBits above? If so, update
|
|
|
|
// Double.doubleToRawLongBits in the Avian class library too.
|
|
|
|
return v;
|
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_Object_wait(Thread* t, object, uintptr_t* arguments)
|
2013-02-22 18:06:49 +00:00
|
|
|
{
|
2014-07-11 15:50:18 +00:00
|
|
|
jlong milliseconds;
|
|
|
|
memcpy(&milliseconds, arguments + 1, sizeof(jlong));
|
2013-02-22 18:06:49 +00:00
|
|
|
|
|
|
|
wait(t, reinterpret_cast<object>(arguments[0]), milliseconds);
|
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_Object_notifyAll(Thread* t, object, uintptr_t* arguments)
|
2013-02-22 18:06:49 +00:00
|
|
|
{
|
|
|
|
notifyAll(t, reinterpret_cast<object>(arguments[0]));
|
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_Object_getClass(Thread* t, object, uintptr_t* arguments)
|
2013-02-21 22:37:17 +00:00
|
|
|
{
|
2014-07-11 15:50:18 +00:00
|
|
|
return reinterpret_cast<uintptr_t>(
|
|
|
|
getJClass(t, objectClass(t, reinterpret_cast<object>(arguments[0]))));
|
2013-02-21 22:37:17 +00:00
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_Object_hashCode(Thread* t, object, uintptr_t* arguments)
|
2013-02-21 22:37:17 +00:00
|
|
|
{
|
|
|
|
return objectHash(t, reinterpret_cast<object>(arguments[0]));
|
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_Object_internalClone(Thread* t,
|
|
|
|
object,
|
|
|
|
uintptr_t* arguments)
|
2013-02-21 22:37:17 +00:00
|
|
|
{
|
2014-07-11 15:50:18 +00:00
|
|
|
return reinterpret_cast<uintptr_t>(
|
|
|
|
clone(t, reinterpret_cast<object>(arguments[1])));
|
2013-02-21 22:37:17 +00:00
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_Class_getModifiers(Thread* t, object, uintptr_t* arguments)
|
2013-02-21 22:37:17 +00:00
|
|
|
{
|
2014-07-11 15:47:57 +00:00
|
|
|
return classModifiers(
|
|
|
|
t, cast<GcJclass>(t, reinterpret_cast<object>(arguments[0]))->vmClass());
|
2013-02-21 22:37:17 +00:00
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_Class_getSuperclass(Thread* t, object, uintptr_t* arguments)
|
2013-02-21 22:37:17 +00:00
|
|
|
{
|
2014-07-11 15:47:57 +00:00
|
|
|
GcClass* c
|
|
|
|
= cast<GcJclass>(t, reinterpret_cast<object>(arguments[0]))->vmClass();
|
2014-06-29 02:44:36 +00:00
|
|
|
if (c->flags() & ACC_INTERFACE) {
|
2013-02-21 22:37:17 +00:00
|
|
|
return 0;
|
|
|
|
} else {
|
2014-06-29 02:44:36 +00:00
|
|
|
GcClass* s = c->super();
|
2013-02-21 22:37:17 +00:00
|
|
|
return s ? reinterpret_cast<uintptr_t>(getJClass(t, s)) : 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_Class_desiredAssertionStatus(Thread*, object, uintptr_t*)
|
2013-02-21 22:37:17 +00:00
|
|
|
{
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_Class_getNameNative(Thread* t, object, uintptr_t* arguments)
|
2013-02-21 22:37:17 +00:00
|
|
|
{
|
2014-07-11 15:47:57 +00:00
|
|
|
GcByteArray* name = cast<GcJclass>(t, reinterpret_cast<object>(arguments[0]))
|
|
|
|
->vmClass()
|
|
|
|
->name();
|
2013-02-21 22:37:17 +00:00
|
|
|
|
2014-06-29 02:44:36 +00:00
|
|
|
THREAD_RUNTIME_ARRAY(t, char, s, name->length());
|
2014-07-11 15:47:57 +00:00
|
|
|
replace('/',
|
|
|
|
'.',
|
|
|
|
RUNTIME_ARRAY_BODY(s),
|
2014-06-29 02:44:36 +00:00
|
|
|
reinterpret_cast<char*>(name->body().begin()));
|
2013-02-26 23:24:02 +00:00
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
return reinterpret_cast<uintptr_t>(
|
|
|
|
makeString(t, "%s", RUNTIME_ARRAY_BODY(s)));
|
2013-02-21 22:37:17 +00:00
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_Class_isInterface(Thread* t, object, uintptr_t* arguments)
|
2013-02-21 22:37:17 +00:00
|
|
|
{
|
2014-07-11 15:47:57 +00:00
|
|
|
return (cast<GcJclass>(t, reinterpret_cast<object>(arguments[0]))
|
|
|
|
->vmClass()
|
|
|
|
->flags() & ACC_INTERFACE) != 0;
|
2013-02-21 22:37:17 +00:00
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_Class_isPrimitive(Thread* t, object, uintptr_t* arguments)
|
2013-02-21 22:37:17 +00:00
|
|
|
{
|
2014-07-11 15:47:57 +00:00
|
|
|
return (cast<GcJclass>(t, reinterpret_cast<object>(arguments[0]))
|
|
|
|
->vmClass()
|
|
|
|
->vmFlags() & PrimitiveFlag) != 0;
|
2013-02-21 22:37:17 +00:00
|
|
|
}
|
|
|
|
|
2013-12-06 22:45:46 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_Class_isAnonymousClass(Thread* t,
|
|
|
|
object,
|
|
|
|
uintptr_t* arguments)
|
2013-12-06 22:45:46 +00:00
|
|
|
{
|
2014-07-11 15:47:57 +00:00
|
|
|
GcByteArray* name = cast<GcJclass>(t, reinterpret_cast<object>(arguments[0]))
|
|
|
|
->vmClass()
|
|
|
|
->name();
|
2013-12-06 22:45:46 +00:00
|
|
|
|
2014-06-29 02:44:36 +00:00
|
|
|
for (unsigned i = 0; i < name->length() - 1; ++i) {
|
|
|
|
int c = name->body()[i];
|
2013-12-06 22:45:46 +00:00
|
|
|
if (c != '$' and (c < '0' or c > '9')) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_Class_getClassLoader(Thread* t,
|
|
|
|
object,
|
|
|
|
uintptr_t* arguments)
|
2013-02-21 22:37:17 +00:00
|
|
|
{
|
2014-07-11 15:47:57 +00:00
|
|
|
return reinterpret_cast<uintptr_t>(
|
|
|
|
cast<GcJclass>(t, reinterpret_cast<object>(arguments[0]))
|
|
|
|
->vmClass()
|
|
|
|
->loader());
|
2013-02-21 22:37:17 +00:00
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_Class_isAssignableFrom(Thread* t,
|
|
|
|
object,
|
|
|
|
uintptr_t* arguments)
|
2013-02-21 22:37:17 +00:00
|
|
|
{
|
2014-06-27 00:17:46 +00:00
|
|
|
GcJclass* this_ = cast<GcJclass>(t, reinterpret_cast<object>(arguments[0]));
|
|
|
|
GcJclass* that = cast<GcJclass>(t, reinterpret_cast<object>(arguments[1]));
|
2013-02-21 22:37:17 +00:00
|
|
|
|
|
|
|
if (LIKELY(that)) {
|
2014-07-11 15:47:57 +00:00
|
|
|
return isAssignableFrom(t, this_->vmClass(), that->vmClass());
|
2013-02-21 22:37:17 +00:00
|
|
|
} else {
|
2014-05-29 04:17:25 +00:00
|
|
|
throwNew(t, GcNullPointerException::Type);
|
2013-02-21 22:37:17 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_Class_isInstance(Thread* t, object, uintptr_t* arguments)
|
2013-02-22 18:06:49 +00:00
|
|
|
{
|
2014-06-27 00:17:46 +00:00
|
|
|
GcJclass* this_ = cast<GcJclass>(t, reinterpret_cast<object>(arguments[0]));
|
2013-02-22 18:06:49 +00:00
|
|
|
object o = reinterpret_cast<object>(arguments[1]);
|
|
|
|
|
|
|
|
if (o) {
|
2014-06-27 00:17:46 +00:00
|
|
|
return instanceOf(t, this_->vmClass(), o);
|
2013-02-22 18:06:49 +00:00
|
|
|
} else {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_Class_getDeclaredMethods(Thread* t,
|
|
|
|
object,
|
|
|
|
uintptr_t* arguments)
|
2013-04-23 19:47:15 +00:00
|
|
|
{
|
2014-06-21 04:16:33 +00:00
|
|
|
GcJclass* c = cast<GcJclass>(t, reinterpret_cast<object>(arguments[0]));
|
2013-04-23 19:47:15 +00:00
|
|
|
PROTECT(t, c);
|
|
|
|
|
|
|
|
bool publicOnly = arguments[1];
|
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
GcMethod* get
|
|
|
|
= resolveMethod(t,
|
|
|
|
roots(t)->bootLoader(),
|
|
|
|
"avian/Classes",
|
|
|
|
"getMethods",
|
|
|
|
"(Lavian/VMClass;Z)[Ljava/lang/reflect/Method;");
|
2013-04-23 19:47:15 +00:00
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
return reinterpret_cast<uintptr_t>(
|
|
|
|
t->m->processor->invoke(t, get, 0, c->vmClass(), publicOnly));
|
2013-04-23 19:47:15 +00:00
|
|
|
}
|
|
|
|
|
2013-12-06 22:45:46 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_Class_getDeclaredFields(Thread* t,
|
|
|
|
object,
|
|
|
|
uintptr_t* arguments)
|
2013-12-06 22:45:46 +00:00
|
|
|
{
|
2014-06-21 04:16:33 +00:00
|
|
|
GcJclass* c = cast<GcJclass>(t, reinterpret_cast<object>(arguments[0]));
|
2013-12-06 22:45:46 +00:00
|
|
|
PROTECT(t, c);
|
|
|
|
|
|
|
|
bool publicOnly = arguments[1];
|
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
GcMethod* get = resolveMethod(t,
|
|
|
|
roots(t)->bootLoader(),
|
|
|
|
"avian/Classes",
|
|
|
|
"getFields",
|
|
|
|
"(Lavian/VMClass;Z)[Ljava/lang/reflect/Field;");
|
2013-12-06 22:45:46 +00:00
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
return reinterpret_cast<uintptr_t>(
|
|
|
|
t->m->processor->invoke(t, get, 0, c->vmClass(), publicOnly));
|
2013-12-06 22:45:46 +00:00
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_reflect_Method_invokeNative(Thread* t,
|
|
|
|
object,
|
|
|
|
uintptr_t* arguments)
|
2013-02-21 22:37:17 +00:00
|
|
|
{
|
|
|
|
object instance = reinterpret_cast<object>(arguments[1]);
|
|
|
|
object args = reinterpret_cast<object>(arguments[2]);
|
2014-07-11 15:47:57 +00:00
|
|
|
GcMethod* method = cast<GcMethod>(
|
|
|
|
t,
|
|
|
|
cast<GcArray>(t,
|
|
|
|
cast<GcJclass>(t, reinterpret_cast<object>(arguments[3]))
|
|
|
|
->vmClass()
|
|
|
|
->methodTable())->body()[arguments[6]]);
|
2013-02-21 22:37:17 +00:00
|
|
|
|
|
|
|
return reinterpret_cast<uintptr_t>(invoke(t, method, instance, args));
|
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_reflect_Method_getMethodModifiers(Thread* t,
|
|
|
|
object,
|
|
|
|
uintptr_t* arguments)
|
2013-02-21 22:37:17 +00:00
|
|
|
{
|
2014-07-11 15:47:57 +00:00
|
|
|
return cast<GcMethod>(
|
|
|
|
t,
|
|
|
|
cast<GcArray>(
|
|
|
|
t,
|
|
|
|
cast<GcJclass>(t, reinterpret_cast<object>(arguments[0]))
|
|
|
|
->vmClass()
|
|
|
|
->methodTable())->body()[arguments[1]])->flags();
|
2013-02-21 22:37:17 +00:00
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_reflect_Method_isAnnotationPresent(Thread* t,
|
|
|
|
object,
|
|
|
|
uintptr_t* arguments)
|
2013-02-21 22:37:17 +00:00
|
|
|
{
|
2014-07-11 15:47:57 +00:00
|
|
|
GcMethod* method = cast<GcMethod>(
|
|
|
|
t,
|
|
|
|
cast<GcArray>(t,
|
|
|
|
cast<GcJclass>(t, reinterpret_cast<object>(arguments[0]))
|
|
|
|
->vmClass()
|
|
|
|
->methodTable())->body()[arguments[1]]);
|
2013-02-21 22:37:17 +00:00
|
|
|
|
2014-06-27 00:17:46 +00:00
|
|
|
GcMethodAddendum* addendum = method->addendum();
|
2013-02-21 22:37:17 +00:00
|
|
|
if (addendum) {
|
2014-06-27 00:17:16 +00:00
|
|
|
object table = addendum->annotationTable();
|
2013-02-21 22:37:17 +00:00
|
|
|
if (table) {
|
|
|
|
for (unsigned i = 0; i < objectArrayLength(t, table); ++i) {
|
|
|
|
if (objectArrayBody(t, objectArrayBody(t, table, i), 1)
|
2014-07-11 15:50:18 +00:00
|
|
|
== reinterpret_cast<object>(arguments[2])) {
|
2013-02-21 22:37:17 +00:00
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_reflect_Method_getAnnotation(Thread* t,
|
|
|
|
object,
|
|
|
|
uintptr_t* arguments)
|
2013-02-21 22:37:17 +00:00
|
|
|
{
|
2014-07-11 15:47:57 +00:00
|
|
|
GcMethod* method = cast<GcMethod>(
|
|
|
|
t,
|
|
|
|
cast<GcArray>(t,
|
|
|
|
cast<GcJclass>(t, reinterpret_cast<object>(arguments[0]))
|
|
|
|
->vmClass()
|
|
|
|
->methodTable())->body()[arguments[1]]);
|
2013-02-21 22:37:17 +00:00
|
|
|
|
2014-06-27 00:17:46 +00:00
|
|
|
GcMethodAddendum* addendum = method->addendum();
|
2013-02-21 22:37:17 +00:00
|
|
|
if (addendum) {
|
2014-06-27 00:17:16 +00:00
|
|
|
object table = addendum->annotationTable();
|
2013-02-21 22:37:17 +00:00
|
|
|
if (table) {
|
|
|
|
for (unsigned i = 0; i < objectArrayLength(t, table); ++i) {
|
|
|
|
if (objectArrayBody(t, objectArrayBody(t, table, i), 1)
|
2014-07-11 15:50:18 +00:00
|
|
|
== reinterpret_cast<object>(arguments[2])) {
|
2013-02-21 22:37:17 +00:00
|
|
|
PROTECT(t, method);
|
|
|
|
PROTECT(t, table);
|
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
GcMethod* get
|
|
|
|
= resolveMethod(t,
|
|
|
|
roots(t)->bootLoader(),
|
|
|
|
"avian/Classes",
|
|
|
|
"getAnnotation",
|
|
|
|
"(Ljava/lang/ClassLoader;[Ljava/lang/Object;)"
|
|
|
|
"Ljava/lang/annotation/Annotation;");
|
2013-02-21 22:37:17 +00:00
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
return reinterpret_cast<uintptr_t>(
|
|
|
|
t->m->processor->invoke(t,
|
|
|
|
get,
|
|
|
|
0,
|
|
|
|
method->class_()->loader(),
|
|
|
|
objectArrayBody(t, table, i)));
|
2013-02-21 22:37:17 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_reflect_Method_getDeclaredAnnotations(Thread* t,
|
|
|
|
object,
|
|
|
|
uintptr_t* arguments)
|
2013-02-21 22:37:17 +00:00
|
|
|
{
|
2014-07-11 15:47:57 +00:00
|
|
|
GcMethod* method = cast<GcMethod>(
|
|
|
|
t,
|
|
|
|
cast<GcArray>(t,
|
|
|
|
cast<GcJclass>(t, reinterpret_cast<object>(arguments[0]))
|
|
|
|
->vmClass()
|
|
|
|
->methodTable())->body()[arguments[1]]);
|
2013-02-21 22:37:17 +00:00
|
|
|
|
2014-06-27 00:17:46 +00:00
|
|
|
GcMethodAddendum* addendum = method->addendum();
|
2013-02-21 22:37:17 +00:00
|
|
|
if (addendum) {
|
2014-06-27 00:17:16 +00:00
|
|
|
object table = addendum->annotationTable();
|
2013-02-21 22:37:17 +00:00
|
|
|
if (table) {
|
|
|
|
PROTECT(t, method);
|
|
|
|
PROTECT(t, table);
|
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
object array = makeObjectArray(
|
|
|
|
t,
|
|
|
|
resolveClass(
|
|
|
|
t, roots(t)->bootLoader(), "java/lang/annotation/Annotation"),
|
|
|
|
objectArrayLength(t, table));
|
2013-02-21 22:37:17 +00:00
|
|
|
PROTECT(t, array);
|
2014-07-11 15:47:57 +00:00
|
|
|
|
|
|
|
GcMethod* get
|
|
|
|
= resolveMethod(t,
|
|
|
|
roots(t)->bootLoader(),
|
|
|
|
"avian/Classes",
|
|
|
|
"getAnnotation",
|
|
|
|
"(Ljava/lang/ClassLoader;[Ljava/lang/Object;)"
|
|
|
|
"Ljava/lang/annotation/Annotation;");
|
2013-02-21 22:37:17 +00:00
|
|
|
PROTECT(t, get);
|
|
|
|
|
|
|
|
for (unsigned i = 0; i < objectArrayLength(t, table); ++i) {
|
2014-07-11 15:47:57 +00:00
|
|
|
object a = t->m->processor->invoke(t,
|
|
|
|
get,
|
|
|
|
0,
|
|
|
|
method->class_()->loader(),
|
|
|
|
objectArrayBody(t, table, i));
|
2013-02-21 22:37:17 +00:00
|
|
|
|
2014-07-10 23:51:38 +00:00
|
|
|
setField(t, array, ArrayBody + (i * BytesPerWord), a);
|
2013-02-21 22:37:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return reinterpret_cast<uintptr_t>(array);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
return reinterpret_cast<uintptr_t>(makeObjectArray(
|
|
|
|
t,
|
|
|
|
resolveClass(
|
|
|
|
t, roots(t)->bootLoader(), "java/lang/annotation/Annotation"),
|
2013-02-21 22:37:17 +00:00
|
|
|
0));
|
|
|
|
}
|
|
|
|
|
2014-05-07 20:43:29 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
|
|
|
Avian_java_lang_reflect_Field_getDeclaredAnnotations(Thread* t,
|
|
|
|
object,
|
|
|
|
uintptr_t* arguments)
|
|
|
|
{
|
2014-07-11 15:47:57 +00:00
|
|
|
GcField* field = cast<GcField>(
|
2014-05-07 20:43:29 +00:00
|
|
|
t,
|
2014-07-11 15:47:57 +00:00
|
|
|
cast<GcArray>(t,
|
|
|
|
cast<GcJclass>(t, reinterpret_cast<object>(arguments[0]))
|
|
|
|
->vmClass()
|
|
|
|
->fieldTable())->body()[arguments[1]]);
|
2014-05-07 20:43:29 +00:00
|
|
|
|
2014-06-27 00:17:16 +00:00
|
|
|
GcFieldAddendum* addendum = field->addendum();
|
2014-05-07 20:43:29 +00:00
|
|
|
if (addendum) {
|
2014-06-27 00:17:16 +00:00
|
|
|
object table = addendum->annotationTable();
|
2014-05-07 20:43:29 +00:00
|
|
|
if (table) {
|
|
|
|
PROTECT(t, field);
|
|
|
|
PROTECT(t, table);
|
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
object array = makeObjectArray(
|
|
|
|
t,
|
|
|
|
resolveClass(
|
|
|
|
t, roots(t)->bootLoader(), "java/lang/annotation/Annotation"),
|
|
|
|
objectArrayLength(t, table));
|
2014-05-07 20:43:29 +00:00
|
|
|
PROTECT(t, array);
|
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
GcMethod* get
|
|
|
|
= resolveMethod(t,
|
|
|
|
roots(t)->bootLoader(),
|
|
|
|
"avian/Classes",
|
|
|
|
"getAnnotation",
|
|
|
|
"(Ljava/lang/ClassLoader;[Ljava/lang/Object;)"
|
|
|
|
"Ljava/lang/annotation/Annotation;");
|
2014-05-07 20:43:29 +00:00
|
|
|
PROTECT(t, get);
|
|
|
|
|
|
|
|
for (unsigned i = 0; i < objectArrayLength(t, table); ++i) {
|
2014-07-11 15:47:57 +00:00
|
|
|
object a = t->m->processor->invoke(
|
|
|
|
t,
|
|
|
|
get,
|
|
|
|
0,
|
|
|
|
cast<GcClass>(t, reinterpret_cast<object>(arguments[0]))->loader(),
|
|
|
|
objectArrayBody(t, table, i));
|
2014-05-07 20:43:29 +00:00
|
|
|
|
2014-07-10 23:51:38 +00:00
|
|
|
setField(t, array, ArrayBody + (i * BytesPerWord), a);
|
2014-05-07 20:43:29 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return reinterpret_cast<uintptr_t>(array);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return reinterpret_cast<uintptr_t>(makeObjectArray(
|
|
|
|
t,
|
|
|
|
resolveClass(
|
2014-06-30 01:44:41 +00:00
|
|
|
t, roots(t)->bootLoader(), "java/lang/annotation/Annotation"),
|
2014-05-07 20:43:29 +00:00
|
|
|
0));
|
|
|
|
}
|
|
|
|
|
2013-12-06 22:45:46 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_reflect_Method_getDefaultValue(Thread* t,
|
|
|
|
object,
|
|
|
|
uintptr_t* arguments)
|
2013-12-06 22:45:46 +00:00
|
|
|
{
|
2014-07-11 15:47:57 +00:00
|
|
|
GcMethod* method = cast<GcMethod>(
|
|
|
|
t,
|
|
|
|
cast<GcArray>(t,
|
|
|
|
cast<GcJclass>(t, reinterpret_cast<object>(arguments[1]))
|
|
|
|
->vmClass()
|
|
|
|
->methodTable())->body()[arguments[2]]);
|
2013-12-06 22:45:46 +00:00
|
|
|
|
2014-06-27 00:17:46 +00:00
|
|
|
GcMethodAddendum* addendum = method->addendum();
|
2013-12-06 22:45:46 +00:00
|
|
|
if (addendum) {
|
2014-07-11 15:47:57 +00:00
|
|
|
GcMethod* get
|
|
|
|
= resolveMethod(t,
|
|
|
|
roots(t)->bootLoader(),
|
|
|
|
"avian/Classes",
|
|
|
|
"getAnnotationDefaultValue",
|
|
|
|
"(Ljava/lang/ClassLoader;Lavian/MethodAddendum;)"
|
|
|
|
"Ljava/lang/Object;");
|
2013-12-06 22:45:46 +00:00
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
return reinterpret_cast<uintptr_t>(t->m->processor->invoke(
|
|
|
|
t, get, 0, method->class_()->loader(), addendum));
|
2013-12-06 22:45:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_reflect_Constructor_constructNative(Thread* t,
|
|
|
|
object,
|
|
|
|
uintptr_t* arguments)
|
2013-02-21 22:37:17 +00:00
|
|
|
{
|
|
|
|
object args = reinterpret_cast<object>(arguments[1]);
|
|
|
|
PROTECT(t, args);
|
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
GcClass* c
|
|
|
|
= cast<GcJclass>(t, reinterpret_cast<object>(arguments[2]))->vmClass();
|
2013-12-06 22:45:46 +00:00
|
|
|
PROTECT(t, c);
|
|
|
|
|
|
|
|
initClass(t, c);
|
2013-02-21 22:37:17 +00:00
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
GcMethod* method = cast<GcMethod>(
|
|
|
|
t, cast<GcArray>(t, c->methodTable())->body()[arguments[4]]);
|
2013-02-21 22:37:17 +00:00
|
|
|
PROTECT(t, method);
|
|
|
|
|
|
|
|
object instance = makeNew(t, c);
|
|
|
|
PROTECT(t, instance);
|
|
|
|
|
|
|
|
t->m->processor->invokeArray(t, method, instance, args);
|
|
|
|
|
|
|
|
return reinterpret_cast<uintptr_t>(instance);
|
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_reflect_Field_getField(Thread* t,
|
|
|
|
object,
|
|
|
|
uintptr_t* arguments)
|
2013-02-22 18:06:49 +00:00
|
|
|
{
|
2014-07-11 15:47:57 +00:00
|
|
|
GcField* field = cast<GcField>(
|
|
|
|
t,
|
|
|
|
cast<GcArray>(t,
|
|
|
|
cast<GcJclass>(t, reinterpret_cast<object>(arguments[2]))
|
|
|
|
->vmClass()
|
|
|
|
->fieldTable())->body()[arguments[4]]);
|
|
|
|
|
2013-12-06 22:45:46 +00:00
|
|
|
PROTECT(t, field);
|
2013-02-22 18:06:49 +00:00
|
|
|
|
2013-12-06 22:45:46 +00:00
|
|
|
object instance = reinterpret_cast<object>(arguments[1]);
|
|
|
|
PROTECT(t, instance);
|
|
|
|
|
|
|
|
return reinterpret_cast<intptr_t>(local::getField(t, field, instance));
|
|
|
|
}
|
|
|
|
|
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_reflect_Field_getIField(Thread* t,
|
|
|
|
object,
|
|
|
|
uintptr_t* arguments)
|
2013-12-06 22:45:46 +00:00
|
|
|
{
|
2014-07-11 15:47:57 +00:00
|
|
|
GcField* field = cast<GcField>(
|
|
|
|
t,
|
|
|
|
cast<GcArray>(t,
|
|
|
|
cast<GcJclass>(t, reinterpret_cast<object>(arguments[2]))
|
|
|
|
->vmClass()
|
|
|
|
->fieldTable())->body()[arguments[4]]);
|
|
|
|
|
2013-12-06 22:45:46 +00:00
|
|
|
PROTECT(t, field);
|
|
|
|
|
|
|
|
object instance = reinterpret_cast<object>(arguments[1]);
|
|
|
|
PROTECT(t, instance);
|
|
|
|
|
2014-06-27 00:17:46 +00:00
|
|
|
return cast<GcInt>(t, local::getField(t, field, instance))->value();
|
2013-12-06 22:45:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_reflect_Field_getJField(Thread* t,
|
|
|
|
object,
|
|
|
|
uintptr_t* arguments)
|
2013-12-06 22:45:46 +00:00
|
|
|
{
|
2014-07-11 15:47:57 +00:00
|
|
|
GcField* field = cast<GcField>(
|
|
|
|
t,
|
|
|
|
cast<GcArray>(t,
|
|
|
|
cast<GcJclass>(t, reinterpret_cast<object>(arguments[2]))
|
|
|
|
->vmClass()
|
|
|
|
->fieldTable())->body()[arguments[4]]);
|
|
|
|
|
2013-12-06 22:45:46 +00:00
|
|
|
PROTECT(t, field);
|
|
|
|
|
|
|
|
object instance = reinterpret_cast<object>(arguments[1]);
|
|
|
|
PROTECT(t, instance);
|
|
|
|
|
2014-06-27 00:17:46 +00:00
|
|
|
return cast<GcLong>(t, local::getField(t, field, instance))->value();
|
2013-12-06 22:45:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_reflect_Field_setField(Thread* t,
|
|
|
|
object,
|
|
|
|
uintptr_t* arguments)
|
2013-12-06 22:45:46 +00:00
|
|
|
{
|
2014-07-11 15:47:57 +00:00
|
|
|
GcField* field = cast<GcField>(
|
|
|
|
t,
|
|
|
|
cast<GcArray>(t,
|
|
|
|
cast<GcJclass>(t, reinterpret_cast<object>(arguments[2]))
|
|
|
|
->vmClass()
|
|
|
|
->fieldTable())->body()[arguments[4]]);
|
|
|
|
|
2013-12-06 22:45:46 +00:00
|
|
|
PROTECT(t, field);
|
|
|
|
|
|
|
|
object instance = reinterpret_cast<object>(arguments[1]);
|
|
|
|
PROTECT(t, instance);
|
|
|
|
|
|
|
|
object value = reinterpret_cast<object>(arguments[6]);
|
|
|
|
PROTECT(t, value);
|
|
|
|
|
|
|
|
local::setField(t, field, instance, value);
|
|
|
|
}
|
|
|
|
|
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_reflect_Field_setIField(Thread* t,
|
|
|
|
object,
|
|
|
|
uintptr_t* arguments)
|
2013-12-06 22:45:46 +00:00
|
|
|
{
|
2014-07-11 15:47:57 +00:00
|
|
|
GcField* field = cast<GcField>(
|
|
|
|
t,
|
|
|
|
cast<GcArray>(t,
|
|
|
|
cast<GcJclass>(t, reinterpret_cast<object>(arguments[2]))
|
|
|
|
->vmClass()
|
|
|
|
->fieldTable())->body()[arguments[4]]);
|
2013-12-06 22:45:46 +00:00
|
|
|
|
|
|
|
object instance = reinterpret_cast<object>(arguments[1]);
|
|
|
|
PROTECT(t, instance);
|
|
|
|
|
2014-05-29 04:17:25 +00:00
|
|
|
object value = reinterpret_cast<object>(makeInt(t, arguments[7]));
|
2013-12-06 22:45:46 +00:00
|
|
|
|
|
|
|
local::setField(t, field, instance, value);
|
2013-02-22 18:06:49 +00:00
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_reflect_Field_getFieldModifiers(Thread* t,
|
|
|
|
object,
|
|
|
|
uintptr_t* arguments)
|
2013-02-22 18:06:49 +00:00
|
|
|
{
|
2014-07-11 15:47:57 +00:00
|
|
|
return cast<GcField>(
|
|
|
|
t,
|
|
|
|
cast<GcArray>(
|
|
|
|
t,
|
|
|
|
cast<GcJclass>(t, reinterpret_cast<object>(arguments[1]))
|
|
|
|
->vmClass()
|
|
|
|
->fieldTable())->body()[arguments[2]])->flags();
|
2013-02-22 18:06:49 +00:00
|
|
|
}
|
|
|
|
|
2013-12-06 22:45:46 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_reflect_Field_getAnnotation(Thread* t,
|
|
|
|
object,
|
|
|
|
uintptr_t* arguments)
|
2013-12-06 22:45:46 +00:00
|
|
|
{
|
2014-07-11 15:47:57 +00:00
|
|
|
GcField* field = cast<GcField>(
|
|
|
|
t,
|
|
|
|
cast<GcArray>(t,
|
|
|
|
cast<GcJclass>(t, reinterpret_cast<object>(arguments[0]))
|
|
|
|
->vmClass()
|
|
|
|
->fieldTable())->body()[arguments[1]]);
|
2013-12-06 22:45:46 +00:00
|
|
|
|
2014-06-27 00:17:16 +00:00
|
|
|
GcFieldAddendum* addendum = field->addendum();
|
2013-12-06 22:45:46 +00:00
|
|
|
if (addendum) {
|
2014-06-27 00:17:16 +00:00
|
|
|
object table = addendum->annotationTable();
|
2013-12-06 22:45:46 +00:00
|
|
|
if (table) {
|
|
|
|
for (unsigned i = 0; i < objectArrayLength(t, table); ++i) {
|
|
|
|
if (objectArrayBody(t, objectArrayBody(t, table, i), 1)
|
2014-07-11 15:50:18 +00:00
|
|
|
== reinterpret_cast<object>(arguments[2])) {
|
2013-12-06 22:45:46 +00:00
|
|
|
PROTECT(t, field);
|
|
|
|
PROTECT(t, table);
|
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
GcMethod* get
|
|
|
|
= resolveMethod(t,
|
|
|
|
roots(t)->bootLoader(),
|
|
|
|
"avian/Classes",
|
|
|
|
"getAnnotation",
|
|
|
|
"(Ljava/lang/ClassLoader;[Ljava/lang/Object;)"
|
|
|
|
"Ljava/lang/annotation/Annotation;");
|
2013-12-06 22:45:46 +00:00
|
|
|
|
2014-07-11 15:47:57 +00:00
|
|
|
return reinterpret_cast<uintptr_t>(
|
|
|
|
t->m->processor->invoke(t,
|
|
|
|
get,
|
|
|
|
0,
|
|
|
|
field->class_()->loader(),
|
|
|
|
objectArrayBody(t, table, i)));
|
2013-12-06 22:45:46 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_reflect_Field_getSignatureAnnotation(Thread* t,
|
|
|
|
object,
|
|
|
|
uintptr_t* arguments)
|
2013-12-06 22:45:46 +00:00
|
|
|
{
|
2014-07-11 15:47:57 +00:00
|
|
|
GcField* field = cast<GcField>(
|
|
|
|
t,
|
|
|
|
cast<GcArray>(t,
|
|
|
|
cast<GcJclass>(t, reinterpret_cast<object>(arguments[1]))
|
|
|
|
->vmClass()
|
|
|
|
->fieldTable())->body()[arguments[2]]);
|
2013-12-06 22:45:46 +00:00
|
|
|
|
2014-06-27 00:17:46 +00:00
|
|
|
GcFieldAddendum* addendum = field->addendum();
|
2013-12-06 22:45:46 +00:00
|
|
|
if (addendum) {
|
2014-06-27 00:17:46 +00:00
|
|
|
GcByteArray* signature = cast<GcByteArray>(t, addendum->signature());
|
2013-12-06 22:45:46 +00:00
|
|
|
if (signature) {
|
|
|
|
object array = makeObjectArray(t, 1);
|
|
|
|
PROTECT(t, array);
|
2014-07-11 15:47:57 +00:00
|
|
|
|
|
|
|
GcString* string = t->m->classpath->makeString(
|
|
|
|
t, signature, 0, signature->length() - 1);
|
|
|
|
|
2014-07-02 21:11:27 +00:00
|
|
|
setField(t, array, ArrayBody, string);
|
2013-12-06 22:45:46 +00:00
|
|
|
|
|
|
|
return reinterpret_cast<uintptr_t>(array);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return reinterpret_cast<uintptr_t>(makeObjectArray(t, 0));
|
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_Throwable_nativeFillInStackTrace(Thread* t,
|
|
|
|
object,
|
|
|
|
uintptr_t*)
|
2013-02-21 22:37:17 +00:00
|
|
|
{
|
|
|
|
return reinterpret_cast<uintptr_t>(getTrace(t, 2));
|
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_Throwable_nativeGetStackTrace(Thread* t,
|
|
|
|
object,
|
|
|
|
uintptr_t* arguments)
|
2013-02-22 18:06:49 +00:00
|
|
|
{
|
2014-07-11 15:50:18 +00:00
|
|
|
return reinterpret_cast<uintptr_t>(
|
|
|
|
local::translateStackTrace(t, reinterpret_cast<object>(arguments[0])));
|
2013-02-22 18:06:49 +00:00
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_avian_Classes_makeMethod(Thread* t, object, uintptr_t* arguments)
|
2013-02-21 22:37:17 +00:00
|
|
|
{
|
2014-07-11 15:47:57 +00:00
|
|
|
return reinterpret_cast<uintptr_t>(local::makeMethodOrConstructor(
|
|
|
|
t,
|
|
|
|
cast<GcJclass>(t, reinterpret_cast<object>(arguments[0])),
|
|
|
|
arguments[1]));
|
2013-02-21 22:37:17 +00:00
|
|
|
}
|
2013-12-06 22:45:46 +00:00
|
|
|
|
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_avian_Classes_makeField(Thread* t, object, uintptr_t* arguments)
|
2013-12-06 22:45:46 +00:00
|
|
|
{
|
2014-07-11 15:47:57 +00:00
|
|
|
return reinterpret_cast<uintptr_t>(local::makeField(
|
|
|
|
t,
|
|
|
|
cast<GcJclass>(t, reinterpret_cast<object>(arguments[0])),
|
|
|
|
arguments[1]));
|
2013-12-06 22:45:46 +00:00
|
|
|
}
|
2013-02-21 22:37:17 +00:00
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_reflect_Array_createObjectArray(Thread* t,
|
|
|
|
object,
|
|
|
|
uintptr_t* arguments)
|
2013-02-21 22:37:17 +00:00
|
|
|
{
|
2014-07-11 15:47:57 +00:00
|
|
|
return reinterpret_cast<uintptr_t>(makeObjectArray(
|
|
|
|
t,
|
|
|
|
cast<GcJclass>(t, reinterpret_cast<object>(arguments[0]))->vmClass(),
|
2013-02-21 22:37:17 +00:00
|
|
|
arguments[1]));
|
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_nio_ByteOrder_isLittleEndian(Thread*, object, uintptr_t*)
|
2013-02-21 22:37:17 +00:00
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_dalvik_system_VMRuntime_newNonMovableArray(Thread* t,
|
|
|
|
object,
|
|
|
|
uintptr_t* arguments)
|
2013-02-21 22:37:17 +00:00
|
|
|
{
|
2014-06-27 00:17:46 +00:00
|
|
|
if (cast<GcJclass>(t, reinterpret_cast<object>(arguments[1]))->vmClass()
|
2014-07-11 15:47:57 +00:00
|
|
|
== type(t, GcJbyte::Type)) {
|
|
|
|
GcByteArray* array
|
|
|
|
= reinterpret_cast<GcByteArray*>(allocate3(t,
|
|
|
|
t->m->heap,
|
|
|
|
Machine::FixedAllocation,
|
|
|
|
ArrayBody + arguments[2],
|
|
|
|
false));
|
2013-02-21 22:37:17 +00:00
|
|
|
|
2014-07-02 21:11:27 +00:00
|
|
|
setObjectClass(t, array, type(t, GcByteArray::Type));
|
2014-06-27 00:17:46 +00:00
|
|
|
array->length() = arguments[2];
|
2013-02-21 22:37:17 +00:00
|
|
|
|
|
|
|
return reinterpret_cast<intptr_t>(array);
|
|
|
|
} else {
|
|
|
|
// todo
|
|
|
|
abort(t);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_dalvik_system_VMRuntime_addressOf(Thread*,
|
|
|
|
object,
|
|
|
|
uintptr_t* arguments)
|
2013-02-21 22:37:17 +00:00
|
|
|
{
|
|
|
|
return arguments[1] + ArrayBody;
|
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_libcore_io_Memory_pokeLong(Thread*, object, uintptr_t* arguments)
|
2013-02-21 22:37:17 +00:00
|
|
|
{
|
2014-07-11 15:50:18 +00:00
|
|
|
int64_t address;
|
|
|
|
memcpy(&address, arguments, 8);
|
|
|
|
int64_t v;
|
|
|
|
memcpy(&v, arguments + 2, 8);
|
2013-02-25 23:41:46 +00:00
|
|
|
if (arguments[4]) {
|
2013-02-21 22:37:17 +00:00
|
|
|
v = swapV8(v);
|
|
|
|
}
|
2013-02-25 23:41:46 +00:00
|
|
|
memcpy(reinterpret_cast<void*>(address), &v, 8);
|
2013-02-21 22:37:17 +00:00
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_libcore_io_Memory_peekLong(Thread*, object, uintptr_t* arguments)
|
2013-02-21 22:37:17 +00:00
|
|
|
{
|
2014-07-11 15:50:18 +00:00
|
|
|
int64_t address;
|
|
|
|
memcpy(&address, arguments, 8);
|
|
|
|
int64_t v;
|
|
|
|
memcpy(&v, reinterpret_cast<void*>(address), 8);
|
2013-02-25 23:41:46 +00:00
|
|
|
return arguments[2] ? swapV8(v) : v;
|
2013-02-21 22:37:17 +00:00
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_libcore_io_Memory_pokeInt(Thread*, object, uintptr_t* arguments)
|
2013-02-21 22:37:17 +00:00
|
|
|
{
|
2014-07-11 15:50:18 +00:00
|
|
|
int64_t address;
|
|
|
|
memcpy(&address, arguments, 8);
|
2013-02-25 23:41:46 +00:00
|
|
|
int32_t v = arguments[3] ? swapV4(arguments[2]) : arguments[2];
|
|
|
|
memcpy(reinterpret_cast<void*>(address), &v, 4);
|
2013-02-21 22:37:17 +00:00
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_libcore_io_Memory_peekInt(Thread*, object, uintptr_t* arguments)
|
2013-02-21 22:37:17 +00:00
|
|
|
{
|
2014-07-11 15:50:18 +00:00
|
|
|
int64_t address;
|
|
|
|
memcpy(&address, arguments, 8);
|
|
|
|
int32_t v;
|
|
|
|
memcpy(&v, reinterpret_cast<void*>(address), 4);
|
2013-02-25 23:41:46 +00:00
|
|
|
return arguments[2] ? swapV4(v) : v;
|
2013-02-21 22:37:17 +00:00
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_libcore_io_Memory_pokeShort(Thread*, object, uintptr_t* arguments)
|
2013-02-21 22:37:17 +00:00
|
|
|
{
|
2014-07-11 15:50:18 +00:00
|
|
|
int64_t address;
|
|
|
|
memcpy(&address, arguments, 8);
|
2013-02-25 23:41:46 +00:00
|
|
|
int16_t v = arguments[3] ? swapV2(arguments[2]) : arguments[2];
|
|
|
|
memcpy(reinterpret_cast<void*>(address), &v, 2);
|
2013-02-21 22:37:17 +00:00
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_libcore_io_Memory_peekShort(Thread*, object, uintptr_t* arguments)
|
2013-02-21 22:37:17 +00:00
|
|
|
{
|
2014-07-11 15:50:18 +00:00
|
|
|
int64_t address;
|
|
|
|
memcpy(&address, arguments, 8);
|
|
|
|
int16_t v;
|
|
|
|
memcpy(&v, reinterpret_cast<void*>(address), 2);
|
2013-02-25 23:41:46 +00:00
|
|
|
return arguments[2] ? swapV2(v) : v;
|
2013-02-21 22:37:17 +00:00
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_libcore_io_Memory_pokeByte(Thread*, object, uintptr_t* arguments)
|
2013-02-21 22:37:17 +00:00
|
|
|
{
|
2014-07-11 15:50:18 +00:00
|
|
|
int64_t address;
|
|
|
|
memcpy(&address, arguments, 8);
|
2013-02-25 23:41:46 +00:00
|
|
|
*reinterpret_cast<int8_t*>(address) = arguments[2];
|
2013-02-21 22:37:17 +00:00
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_libcore_io_Memory_peekByte(Thread*, object, uintptr_t* arguments)
|
2013-02-21 22:37:17 +00:00
|
|
|
{
|
2014-07-11 15:50:18 +00:00
|
|
|
int64_t address;
|
|
|
|
memcpy(&address, arguments, 8);
|
2013-02-25 23:41:46 +00:00
|
|
|
return *reinterpret_cast<int8_t*>(address);
|
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_System_nanoTime(Thread* t, object, uintptr_t*)
|
2013-02-21 22:37:17 +00:00
|
|
|
{
|
|
|
|
return t->m->system->now() * 1000 * 1000;
|
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_System_currentTimeMillis(Thread* t, object, uintptr_t*)
|
2013-02-21 22:37:17 +00:00
|
|
|
{
|
|
|
|
return t->m->system->now();
|
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_lang_System_identityHashCode(Thread* t,
|
|
|
|
object,
|
|
|
|
uintptr_t* arguments)
|
2013-02-21 22:37:17 +00:00
|
|
|
{
|
|
|
|
return objectHash(t, reinterpret_cast<object>(arguments[0]));
|
|
|
|
}
|
2013-02-26 23:24:02 +00:00
|
|
|
|
2014-02-24 21:16:40 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_java_util_concurrent_atomic_AtomicLong_VMSupportsCS8(Thread*,
|
|
|
|
object,
|
|
|
|
uintptr_t*)
|
2014-02-24 21:16:40 +00:00
|
|
|
{
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2013-03-14 21:33:05 +00:00
|
|
|
#ifdef PLATFORM_WINDOWS
|
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
#include <io.h>
|
2013-03-14 21:33:05 +00:00
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
void register_java_io_Console(_JNIEnv*)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
void register_java_lang_ProcessManager(_JNIEnv*)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
void register_libcore_net_RawSocket(_JNIEnv*)
|
|
|
|
{
|
|
|
|
}
|
|
|
|
// void register_org_apache_harmony_xnet_provider_jsse_NativeCrypto(_JNIEnv*) {
|
|
|
|
// }
|
2013-03-14 21:33:05 +00:00
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_libcore_io_OsConstants_initConstants(Thread* t,
|
|
|
|
object method,
|
|
|
|
uintptr_t*)
|
2013-03-14 21:33:05 +00:00
|
|
|
{
|
2014-06-27 00:17:46 +00:00
|
|
|
object c = method->class_();
|
2013-03-14 21:33:05 +00:00
|
|
|
PROTECT(t, c);
|
|
|
|
|
|
|
|
object table = classStaticTable(t, c);
|
|
|
|
PROTECT(t, table);
|
|
|
|
|
2014-06-28 23:24:24 +00:00
|
|
|
GcField* field = resolveField(t, c, "STDIN_FILENO", "I");
|
2014-06-28 21:11:31 +00:00
|
|
|
fieldAtOffset<jint>(table, field->offset()) = 0;
|
2013-03-14 21:33:05 +00:00
|
|
|
|
|
|
|
field = resolveField(t, c, "STDOUT_FILENO", "I");
|
2014-06-28 21:11:31 +00:00
|
|
|
fieldAtOffset<jint>(table, field->offset()) = 1;
|
2013-03-14 21:33:05 +00:00
|
|
|
|
|
|
|
field = resolveField(t, c, "STDERR_FILENO", "I");
|
2014-06-28 21:11:31 +00:00
|
|
|
fieldAtOffset<jint>(table, field->offset()) = 2;
|
2013-03-14 21:33:05 +00:00
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_libcore_io_Posix_getenv(Thread* t, object, uintptr_t* arguments)
|
2013-03-14 21:33:05 +00:00
|
|
|
{
|
|
|
|
object name = reinterpret_cast<object>(arguments[1]);
|
|
|
|
|
2014-06-28 23:24:24 +00:00
|
|
|
THREAD_RUNTIME_ARRAY(t, uint16_t, chars, name->length(t) + 1);
|
2013-03-14 21:33:05 +00:00
|
|
|
stringChars(t, name, RUNTIME_ARRAY_BODY(chars));
|
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
wchar_t* value
|
|
|
|
= _wgetenv(reinterpret_cast<wchar_t*>(RUNTIME_ARRAY_BODY(chars)));
|
2013-03-14 21:33:05 +00:00
|
|
|
|
|
|
|
if (value) {
|
|
|
|
unsigned size = wcslen(value);
|
2014-07-11 15:50:18 +00:00
|
|
|
|
2013-03-14 21:33:05 +00:00
|
|
|
object a = makeCharArray(t, size);
|
|
|
|
if (size) {
|
|
|
|
memcpy(&charArrayBody(t, a, 0), value, size * sizeof(jchar));
|
|
|
|
}
|
2014-07-11 15:50:18 +00:00
|
|
|
|
|
|
|
return reinterpret_cast<uintptr_t>(
|
|
|
|
t->m->classpath->makeString(t, a, 0, size));
|
2013-03-14 21:33:05 +00:00
|
|
|
} else {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_libcore_io_Posix_uname(Thread* t, object, uintptr_t*)
|
2013-03-14 21:33:05 +00:00
|
|
|
{
|
2014-07-11 15:47:57 +00:00
|
|
|
GcClass* c
|
|
|
|
= resolveClass(t, roots(t)->bootLoader(), "libcore/io/StructUtsname");
|
2013-03-14 21:33:05 +00:00
|
|
|
PROTECT(t, c);
|
2014-07-11 15:50:18 +00:00
|
|
|
|
2013-03-14 21:33:05 +00:00
|
|
|
object instance = makeNew(t, c);
|
|
|
|
PROTECT(t, instance);
|
2014-07-11 15:50:18 +00:00
|
|
|
|
2013-03-14 21:33:05 +00:00
|
|
|
#ifdef ARCH_x86_32
|
|
|
|
object arch = makeString(t, "x86");
|
|
|
|
#elif defined ARCH_x86_64
|
|
|
|
object arch = makeString(t, "x86_64");
|
|
|
|
#elif defined ARCH_arm
|
|
|
|
object arch = makeString(t, "arm");
|
|
|
|
#else
|
|
|
|
object arch = makeString(t, "unknown");
|
|
|
|
#endif
|
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
set(t,
|
|
|
|
instance,
|
|
|
|
fieldOffset(t, resolveField(t, c, "machine", "Ljava/lang/String;")),
|
|
|
|
arch);
|
2013-03-14 21:33:05 +00:00
|
|
|
|
|
|
|
object platform = makeString(t, "Windows");
|
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
set(t,
|
|
|
|
instance,
|
|
|
|
fieldOffset(t, resolveField(t, c, "sysname", "Ljava/lang/String;")),
|
|
|
|
platform);
|
2013-03-14 21:33:05 +00:00
|
|
|
|
|
|
|
object version = makeString(t, "unknown");
|
|
|
|
|
2014-07-11 15:50:18 +00:00
|
|
|
set(t,
|
|
|
|
instance,
|
|
|
|
fieldOffset(t, resolveField(t, c, "release", "Ljava/lang/String;")),
|
|
|
|
version);
|
2013-03-14 21:33:05 +00:00
|
|
|
|
|
|
|
return reinterpret_cast<uintptr_t>(instance);
|
|
|
|
}
|
|
|
|
|
2013-11-08 15:35:10 +00:00
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
2014-07-11 15:50:18 +00:00
|
|
|
Avian_libcore_io_Posix_writeBytes(Thread* t, object, uintptr_t* arguments)
|
2013-03-14 21:33:05 +00:00
|
|
|
{
|
|
|
|
object fd = reinterpret_cast<object>(arguments[1]);
|
|
|
|
PROTECT(t, fd);
|
|
|
|
|
|
|
|
object buffer = reinterpret_cast<object>(arguments[2]);
|
|
|
|
PROTECT(t, buffer);
|
|
|
|
|
|
|
|
int offset = arguments[3];
|
|
|
|
int count = arguments[4];
|
|
|
|
|
|
|
|
int d = jniGetFDFromFileDescriptor(t, &fd);
|
|
|
|
|
|
|
|
int r;
|
2014-05-29 04:17:25 +00:00
|
|
|
if (objectClass(t, buffer) == type(t, GcByteArray::Type)) {
|
2013-03-14 21:33:05 +00:00
|
|
|
void* tmp = t->m->heap->allocate(count);
|
|
|
|
memcpy(tmp, &byteArrayBody(t, buffer, offset), count);
|
2014-07-11 15:50:18 +00:00
|
|
|
{
|
|
|
|
ENTER(t, Thread::IdleState);
|
2013-03-14 21:33:05 +00:00
|
|
|
r = _write(d, tmp, count);
|
|
|
|
}
|
|
|
|
t->m->heap->free(tmp, count);
|
|
|
|
} else {
|
|
|
|
void* p = local::getDirectBufferAddress(t, buffer);
|
2014-07-11 15:50:18 +00:00
|
|
|
{
|
|
|
|
ENTER(t, Thread::IdleState);
|
2013-03-14 21:33:05 +00:00
|
|
|
r = _write(d, p, count);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (r < 0) {
|
|
|
|
THREAD_RUNTIME_ARRAY(t, char, message, 256);
|
2014-07-11 15:47:57 +00:00
|
|
|
throwNew(t,
|
|
|
|
GcRuntimeException::Type,
|
|
|
|
"writeBytes %d: %s",
|
|
|
|
d,
|
2013-03-14 21:33:05 +00:00
|
|
|
jniStrError(errno, RUNTIME_ARRAY_BODY(message), 0));
|
|
|
|
} else {
|
|
|
|
return r;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|