corda/src/classpath-android.cpp

2547 lines
68 KiB
C++
Raw Normal View History

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