mirror of
https://github.com/corda/corda.git
synced 2025-01-07 13:38:47 +00:00
34962ff334
add support for using the OpenJDK 8 class library
5440 lines
144 KiB
C++
5440 lines
144 KiB
C++
/* 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. */
|
|
|
|
#include "avian/machine.h"
|
|
#include "avian/classpath-common.h"
|
|
#include "avian/util.h"
|
|
#include "avian/process.h"
|
|
|
|
#ifdef PLATFORM_WINDOWS
|
|
|
|
# include <windows.h>
|
|
# include <io.h>
|
|
# include <direct.h>
|
|
# include <share.h>
|
|
# include <errno.h>
|
|
# include <fcntl.h>
|
|
# include <sys/stat.h>
|
|
# include <sys/types.h>
|
|
|
|
# undef interface
|
|
|
|
# define CLOSE _close
|
|
# define READ _read
|
|
# define WRITE _write
|
|
# define FSTAT _fstat
|
|
# define STAT _stat
|
|
# define LSEEK _lseek
|
|
|
|
# define S_ISSOCK(x) false
|
|
|
|
# ifdef _MSC_VER
|
|
# define S_ISREG(x) ((x) | _S_IFREG)
|
|
# define S_ISDIR(x) ((x) | _S_IFDIR)
|
|
# define S_IRUSR _S_IREAD
|
|
# define S_IWUSR _S_IWRITE
|
|
# else
|
|
# define OPEN _open
|
|
# endif
|
|
|
|
# define O_RDONLY _O_RDONLY
|
|
|
|
# if (defined AVIAN_OPENJDK_SRC) \
|
|
|| ((defined __x86_64__) && (defined __MINGW32__))
|
|
# define EXPORT(x) x
|
|
# else
|
|
# define EXPORT(x) _##x
|
|
# endif
|
|
|
|
typedef int socklen_t;
|
|
|
|
# define RTLD_DEFAULT 0
|
|
|
|
#else // not PLATFORM_WINDOWS
|
|
|
|
# include <unistd.h>
|
|
# include <limits.h>
|
|
# include <sys/types.h>
|
|
# include <sys/stat.h>
|
|
# include <sys/socket.h>
|
|
# include <sys/ioctl.h>
|
|
# include <fcntl.h>
|
|
# include <errno.h>
|
|
# include <sched.h>
|
|
# include <dlfcn.h>
|
|
|
|
# define OPEN open
|
|
# define CLOSE close
|
|
# define READ read
|
|
# define WRITE write
|
|
# define STAT stat
|
|
# define FSTAT fstat
|
|
# define LSEEK lseek
|
|
|
|
# define EXPORT(x) x
|
|
|
|
#endif // not PLATFORM_WINDOWS
|
|
|
|
#define JVM_EEXIST -100
|
|
|
|
using namespace vm;
|
|
|
|
namespace {
|
|
|
|
#ifdef _MSC_VER
|
|
inline int
|
|
OPEN(string_t path, int mask, int mode)
|
|
{
|
|
int fd;
|
|
if (_wsopen_s(&fd, path, mask, _SH_DENYNO, mode) == 0) {
|
|
return fd;
|
|
} else {
|
|
return -1;
|
|
}
|
|
}
|
|
#endif
|
|
|
|
namespace local {
|
|
|
|
const int JMM_VERSION_1_0 = 0x20010000;
|
|
|
|
struct jmmOptionalSupport {
|
|
unsigned isLowMemoryDetectionSupported : 1;
|
|
unsigned isCompilationTimeMonitoringSupported : 1;
|
|
unsigned isThreadContentionMonitoringSupported : 1;
|
|
unsigned isCurrentThreadCpuTimeSupported : 1;
|
|
unsigned isOtherThreadCpuTimeSupported : 1;
|
|
unsigned isBootClassPathSupported : 1;
|
|
unsigned isObjectMonitorUsageSupported : 1;
|
|
unsigned isSynchronizerUsageSupported : 1;
|
|
};
|
|
|
|
typedef unsigned jmmLongAttribute;
|
|
typedef unsigned jmmBoolAttribute;
|
|
typedef unsigned jmmStatisticType;
|
|
typedef unsigned jmmThresholdType;
|
|
typedef unsigned jmmVMGlobalType;
|
|
typedef unsigned jmmVMGlobalOrigin;
|
|
|
|
struct jmmVMGlobal {
|
|
jstring name;
|
|
jvalue value;
|
|
jmmVMGlobalType type;
|
|
jmmVMGlobalOrigin origin;
|
|
unsigned writeable : 1;
|
|
unsigned external : 1;
|
|
unsigned reserved : 30;
|
|
void* reserved1;
|
|
void* reserved2;
|
|
};
|
|
|
|
struct jmmExtAttributeInfo {
|
|
const char* name;
|
|
char type;
|
|
const char* description;
|
|
};
|
|
|
|
struct jmmGCStat {
|
|
jlong gc_index;
|
|
jlong start_time;
|
|
jlong end_time;
|
|
jobjectArray usage_before_gc;
|
|
jobjectArray usage_after_gc;
|
|
jint gc_ext_attribute_values_size;
|
|
jvalue* gc_ext_attribute_values;
|
|
jint num_gc_ext_attributes;
|
|
};
|
|
|
|
struct JmmInterface {
|
|
void* reserved1;
|
|
void* reserved2;
|
|
|
|
jint
|
|
(JNICALL *GetVersion)
|
|
(JNIEnv*);
|
|
|
|
jint
|
|
(JNICALL *GetOptionalSupport)
|
|
(JNIEnv*, jmmOptionalSupport*);
|
|
|
|
jobject
|
|
(JNICALL *GetInputArguments)
|
|
(JNIEnv*);
|
|
|
|
jint
|
|
(JNICALL *GetThreadInfo)
|
|
(JNIEnv*, jlongArray, jint, jobjectArray);
|
|
|
|
jobjectArray
|
|
(JNICALL *GetInputArgumentArray)
|
|
(JNIEnv*);
|
|
|
|
jobjectArray
|
|
(JNICALL *GetMemoryPools)
|
|
(JNIEnv*, jobject);
|
|
|
|
jobjectArray
|
|
(JNICALL *GetMemoryManagers)
|
|
(JNIEnv*, jobject);
|
|
|
|
jobject
|
|
(JNICALL *GetMemoryPoolUsage)
|
|
(JNIEnv*, jobject);
|
|
|
|
jobject
|
|
(JNICALL *GetPeakMemoryPoolUsage)
|
|
(JNIEnv*, jobject);
|
|
|
|
void* reserved4;
|
|
|
|
jobject
|
|
(JNICALL *GetMemoryUsage)
|
|
(JNIEnv*, jboolean);
|
|
|
|
jlong
|
|
(JNICALL *GetLongAttribute)
|
|
(JNIEnv*, jobject, jmmLongAttribute);
|
|
|
|
jboolean (JNICALL *GetBoolAttribute)
|
|
(JNIEnv*, jmmBoolAttribute);
|
|
|
|
jboolean
|
|
(JNICALL *SetBoolAttribute)
|
|
(JNIEnv*, jmmBoolAttribute, jboolean);
|
|
|
|
jint
|
|
(JNICALL *GetLongAttributes)
|
|
(JNIEnv*, jobject, jmmLongAttribute*, jint, jlong*);
|
|
|
|
jobjectArray
|
|
(JNICALL *FindCircularBlockedThreads)
|
|
(JNIEnv*);
|
|
|
|
jlong
|
|
(JNICALL *GetThreadCpuTime)
|
|
(JNIEnv*, jlong);
|
|
|
|
jobjectArray
|
|
(JNICALL *GetVMGlobalNames)
|
|
(JNIEnv*);
|
|
|
|
jint
|
|
(JNICALL *GetVMGlobals)
|
|
(JNIEnv*, jobjectArray, jmmVMGlobal*, jint);
|
|
|
|
jint
|
|
(JNICALL *GetInternalThreadTimes)
|
|
(JNIEnv*, jobjectArray, jlongArray);
|
|
|
|
jboolean
|
|
(JNICALL *ResetStatistic)
|
|
(JNIEnv*, jvalue, jmmStatisticType);
|
|
|
|
void
|
|
(JNICALL *SetPoolSensor)
|
|
(JNIEnv*, jobject, jmmThresholdType, jobject);
|
|
|
|
jlong
|
|
(JNICALL *SetPoolThreshold)
|
|
(JNIEnv*, jobject, jmmThresholdType, jlong);
|
|
|
|
jobject
|
|
(JNICALL *GetPoolCollectionUsage)
|
|
(JNIEnv*, jobject);
|
|
|
|
jint
|
|
(JNICALL *GetGCExtAttributeInfo)
|
|
(JNIEnv*, jobject, jmmExtAttributeInfo*, jint);
|
|
|
|
void
|
|
(JNICALL *GetLastGCStat)
|
|
(JNIEnv*, jobject, jmmGCStat*);
|
|
|
|
jlong
|
|
(JNICALL *GetThreadCpuTimeWithKind)
|
|
(JNIEnv*, jlong, jboolean);
|
|
|
|
void* reserved5;
|
|
|
|
jint
|
|
(JNICALL *DumpHeap0)
|
|
(JNIEnv*, jstring, jboolean);
|
|
|
|
jobjectArray
|
|
(JNICALL *FindDeadlocks)
|
|
(JNIEnv*, jboolean);
|
|
|
|
void
|
|
(JNICALL *SetVMGlobal)
|
|
(JNIEnv*, jstring, jvalue );
|
|
|
|
void* reserved6;
|
|
|
|
jobjectArray
|
|
(JNICALL *DumpThreads)
|
|
(JNIEnv*, jlongArray, jboolean, jboolean);
|
|
};
|
|
|
|
const unsigned InterfaceVersion = 4;
|
|
const unsigned PageSize = 4 * 1024;
|
|
#ifdef AVIAN_OPENJDK_SRC
|
|
const int VirtualFileBase = 1000000000;
|
|
#endif
|
|
|
|
Machine* globalMachine;
|
|
|
|
const char*
|
|
primitiveName(Thread* t, object c)
|
|
{
|
|
if (c == primitiveClass(t, 'V')) {
|
|
return "void";
|
|
} else if (c == primitiveClass(t, 'Z')) {
|
|
return "boolean";
|
|
} else if (c == primitiveClass(t, 'B')) {
|
|
return "byte";
|
|
} else if (c == primitiveClass(t, 'C')) {
|
|
return "char";
|
|
} else if (c == primitiveClass(t, 'S')) {
|
|
return "short";
|
|
} else if (c == primitiveClass(t, 'I')) {
|
|
return "int";
|
|
} else if (c == primitiveClass(t, 'F')) {
|
|
return "float";
|
|
} else if (c == primitiveClass(t, 'J')) {
|
|
return "long";
|
|
} else if (c == primitiveClass(t, 'D')) {
|
|
return "double";
|
|
} else {
|
|
abort(t);
|
|
}
|
|
}
|
|
|
|
object
|
|
getClassName(Thread* t, object c)
|
|
{
|
|
if (className(t, c) == 0) {
|
|
if (classVmFlags(t, c) & PrimitiveFlag) {
|
|
PROTECT(t, c);
|
|
|
|
object name = makeByteArray(t, primitiveName(t, c));
|
|
|
|
set(t, c, ClassName, name);
|
|
} else {
|
|
abort(t);
|
|
}
|
|
}
|
|
|
|
return className(t, c);
|
|
}
|
|
|
|
object
|
|
makeClassNameString(Thread* t, object name)
|
|
{
|
|
THREAD_RUNTIME_ARRAY(t, char, s, byteArrayLength(t, name));
|
|
replace('/', '.', RUNTIME_ARRAY_BODY(s),
|
|
reinterpret_cast<char*>(&byteArrayBody(t, name, 0)));
|
|
|
|
return makeString(t, "%s", RUNTIME_ARRAY_BODY(s));
|
|
}
|
|
|
|
object
|
|
makeJmethod(Thread* t, object vmMethod, int index = -1);
|
|
|
|
object
|
|
makeJconstructor(Thread* t, object vmMethod, int index = -1);
|
|
|
|
object
|
|
makeJfield(Thread* t, object vmField, int index = -1);
|
|
|
|
#ifdef AVIAN_OPENJDK_SRC
|
|
void
|
|
interceptFileOperations(Thread*, bool);
|
|
#endif
|
|
|
|
class MyClasspath : public Classpath {
|
|
public:
|
|
MyClasspath(System* s, Allocator* allocator, const char* javaHome,
|
|
const char* embedPrefix):
|
|
allocator(allocator), ranNetOnLoad(0), ranManagementOnLoad(0)
|
|
{
|
|
class StringBuilder {
|
|
public:
|
|
StringBuilder(System* s, Allocator* allocator):
|
|
s(s),
|
|
allocator(allocator),
|
|
bufferSize(1024),
|
|
buffer(static_cast<char*>(allocator->allocate(bufferSize))),
|
|
offset(0)
|
|
{ }
|
|
|
|
void ensure(unsigned capacity) {
|
|
if (capacity > bufferSize) {
|
|
unsigned size = max(bufferSize * 2, capacity);
|
|
char* b = static_cast<char*>(allocator->allocate(size));
|
|
|
|
if (offset) {
|
|
memcpy(b, buffer, offset);
|
|
}
|
|
|
|
allocator->free(buffer, bufferSize);
|
|
|
|
buffer = b;
|
|
bufferSize = size;
|
|
}
|
|
}
|
|
|
|
void append(const char* append) {
|
|
unsigned length = strlen(append);
|
|
ensure(offset + length + 1);
|
|
|
|
strncpy(buffer + offset, append, length + 1);
|
|
|
|
offset += length;
|
|
}
|
|
|
|
void append(char c) {
|
|
ensure(2);
|
|
|
|
buffer[offset] = c;
|
|
buffer[offset + 1] = 0;
|
|
|
|
++ offset;
|
|
}
|
|
|
|
System* s;
|
|
Allocator* allocator;
|
|
unsigned bufferSize;
|
|
char* buffer;
|
|
unsigned offset;
|
|
} sb(s, allocator);
|
|
|
|
unsigned javaHomeOffset = sb.offset;
|
|
sb.append(javaHome);
|
|
sb.append('\0');
|
|
|
|
unsigned classpathOffset = sb.offset;
|
|
sb.append(AVIAN_CLASSPATH);
|
|
sb.append(s->pathSeparator());
|
|
sb.append(javaHome);
|
|
sb.append("/lib/rt.jar");
|
|
sb.append(s->pathSeparator());
|
|
sb.append(javaHome);
|
|
sb.append("/lib/jsse.jar");
|
|
sb.append(s->pathSeparator());
|
|
sb.append(javaHome);
|
|
sb.append("/lib/jce.jar");
|
|
sb.append(s->pathSeparator());
|
|
sb.append(javaHome);
|
|
sb.append("/lib/ext/sunjce_provider.jar");
|
|
sb.append(s->pathSeparator());
|
|
sb.append(javaHome);
|
|
sb.append("/lib/resources.jar");
|
|
sb.append('\0');
|
|
|
|
unsigned libraryPathOffset = sb.offset;
|
|
sb.append(javaHome);
|
|
#ifdef PLATFORM_WINDOWS
|
|
# define LIB_DIR "/bin"
|
|
#elif defined __APPLE__
|
|
# define LIB_DIR "/lib"
|
|
#elif defined ARCH_x86_64
|
|
# define LIB_DIR "/lib/amd64"
|
|
#elif defined ARCH_arm
|
|
# define LIB_DIR "/lib/arm"
|
|
#else
|
|
// todo: handle other architectures
|
|
# define LIB_DIR "/lib/i386"
|
|
#endif
|
|
|
|
#ifdef PLATFORM_WINDOWS
|
|
sb.append(LIB_DIR);
|
|
#else
|
|
sb.append(LIB_DIR ":");
|
|
sb.append(javaHome);
|
|
sb.append(LIB_DIR "/xawt");
|
|
#endif
|
|
sb.append('\0');
|
|
|
|
unsigned tzMappingsOffset = sb.offset;
|
|
sb.append(javaHome);
|
|
sb.append("/lib/tzmappings");
|
|
this->tzMappingsLength = sb.offset - tzMappingsOffset;
|
|
sb.append('\0');
|
|
|
|
unsigned embedPrefixOffset = sb.offset;
|
|
sb.append(embedPrefix);
|
|
this->embedPrefixLength = sb.offset - embedPrefixOffset;
|
|
|
|
this->javaHome = sb.buffer + javaHomeOffset;
|
|
this->classpath = sb.buffer + classpathOffset;
|
|
this->libraryPath = sb.buffer + libraryPathOffset;
|
|
this->tzMappings = sb.buffer + tzMappingsOffset;
|
|
this->embedPrefix = sb.buffer + embedPrefixOffset;
|
|
this->buffer = sb.buffer;
|
|
this->bufferSize = sb.bufferSize;
|
|
}
|
|
|
|
virtual object
|
|
makeJclass(Thread* t, object class_)
|
|
{
|
|
PROTECT(t, class_);
|
|
|
|
object name = makeClassNameString(t, getClassName(t, class_));
|
|
PROTECT(t, name);
|
|
|
|
object c = allocate(t, FixedSizeOfJclass, true);
|
|
setObjectClass(t, c, type(t, Machine::JclassType));
|
|
set(t, c, JclassName, name);
|
|
set(t, c, JclassVmClass, class_);
|
|
|
|
return c;
|
|
}
|
|
|
|
virtual object
|
|
makeString(Thread* t, object array, int32_t offset, int32_t length)
|
|
{
|
|
if (objectClass(t, array) == type(t, Machine::ByteArrayType)) {
|
|
PROTECT(t, array);
|
|
|
|
object charArray = makeCharArray(t, length);
|
|
for (int i = 0; i < length; ++i) {
|
|
if (byteArrayBody(t, array, offset + i) & 0x80) {
|
|
object constructor = resolveMethod
|
|
(t, type(t, Machine::StringType), "<init>",
|
|
"([BIILjava/lang/String;)V");
|
|
PROTECT(t, constructor);
|
|
|
|
object utf8 = vm::makeString(t, "UTF8");
|
|
PROTECT(t, utf8);
|
|
|
|
object s = makeNew(t, type(t, Machine::StringType));
|
|
PROTECT(t, s);
|
|
|
|
t->m->processor->invoke
|
|
(t, constructor, s, array, offset, length, utf8);
|
|
|
|
return s;
|
|
}
|
|
|
|
charArrayBody(t, charArray, i) = byteArrayBody(t, array, offset + i);
|
|
}
|
|
|
|
array = charArray;
|
|
} else {
|
|
expect(t, objectClass(t, array) == type(t, Machine::CharArrayType));
|
|
}
|
|
|
|
return vm::makeString(t, array, offset, length, 0);
|
|
}
|
|
|
|
virtual object
|
|
makeThread(Thread* t, Thread* parent)
|
|
{
|
|
const unsigned MaxPriority = 10;
|
|
const unsigned NormalPriority = 5;
|
|
|
|
object group;
|
|
if (parent) {
|
|
group = threadGroup(t, parent->javaThread);
|
|
} else {
|
|
group = allocate(t, FixedSizeOfThreadGroup, true);
|
|
setObjectClass(t, group, type(t, Machine::ThreadGroupType));
|
|
threadGroupMaxPriority(t, group) = MaxPriority;
|
|
}
|
|
|
|
PROTECT(t, group);
|
|
|
|
object thread = allocate(t, FixedSizeOfThread, true);
|
|
setObjectClass(t, thread, type(t, Machine::ThreadType));
|
|
threadPriority(t, thread) = NormalPriority;
|
|
threadGroup(t, thread) = group;
|
|
threadContextClassLoader(t, thread) = root(t, Machine::AppLoader);
|
|
|
|
PROTECT(t, thread);
|
|
|
|
object blockerLock = makeJobject(t);
|
|
set(t, thread, ThreadBlockerLock, blockerLock);
|
|
|
|
const unsigned BufferSize = 256;
|
|
char buffer[BufferSize];
|
|
unsigned length = vm::snprintf(buffer, BufferSize, "Thread-%p", thread);
|
|
object name = makeCharArray(t, length);
|
|
for (unsigned i = 0; i < length; ++i) {
|
|
charArrayBody(t, name, i) = buffer[i];
|
|
}
|
|
set(t, thread, ThreadName, name);
|
|
|
|
return thread;
|
|
}
|
|
|
|
virtual object
|
|
makeJMethod(Thread* t, object vmMethod)
|
|
{
|
|
PROTECT(t, vmMethod);
|
|
|
|
return byteArrayBody(t, methodName(t, vmMethod), 0) == '<'
|
|
? makeJconstructor(t, vmMethod)
|
|
: makeJmethod(t, vmMethod);
|
|
}
|
|
|
|
virtual object
|
|
getVMMethod(Thread* t, object jmethod)
|
|
{
|
|
return objectClass(t, jmethod) == type(t, Machine::JmethodType)
|
|
? arrayBody
|
|
(t, classMethodTable
|
|
(t, jclassVmClass(t, jmethodClazz(t, jmethod))),
|
|
jmethodSlot(t, jmethod))
|
|
: arrayBody
|
|
(t, classMethodTable
|
|
(t, jclassVmClass(t, jconstructorClazz(t, jmethod))),
|
|
jconstructorSlot(t, jmethod));
|
|
}
|
|
|
|
virtual object
|
|
makeJField(Thread* t, object vmField)
|
|
{
|
|
return makeJfield(t, vmField);
|
|
}
|
|
|
|
virtual object
|
|
getVMField(Thread* t, object jfield)
|
|
{
|
|
return arrayBody
|
|
(t, classFieldTable
|
|
(t, jclassVmClass(t, jfieldClazz(t, jfield))), jfieldSlot(t, jfield));
|
|
}
|
|
|
|
virtual void
|
|
clearInterrupted(Thread* t)
|
|
{
|
|
vm::clearInterrupted(t);
|
|
}
|
|
|
|
virtual void
|
|
runThread(Thread* t)
|
|
{
|
|
// force monitor creation so we don't get an OutOfMemory error
|
|
// later when we try to acquire it:
|
|
objectMonitor(t, t->javaThread, true);
|
|
|
|
THREAD_RESOURCE0(t, {
|
|
vm::acquire(t, t->javaThread);
|
|
t->flags &= ~Thread::ActiveFlag;
|
|
vm::notifyAll(t, t->javaThread);
|
|
vm::release(t, t->javaThread);
|
|
|
|
object e = t->exception;
|
|
PROTECT(t, e);
|
|
|
|
t->exception = 0;
|
|
|
|
t->m->processor->invoke
|
|
(t, root(t, Machine::ThreadTerminated),
|
|
threadGroup(t, t->javaThread), t->javaThread);
|
|
|
|
t->exception = e;
|
|
});
|
|
|
|
object method = resolveMethod
|
|
(t, root(t, Machine::BootLoader), "java/lang/Thread", "run", "()V");
|
|
|
|
t->m->processor->invoke(t, method, t->javaThread);
|
|
}
|
|
|
|
virtual void
|
|
resolveNative(Thread* t, object method)
|
|
{
|
|
if (strcmp(reinterpret_cast<const int8_t*>("sun/font/SunFontManager"),
|
|
&byteArrayBody(t, className(t, methodClass(t, method)), 0)) == 0
|
|
and strcmp(reinterpret_cast<const int8_t*>("initIDs"),
|
|
&byteArrayBody(t, methodName(t, method), 0)) == 0
|
|
and strcmp(reinterpret_cast<const int8_t*>("()V"),
|
|
&byteArrayBody(t, methodSpec(t, method), 0)) == 0)
|
|
{
|
|
PROTECT(t, method);
|
|
|
|
expect(t, loadLibrary(t, libraryPath, "fontmanager", true, true));
|
|
}
|
|
|
|
vm::resolveNative(t, method);
|
|
}
|
|
|
|
virtual void
|
|
interceptMethods(Thread* t UNUSED)
|
|
{
|
|
#ifdef AVIAN_OPENJDK_SRC
|
|
interceptFileOperations(t, false);
|
|
#endif
|
|
}
|
|
|
|
virtual void
|
|
preBoot(Thread*)
|
|
{
|
|
// ignore
|
|
}
|
|
|
|
virtual bool mayInitClasses()
|
|
{
|
|
return true;
|
|
}
|
|
|
|
virtual void
|
|
boot(Thread* t)
|
|
{
|
|
globalMachine = t->m;
|
|
|
|
resolveSystemClass(t, root(t, Machine::BootLoader),
|
|
className(t, type(t, Machine::ClassLoaderType)));
|
|
|
|
setRoot(t, Machine::ThreadTerminated, resolveMethod
|
|
(t, root(t, Machine::BootLoader), "java/lang/ThreadGroup",
|
|
"threadTerminated", "(Ljava/lang/Thread;)V"));
|
|
|
|
#ifdef AVIAN_OPENJDK_SRC
|
|
interceptFileOperations(t, true);
|
|
#else // not AVIAN_OPENJDK_SRC
|
|
expect(t, loadLibrary(t, libraryPath, "verify", true, true));
|
|
expect(t, loadLibrary(t, libraryPath, "java", true, true));
|
|
#endif // not AVIAN_OPENJDK_SRC
|
|
|
|
{ object assertionLock = resolveField
|
|
(t, type(t, Machine::ClassLoaderType), "assertionLock",
|
|
"Ljava/lang/Object;");
|
|
|
|
set(t, root(t, Machine::BootLoader), fieldOffset(t, assertionLock),
|
|
root(t, Machine::BootLoader));
|
|
}
|
|
|
|
{ object class_ = resolveClass
|
|
(t, root(t, Machine::BootLoader), "java/util/Properties", true,
|
|
Machine::NoClassDefFoundErrorType);
|
|
|
|
PROTECT(t, class_);
|
|
|
|
object instance = makeNew(t, class_);
|
|
|
|
PROTECT(t, instance);
|
|
|
|
object constructor = resolveMethod(t, class_, "<init>", "()V");
|
|
|
|
t->m->processor->invoke(t, constructor, instance);
|
|
|
|
t->m->processor->invoke
|
|
(t, root(t, Machine::BootLoader), "java/lang/System",
|
|
"setProperties", "(Ljava/util/Properties;)V", 0, instance);
|
|
}
|
|
|
|
{ object constructor = resolveMethod
|
|
(t, type(t, Machine::ClassLoaderType), "<init>",
|
|
"(Ljava/lang/ClassLoader;)V");
|
|
|
|
PROTECT(t, constructor);
|
|
|
|
t->m->processor->invoke(t, constructor, root(t, Machine::BootLoader), 0);
|
|
|
|
t->m->processor->invoke
|
|
(t, constructor, root(t, Machine::AppLoader),
|
|
root(t, Machine::BootLoader));
|
|
}
|
|
|
|
{ object scl = resolveField
|
|
(t, type(t, Machine::ClassLoaderType), "scl",
|
|
"Ljava/lang/ClassLoader;");
|
|
|
|
PROTECT(t, scl);
|
|
|
|
object sclSet = resolveField
|
|
(t, type(t, Machine::ClassLoaderType), "sclSet", "Z");
|
|
|
|
set(t, classStaticTable(t, type(t, Machine::ClassLoaderType)),
|
|
fieldOffset(t, scl), root(t, Machine::AppLoader));
|
|
|
|
fieldAtOffset<uint8_t>(classStaticTable(t, type(t, Machine::ClassLoaderType)),
|
|
fieldOffset(t, sclSet)) = true;
|
|
}
|
|
|
|
t->m->processor->invoke
|
|
(t, root(t, Machine::BootLoader), "java/lang/System",
|
|
"initializeSystemClass", "()V", 0);
|
|
|
|
t->m->processor->invoke
|
|
(t, root(t, Machine::BootLoader), "sun/misc/Launcher",
|
|
"getLauncher", "()Lsun/misc/Launcher;", 0);
|
|
|
|
set(t, t->javaThread, ThreadContextClassLoader,
|
|
root(t, Machine::AppLoader));
|
|
}
|
|
|
|
virtual const char*
|
|
bootClasspath()
|
|
{
|
|
return classpath;
|
|
}
|
|
|
|
virtual object
|
|
makeDirectByteBuffer(Thread* t, void* p, jlong capacity)
|
|
{
|
|
object c = resolveClass
|
|
(t, root(t, Machine::BootLoader), "java/nio/DirectByteBuffer");
|
|
PROTECT(t, c);
|
|
|
|
object instance = makeNew(t, c);
|
|
PROTECT(t, instance);
|
|
|
|
object constructor = resolveMethod(t, c, "<init>", "(JI)V");
|
|
|
|
t->m->processor->invoke
|
|
(t, constructor, instance, reinterpret_cast<int64_t>(p),
|
|
static_cast<int32_t>(capacity));
|
|
|
|
return instance;
|
|
}
|
|
|
|
virtual void*
|
|
getDirectBufferAddress(Thread* t, object b)
|
|
{
|
|
PROTECT(t, b);
|
|
|
|
object field = resolveField(t, objectClass(t, b), "address", "J");
|
|
|
|
return reinterpret_cast<void*>
|
|
(fieldAtOffset<int64_t>(b, fieldOffset(t, field)));
|
|
}
|
|
|
|
virtual int64_t
|
|
getDirectBufferCapacity(Thread* t, object b)
|
|
{
|
|
PROTECT(t, b);
|
|
|
|
object field = resolveField
|
|
(t, objectClass(t, b), "capacity", "I");
|
|
|
|
return fieldAtOffset<int32_t>(b, fieldOffset(t, field));
|
|
}
|
|
|
|
virtual bool
|
|
canTailCall(Thread* t, object, object calleeClassName,
|
|
object calleeMethodName, object)
|
|
{
|
|
// we can't tail call System.loadLibrary or Runtime.loadLibrary
|
|
// due to their use of System.getCallerClass, which gets confused
|
|
// if we elide stack frames.
|
|
|
|
return (strcmp("loadLibrary", reinterpret_cast<char*>
|
|
(&byteArrayBody(t, calleeMethodName, 0)))
|
|
or (strcmp("java/lang/System", reinterpret_cast<char*>
|
|
(&byteArrayBody(t, calleeClassName, 0)))
|
|
and strcmp("java/lang/Runtime", reinterpret_cast<char*>
|
|
(&byteArrayBody(t, calleeClassName, 0)))))
|
|
|
|
// and we can't tail call Reflection.getCallerClass because the
|
|
// number of stack frames will be wrong
|
|
|
|
and (strcmp("getCallerClass", reinterpret_cast<char*>
|
|
(&byteArrayBody(t, calleeMethodName, 0)))
|
|
or strcmp("sun/reflect/Reflection", reinterpret_cast<char*>
|
|
(&byteArrayBody(t, calleeClassName, 0))));
|
|
}
|
|
|
|
virtual void
|
|
shutDown(Thread* t)
|
|
{
|
|
object c = resolveClass
|
|
(t, root(t, Machine::BootLoader), "java/lang/Shutdown", false);
|
|
|
|
if (c) {
|
|
object m = findMethodOrNull(t, c, "shutdown", "()V");
|
|
|
|
if (m) {
|
|
t->m->processor->invoke(t, m, 0);
|
|
}
|
|
}
|
|
}
|
|
|
|
virtual void
|
|
dispose()
|
|
{
|
|
allocator->free(buffer, bufferSize);
|
|
allocator->free(this, sizeof(*this));
|
|
}
|
|
|
|
Allocator* allocator;
|
|
const char* javaHome;
|
|
const char* classpath;
|
|
const char* libraryPath;
|
|
const char* tzMappings;
|
|
const char* embedPrefix;
|
|
char* buffer;
|
|
unsigned bufferSize;
|
|
unsigned tzMappingsLength;
|
|
unsigned embedPrefixLength;
|
|
unsigned filePathField;
|
|
unsigned fileDescriptorFdField;
|
|
unsigned fileInputStreamFdField;
|
|
unsigned zipFileJzfileField;
|
|
unsigned zipEntryNameField;
|
|
unsigned zipEntryTimeField;
|
|
unsigned zipEntryCrcField;
|
|
unsigned zipEntrySizeField;
|
|
unsigned zipEntryCsizeField;
|
|
unsigned zipEntryMethodField;
|
|
bool ranNetOnLoad;
|
|
bool ranManagementOnLoad;
|
|
JmmInterface jmmInterface;
|
|
};
|
|
|
|
struct JVM_ExceptionTableEntryType {
|
|
jint start_pc;
|
|
jint end_pc;
|
|
jint handler_pc;
|
|
jint catchType;
|
|
};
|
|
|
|
struct jvm_version_info {
|
|
unsigned jvm_version;
|
|
unsigned update_version: 8;
|
|
unsigned special_update_version: 8;
|
|
unsigned reserved1: 16;
|
|
unsigned reserved2;
|
|
unsigned is_attach_supported: 1;
|
|
unsigned is_kernel_jvm: 1;
|
|
unsigned: 30;
|
|
unsigned: 32;
|
|
unsigned: 32;
|
|
};
|
|
|
|
bool
|
|
pathEqual(const char* a, const char* b, unsigned length)
|
|
{
|
|
#ifdef PLATFORM_WINDOWS
|
|
return strncasecmp(a, b, length) == 0;
|
|
#else
|
|
return strncmp(a, b, length) == 0;
|
|
#endif
|
|
}
|
|
|
|
class EmbeddedFile {
|
|
public:
|
|
EmbeddedFile(MyClasspath* cp, const char* path, unsigned pathLength) {
|
|
if (pathEqual(cp->embedPrefix, path, cp->embedPrefixLength)) {
|
|
const char* p = path + cp->embedPrefixLength;
|
|
while (*p == '/') ++ p;
|
|
|
|
this->jar = p;
|
|
|
|
if (*p == 0) {
|
|
this->jarLength = 0;
|
|
this->path = 0;
|
|
this->pathLength = 0;
|
|
return;
|
|
}
|
|
|
|
while (*p and *p != '/') ++p;
|
|
|
|
this->jarLength = p - this->jar;
|
|
|
|
while (*p == '/') ++p;
|
|
|
|
this->path = p;
|
|
this->pathLength = pathLength - (p - path);
|
|
} else {
|
|
this->jar = 0;
|
|
this->jarLength =0;
|
|
this->path = 0;
|
|
this->pathLength = 0;
|
|
}
|
|
}
|
|
|
|
const char* jar;
|
|
const char* path;
|
|
unsigned jarLength;
|
|
unsigned pathLength;
|
|
};
|
|
|
|
#ifdef AVIAN_OPENJDK_SRC
|
|
int64_t JNICALL
|
|
getFileAttributes
|
|
(Thread* t, object method, uintptr_t* arguments)
|
|
{
|
|
const unsigned Exists = 1;
|
|
const unsigned Regular = 2;
|
|
const unsigned Directory = 4;
|
|
|
|
MyClasspath* cp = static_cast<MyClasspath*>(t->m->classpath);
|
|
|
|
object file = reinterpret_cast<object>(arguments[1]);
|
|
object path = fieldAtOffset<object>(file, cp->filePathField);
|
|
|
|
THREAD_RUNTIME_ARRAY(t, char, p, stringLength(t, path) + 1);
|
|
stringChars(t, path, RUNTIME_ARRAY_BODY(p));
|
|
replace('\\', '/', RUNTIME_ARRAY_BODY(p));
|
|
|
|
EmbeddedFile ef(cp, RUNTIME_ARRAY_BODY(p), stringLength(t, path));
|
|
if (ef.jar) {
|
|
if (ef.jarLength == 0) {
|
|
return Exists | Directory;
|
|
}
|
|
|
|
Finder* finder = getFinder(t, ef.jar, ef.jarLength);
|
|
if (finder) {
|
|
if (ef.pathLength == 0) {
|
|
return Exists | Directory;
|
|
}
|
|
|
|
unsigned length;
|
|
System::FileType type = finder->stat(ef.path, &length, true);
|
|
switch (type) {
|
|
case System::TypeUnknown: return Exists;
|
|
case System::TypeDoesNotExist: return 0;
|
|
case System::TypeFile: return Exists | Regular;
|
|
case System::TypeDirectory: return Exists | Directory;
|
|
default: abort(t);
|
|
}
|
|
} else {
|
|
return 0;
|
|
}
|
|
} else {
|
|
return intValue
|
|
(t, t->m->processor->invoke
|
|
(t, nativeInterceptOriginal
|
|
(t, methodRuntimeDataNative(t, getMethodRuntimeData(t, method))),
|
|
reinterpret_cast<object>(arguments[0]), file));
|
|
}
|
|
}
|
|
|
|
int64_t JNICALL
|
|
checkFileAccess
|
|
(Thread* t, object method, uintptr_t* arguments)
|
|
{
|
|
const unsigned Read = 4;
|
|
|
|
MyClasspath* cp = static_cast<MyClasspath*>(t->m->classpath);
|
|
|
|
object file = reinterpret_cast<object>(arguments[1]);
|
|
unsigned mask = arguments[2];
|
|
object path = fieldAtOffset<object>(file, cp->filePathField);
|
|
|
|
THREAD_RUNTIME_ARRAY(t, char, p, stringLength(t, path) + 1);
|
|
stringChars(t, path, RUNTIME_ARRAY_BODY(p));
|
|
replace('\\', '/', RUNTIME_ARRAY_BODY(p));
|
|
|
|
EmbeddedFile ef(cp, RUNTIME_ARRAY_BODY(p), stringLength(t, path));
|
|
if (ef.jar) {
|
|
if (ef.jarLength == 0) {
|
|
return mask == Read;
|
|
}
|
|
|
|
Finder* finder = getFinder(t, ef.jar, ef.jarLength);
|
|
if (finder) {
|
|
if (ef.pathLength == 0) {
|
|
return mask == Read;
|
|
}
|
|
|
|
unsigned length;
|
|
System::FileType type = finder->stat(ef.path, &length, true);
|
|
switch (type) {
|
|
case System::TypeDoesNotExist: return false;
|
|
case System::TypeUnknown:
|
|
case System::TypeFile:
|
|
case System::TypeDirectory: return mask == Read;
|
|
default: abort(t);
|
|
}
|
|
} else {
|
|
return 0;
|
|
}
|
|
} else {
|
|
return intValue
|
|
(t, t->m->processor->invoke
|
|
(t, nativeInterceptOriginal
|
|
(t, methodRuntimeDataNative(t, getMethodRuntimeData(t, method))),
|
|
reinterpret_cast<object>(arguments[0]), file, mask)) != 0;
|
|
}
|
|
}
|
|
|
|
int64_t JNICALL
|
|
getFileLength
|
|
(Thread* t, object method, uintptr_t* arguments)
|
|
{
|
|
MyClasspath* cp = static_cast<MyClasspath*>(t->m->classpath);
|
|
|
|
object file = reinterpret_cast<object>(arguments[1]);
|
|
object path = fieldAtOffset<object>(file, cp->filePathField);
|
|
|
|
THREAD_RUNTIME_ARRAY(t, char, p, stringLength(t, path) + 1);
|
|
stringChars(t, path, RUNTIME_ARRAY_BODY(p));
|
|
replace('\\', '/', RUNTIME_ARRAY_BODY(p));
|
|
|
|
EmbeddedFile ef(cp, RUNTIME_ARRAY_BODY(p), stringLength(t, path));
|
|
if (ef.jar) {
|
|
if (ef.jarLength == 0) {
|
|
return 0;
|
|
}
|
|
|
|
Finder* finder = getFinder(t, ef.jar, ef.jarLength);
|
|
if (finder) {
|
|
if (ef.pathLength == 0) {
|
|
return 0;
|
|
}
|
|
|
|
unsigned fileLength;
|
|
finder->stat(ef.path, &fileLength);
|
|
return fileLength;
|
|
}
|
|
|
|
return 0;
|
|
} else {
|
|
return longValue
|
|
(t, t->m->processor->invoke
|
|
(t, nativeInterceptOriginal
|
|
(t, methodRuntimeDataNative(t, getMethodRuntimeData(t, method))),
|
|
reinterpret_cast<object>(arguments[0]), file));
|
|
}
|
|
}
|
|
|
|
void JNICALL
|
|
openFile(Thread* t, object method, uintptr_t* arguments)
|
|
{
|
|
object this_ = reinterpret_cast<object>(arguments[0]);
|
|
object path = reinterpret_cast<object>(arguments[1]);
|
|
|
|
MyClasspath* cp = static_cast<MyClasspath*>(t->m->classpath);
|
|
|
|
THREAD_RUNTIME_ARRAY(t, char, p, stringLength(t, path) + 1);
|
|
stringChars(t, path, RUNTIME_ARRAY_BODY(p));
|
|
replace('\\', '/', RUNTIME_ARRAY_BODY(p));
|
|
|
|
EmbeddedFile ef(cp, RUNTIME_ARRAY_BODY(p), stringLength(t, path));
|
|
if (ef.jar) {
|
|
if (ef.jarLength == 0 or ef.pathLength == 0) {
|
|
throwNew(t, Machine::FileNotFoundExceptionType);
|
|
}
|
|
|
|
Finder* finder = getFinder(t, ef.jar, ef.jarLength);
|
|
if (finder == 0) {
|
|
throwNew(t, Machine::FileNotFoundExceptionType);
|
|
}
|
|
|
|
System::Region* r = finder->find(ef.path);
|
|
if (r == 0) {
|
|
throwNew(t, Machine::FileNotFoundExceptionType);
|
|
}
|
|
|
|
PROTECT(t, this_);
|
|
|
|
ACQUIRE(t, t->m->referenceLock);
|
|
|
|
int index = -1;
|
|
unsigned oldLength = root(t, Machine::VirtualFiles)
|
|
? arrayLength(t, root(t, Machine::VirtualFiles)) : 0;
|
|
|
|
for (unsigned i = 0; i < oldLength; ++i) {
|
|
if (arrayBody(t, root(t, Machine::VirtualFiles), i) == 0) {
|
|
index = i;
|
|
break;
|
|
}
|
|
}
|
|
|
|
if (index == -1) {
|
|
object newArray = growArray(t, root(t, Machine::VirtualFiles));
|
|
setRoot(t, Machine::VirtualFiles, newArray);
|
|
index = oldLength;
|
|
}
|
|
|
|
object region = makeRegion(t, r, 0);
|
|
set(t, root(t, Machine::VirtualFiles), ArrayBody + (index * BytesPerWord),
|
|
region);
|
|
|
|
fieldAtOffset<int32_t>
|
|
(fieldAtOffset<object>
|
|
(this_, cp->fileInputStreamFdField), cp->fileDescriptorFdField)
|
|
= index + VirtualFileBase;
|
|
} else {
|
|
t->m->processor->invoke
|
|
(t, nativeInterceptOriginal
|
|
(t, methodRuntimeDataNative(t, getMethodRuntimeData(t, method))),
|
|
this_, path);
|
|
}
|
|
}
|
|
|
|
int64_t JNICALL
|
|
readByteFromFile(Thread* t, object method, uintptr_t* arguments)
|
|
{
|
|
object this_ = reinterpret_cast<object>(arguments[0]);
|
|
|
|
MyClasspath* cp = static_cast<MyClasspath*>(t->m->classpath);
|
|
|
|
int fd = fieldAtOffset<int32_t>
|
|
(fieldAtOffset<object>
|
|
(this_, cp->fileInputStreamFdField), cp->fileDescriptorFdField);
|
|
|
|
if (fd >= VirtualFileBase) {
|
|
ACQUIRE(t, t->m->referenceLock);
|
|
|
|
object region = arrayBody
|
|
(t, root(t, Machine::VirtualFiles), fd - VirtualFileBase);
|
|
|
|
if (region) {
|
|
System::Region* r = static_cast<System::Region*>
|
|
(regionRegion(t, region));
|
|
|
|
if (r->length() > regionPosition(t, region)) {
|
|
return r->start()[regionPosition(t, region)++];
|
|
} else {
|
|
return -1;
|
|
}
|
|
} else {
|
|
throwNew(t, Machine::IoExceptionType);
|
|
}
|
|
} else {
|
|
return intValue
|
|
(t, t->m->processor->invoke
|
|
(t, nativeInterceptOriginal
|
|
(t, methodRuntimeDataNative(t, getMethodRuntimeData(t, method))),
|
|
this_));
|
|
}
|
|
}
|
|
|
|
int64_t JNICALL
|
|
readBytesFromFile(Thread* t, object method, uintptr_t* arguments)
|
|
{
|
|
object this_ = reinterpret_cast<object>(arguments[0]);
|
|
object dst = reinterpret_cast<object>(arguments[1]);
|
|
int32_t offset = arguments[2];
|
|
int32_t length = arguments[3];
|
|
|
|
MyClasspath* cp = static_cast<MyClasspath*>(t->m->classpath);
|
|
|
|
int fd = fieldAtOffset<int32_t>
|
|
(fieldAtOffset<object>
|
|
(this_, cp->fileInputStreamFdField), cp->fileDescriptorFdField);
|
|
|
|
if (fd >= VirtualFileBase) {
|
|
PROTECT(t, dst);
|
|
|
|
ACQUIRE(t, t->m->referenceLock);
|
|
|
|
object region = arrayBody
|
|
(t, root(t, Machine::VirtualFiles), fd - VirtualFileBase);
|
|
|
|
if (region) {
|
|
System::Region* r = static_cast<System::Region*>
|
|
(regionRegion(t, region));
|
|
|
|
int available = r->length() - regionPosition(t, region);
|
|
if (available == 0) {
|
|
return -1;
|
|
}
|
|
|
|
if (length > available) {
|
|
length = available;
|
|
}
|
|
|
|
memcpy(&byteArrayBody(t, dst, offset),
|
|
r->start() + regionPosition(t, region),
|
|
length);
|
|
|
|
regionPosition(t, region) += length;
|
|
|
|
return length;
|
|
} else {
|
|
throwNew(t, Machine::IoExceptionType);
|
|
}
|
|
} else {
|
|
return intValue
|
|
(t, t->m->processor->invoke
|
|
(t, nativeInterceptOriginal
|
|
(t, methodRuntimeDataNative(t, getMethodRuntimeData(t, method))),
|
|
this_, dst, offset, length));
|
|
}
|
|
}
|
|
|
|
int64_t JNICALL
|
|
skipBytesInFile(Thread* t, object method, uintptr_t* arguments)
|
|
{
|
|
object this_ = reinterpret_cast<object>(arguments[0]);
|
|
int64_t count; memcpy(&count, arguments + 1, 8);
|
|
|
|
MyClasspath* cp = static_cast<MyClasspath*>(t->m->classpath);
|
|
|
|
int fd = fieldAtOffset<int32_t>
|
|
(fieldAtOffset<object>
|
|
(this_, cp->fileInputStreamFdField), cp->fileDescriptorFdField);
|
|
|
|
if (fd >= VirtualFileBase) {
|
|
ACQUIRE(t, t->m->referenceLock);
|
|
|
|
object region = arrayBody
|
|
(t, root(t, Machine::VirtualFiles), fd - VirtualFileBase);
|
|
|
|
if (region) {
|
|
System::Region* r = static_cast<System::Region*>
|
|
(regionRegion(t, region));
|
|
|
|
int available = r->length() - regionPosition(t, region);
|
|
if (count > available) {
|
|
count = available;
|
|
}
|
|
|
|
regionPosition(t, region) += count;
|
|
|
|
return count;
|
|
} else {
|
|
throwNew(t, Machine::IoExceptionType);
|
|
}
|
|
} else {
|
|
return longValue
|
|
(t, t->m->processor->invoke
|
|
(t, nativeInterceptOriginal
|
|
(t, methodRuntimeDataNative(t, getMethodRuntimeData(t, method))),
|
|
this_, count));
|
|
}
|
|
}
|
|
|
|
int64_t JNICALL
|
|
availableBytesInFile(Thread* t, object method, uintptr_t* arguments)
|
|
{
|
|
object this_ = reinterpret_cast<object>(arguments[0]);
|
|
|
|
MyClasspath* cp = static_cast<MyClasspath*>(t->m->classpath);
|
|
|
|
int fd = fieldAtOffset<int32_t>
|
|
(fieldAtOffset<object>
|
|
(this_, cp->fileInputStreamFdField), cp->fileDescriptorFdField);
|
|
|
|
if (fd >= VirtualFileBase) {
|
|
ACQUIRE(t, t->m->referenceLock);
|
|
|
|
object region = arrayBody
|
|
(t, root(t, Machine::VirtualFiles), fd - VirtualFileBase);
|
|
|
|
if (region) {
|
|
return static_cast<System::Region*>(regionRegion(t, region))->length()
|
|
- regionPosition(t, region);
|
|
} else {
|
|
throwNew(t, Machine::IoExceptionType);
|
|
}
|
|
} else {
|
|
object r = t->m->processor->invoke
|
|
(t, nativeInterceptOriginal
|
|
(t, methodRuntimeDataNative(t, getMethodRuntimeData(t, method))),
|
|
this_);
|
|
|
|
return r ? intValue(t, r) : 0;
|
|
}
|
|
}
|
|
|
|
void JNICALL
|
|
closeFile(Thread* t, object method, uintptr_t* arguments)
|
|
{
|
|
object this_ = reinterpret_cast<object>(arguments[0]);
|
|
|
|
MyClasspath* cp = static_cast<MyClasspath*>(t->m->classpath);
|
|
|
|
int fd = fieldAtOffset<int32_t>
|
|
(fieldAtOffset<object>
|
|
(this_, cp->fileInputStreamFdField), cp->fileDescriptorFdField);
|
|
|
|
if (fd >= VirtualFileBase) {
|
|
ACQUIRE(t, t->m->referenceLock);
|
|
|
|
int index = fd - VirtualFileBase;
|
|
object region = arrayBody(t, root(t, Machine::VirtualFiles), index);
|
|
|
|
if (region) {
|
|
static_cast<System::Region*>(regionRegion(t, region))->dispose();
|
|
}
|
|
|
|
set(t, root(t, Machine::VirtualFiles), ArrayBody + (index * BytesPerWord),
|
|
0);
|
|
} else {
|
|
t->m->processor->invoke
|
|
(t, nativeInterceptOriginal
|
|
(t, methodRuntimeDataNative(t, getMethodRuntimeData(t, method))),
|
|
this_);
|
|
}
|
|
}
|
|
|
|
class ZipFile {
|
|
public:
|
|
class Entry {
|
|
public:
|
|
Entry(unsigned hash, const uint8_t* start, Entry* next):
|
|
hash(hash), start(start), next(next), entry(0)
|
|
{ }
|
|
|
|
Entry(int64_t entry):
|
|
hash(0), start(0), next(0), entry(entry)
|
|
{ }
|
|
|
|
Entry():
|
|
hash(0), start(0), next(0), entry(0)
|
|
{ }
|
|
|
|
unsigned hash;
|
|
const uint8_t* start;
|
|
Entry* next;
|
|
int64_t entry;
|
|
};
|
|
|
|
ZipFile(Thread* t, System::Region* region, unsigned entryCount):
|
|
region(region),
|
|
entryCount(entryCount),
|
|
indexSize(nextPowerOfTwo(entryCount)),
|
|
index(reinterpret_cast<ZipFile::Entry**>
|
|
(t->m->heap->allocate(sizeof(ZipFile::Entry*) * indexSize))),
|
|
file(0)
|
|
{
|
|
memset(index, 0, sizeof(ZipFile::Entry*) * indexSize);
|
|
}
|
|
|
|
ZipFile(int64_t file):
|
|
region(0), entryCount(0), indexSize(0), index(0), file(file)
|
|
{ }
|
|
|
|
System::Region* region;
|
|
unsigned entryCount;
|
|
unsigned indexSize;
|
|
Entry** index;
|
|
int64_t file;
|
|
Entry entries[0];
|
|
};
|
|
|
|
int64_t JNICALL
|
|
openZipFile(Thread* t, object method, uintptr_t* arguments)
|
|
{
|
|
object path = reinterpret_cast<object>(arguments[0]);
|
|
int mode = arguments[1];
|
|
int64_t lastModified; memcpy(&lastModified, arguments + 2, 8);
|
|
|
|
MyClasspath* cp = static_cast<MyClasspath*>(t->m->classpath);
|
|
|
|
THREAD_RUNTIME_ARRAY(t, char, p, stringLength(t, path) + 1);
|
|
stringChars(t, path, RUNTIME_ARRAY_BODY(p));
|
|
replace('\\', '/', RUNTIME_ARRAY_BODY(p));
|
|
|
|
EmbeddedFile ef(cp, RUNTIME_ARRAY_BODY(p), stringLength(t, path));
|
|
if (ef.jar) {
|
|
if (ef.jarLength == 0 or ef.pathLength == 0) {
|
|
throwNew(t, Machine::FileNotFoundExceptionType);
|
|
}
|
|
|
|
Finder* finder = getFinder(t, ef.jar, ef.jarLength);
|
|
if (finder == 0) {
|
|
throwNew(t, Machine::FileNotFoundExceptionType);
|
|
}
|
|
|
|
System::Region* r = finder->find(ef.path);
|
|
if (r == 0) {
|
|
throwNew(t, Machine::FileNotFoundExceptionType);
|
|
}
|
|
|
|
const uint8_t* start = r->start();
|
|
const uint8_t* end = start + r->length();
|
|
unsigned entryCount = 0;
|
|
for (const uint8_t* p = end - CentralDirectorySearchStart; p > start;) {
|
|
if (get4(p) == CentralDirectorySignature) {
|
|
p = start + centralDirectoryOffset(p);
|
|
|
|
while (p < end) {
|
|
if (get4(p) == EntrySignature) {
|
|
++ entryCount;
|
|
|
|
p = endOfEntry(p);
|
|
} else {
|
|
goto make;
|
|
}
|
|
}
|
|
} else {
|
|
-- p;
|
|
}
|
|
}
|
|
|
|
make:
|
|
ZipFile* file = new
|
|
(t->m->heap->allocate
|
|
(sizeof(ZipFile) + (sizeof(ZipFile::Entry) * entryCount)))
|
|
ZipFile(t, r, entryCount);
|
|
|
|
{ unsigned position = 0;
|
|
for (const uint8_t* p = end - CentralDirectorySearchStart; p > start;) {
|
|
if (get4(p) == CentralDirectorySignature) {
|
|
p = start + centralDirectoryOffset(p);
|
|
|
|
while (p < end) {
|
|
if (get4(p) == EntrySignature) {
|
|
unsigned h = hash(fileName(p), fileNameLength(p));
|
|
unsigned i = h & (file->indexSize - 1);
|
|
|
|
file->index[i] = new (file->entries + (position++))
|
|
ZipFile::Entry(h, p, file->index[i]);
|
|
|
|
p = endOfEntry(p);
|
|
} else {
|
|
goto exit;
|
|
}
|
|
}
|
|
} else {
|
|
-- p;
|
|
}
|
|
}
|
|
}
|
|
|
|
exit:
|
|
return reinterpret_cast<int64_t>(file);
|
|
} else {
|
|
return reinterpret_cast<int64_t>
|
|
(new (t->m->heap->allocate(sizeof(ZipFile))) ZipFile
|
|
(longValue
|
|
(t, t->m->processor->invoke
|
|
(t, nativeInterceptOriginal
|
|
(t, methodRuntimeDataNative(t, getMethodRuntimeData(t, method))),
|
|
0, path, mode, lastModified))));
|
|
}
|
|
}
|
|
|
|
int64_t JNICALL
|
|
getZipFileEntryCount(Thread* t, object method, uintptr_t* arguments)
|
|
{
|
|
int64_t peer; memcpy(&peer, arguments, 8);
|
|
|
|
ZipFile* file = reinterpret_cast<ZipFile*>(peer);
|
|
if (file->region) {
|
|
return file->entryCount;
|
|
} else {
|
|
return intValue
|
|
(t, t->m->processor->invoke
|
|
(t, nativeInterceptOriginal
|
|
(t, methodRuntimeDataNative(t, getMethodRuntimeData(t, method))),
|
|
0, file->file));
|
|
}
|
|
}
|
|
|
|
ZipFile::Entry*
|
|
find(ZipFile* file, const char* path, unsigned pathLength)
|
|
{
|
|
unsigned i = hash(path) & (file->indexSize - 1);
|
|
for (ZipFile::Entry* e = file->index[i]; e; e = e->next) {
|
|
const uint8_t* p = e->start;
|
|
if (equal(path, pathLength, fileName(p), fileNameLength(p))) {
|
|
return e;
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
int64_t JNICALL
|
|
getZipFileEntry(Thread* t, object method, uintptr_t* arguments)
|
|
{
|
|
int64_t peer; memcpy(&peer, arguments, 8);
|
|
object path = reinterpret_cast<object>(arguments[2]);
|
|
bool addSlash = arguments[3];
|
|
|
|
ZipFile* file = reinterpret_cast<ZipFile*>(peer);
|
|
if (file->region) {
|
|
THREAD_RUNTIME_ARRAY(t, char, p, byteArrayLength(t, path) + 2);
|
|
memcpy(RUNTIME_ARRAY_BODY(p), &byteArrayBody(t, path, 0),
|
|
byteArrayLength(t, path));
|
|
RUNTIME_ARRAY_BODY(p)[byteArrayLength(t, path)] = 0;
|
|
replace('\\', '/', RUNTIME_ARRAY_BODY(p));
|
|
if (addSlash) {
|
|
RUNTIME_ARRAY_BODY(p)[byteArrayLength(t, path)] = '/';
|
|
RUNTIME_ARRAY_BODY(p)[byteArrayLength(t, path) + 1] = 0;
|
|
}
|
|
|
|
return reinterpret_cast<int64_t>
|
|
(find(file, RUNTIME_ARRAY_BODY(p), byteArrayLength(t, path)));
|
|
} else {
|
|
int64_t entry = longValue
|
|
(t, t->m->processor->invoke
|
|
(t, nativeInterceptOriginal
|
|
(t, methodRuntimeDataNative(t, getMethodRuntimeData(t, method))),
|
|
0, file->file, path, addSlash));
|
|
|
|
return entry ? reinterpret_cast<int64_t>
|
|
(new (t->m->heap->allocate(sizeof(ZipFile::Entry)))
|
|
ZipFile::Entry(entry)) : 0;
|
|
}
|
|
}
|
|
|
|
int64_t JNICALL
|
|
getZipFileEntryBytes(Thread* t, object method, uintptr_t* arguments)
|
|
{
|
|
int64_t peer; memcpy(&peer, arguments, 8);
|
|
int type = arguments[2];
|
|
|
|
ZipFile::Entry* entry = reinterpret_cast<ZipFile::Entry*>(peer);
|
|
if (entry->start) {
|
|
switch (type) {
|
|
case 0: { // name
|
|
unsigned nameLength = fileNameLength(entry->start);
|
|
object array = makeByteArray(t, nameLength + 1);
|
|
memcpy(&byteArrayBody(t, array, 0), fileName(entry->start), nameLength);
|
|
byteArrayBody(t, array, nameLength) = 0;
|
|
return reinterpret_cast<int64_t>(array);
|
|
} break;
|
|
|
|
case 1: { // extra
|
|
return 0;
|
|
} break;
|
|
|
|
case 2: { // comment
|
|
return 0;
|
|
} break;
|
|
|
|
default: abort(t);
|
|
}
|
|
return compressedSize(entry->start);
|
|
} else {
|
|
return reinterpret_cast<int64_t>
|
|
(t->m->processor->invoke
|
|
(t, nativeInterceptOriginal
|
|
(t, methodRuntimeDataNative(t, getMethodRuntimeData(t, method))),
|
|
0, entry->entry, type));
|
|
}
|
|
}
|
|
|
|
int64_t JNICALL
|
|
getNextZipFileEntry(Thread* t, object method, uintptr_t* arguments)
|
|
{
|
|
int64_t peer; memcpy(&peer, arguments, 8);
|
|
int index = arguments[2];
|
|
|
|
ZipFile* file = reinterpret_cast<ZipFile*>(peer);
|
|
if (file->region) {
|
|
return reinterpret_cast<int64_t>(file->entries + index);
|
|
} else {
|
|
int64_t entry = longValue
|
|
(t, t->m->processor->invoke
|
|
(t, nativeInterceptOriginal
|
|
(t, methodRuntimeDataNative(t, getMethodRuntimeData(t, method))),
|
|
0, file->file, index));
|
|
|
|
return entry ? reinterpret_cast<int64_t>
|
|
(new (t->m->heap->allocate(sizeof(ZipFile::Entry)))
|
|
ZipFile::Entry(entry)) : 0;
|
|
}
|
|
}
|
|
|
|
int64_t JNICALL
|
|
getZipFileEntryMethod(Thread* t, object method, uintptr_t* arguments)
|
|
{
|
|
int64_t peer; memcpy(&peer, arguments, 8);
|
|
|
|
ZipFile::Entry* entry = reinterpret_cast<ZipFile::Entry*>(peer);
|
|
if (entry->start) {
|
|
return compressionMethod(entry->start);
|
|
} else {
|
|
return intValue
|
|
(t, t->m->processor->invoke
|
|
(t, nativeInterceptOriginal
|
|
(t, methodRuntimeDataNative(t, getMethodRuntimeData(t, method))),
|
|
0, entry->entry));
|
|
}
|
|
}
|
|
|
|
int64_t JNICALL
|
|
getZipFileEntryCompressedSize(Thread* t, object method, uintptr_t* arguments)
|
|
{
|
|
int64_t peer; memcpy(&peer, arguments, 8);
|
|
|
|
ZipFile::Entry* entry = reinterpret_cast<ZipFile::Entry*>(peer);
|
|
if (entry->start) {
|
|
return compressedSize(entry->start);
|
|
} else {
|
|
return longValue
|
|
(t, t->m->processor->invoke
|
|
(t, nativeInterceptOriginal
|
|
(t, methodRuntimeDataNative(t, getMethodRuntimeData(t, method))),
|
|
0, entry->entry));
|
|
}
|
|
}
|
|
|
|
int64_t JNICALL
|
|
getZipFileEntryUncompressedSize(Thread* t, object method, uintptr_t* arguments)
|
|
{
|
|
int64_t peer; memcpy(&peer, arguments, 8);
|
|
|
|
ZipFile::Entry* entry = reinterpret_cast<ZipFile::Entry*>(peer);
|
|
if (entry->start) {
|
|
return uncompressedSize(entry->start);
|
|
} else {
|
|
return longValue
|
|
(t, t->m->processor->invoke
|
|
(t, nativeInterceptOriginal
|
|
(t, methodRuntimeDataNative(t, getMethodRuntimeData(t, method))),
|
|
0, entry->entry));
|
|
}
|
|
}
|
|
|
|
void JNICALL
|
|
freeZipFileEntry(Thread* t, object method, uintptr_t* arguments)
|
|
{
|
|
int64_t filePeer; memcpy(&filePeer, arguments, 8);
|
|
int64_t entryPeer; memcpy(&entryPeer, arguments + 2, 8);
|
|
|
|
ZipFile* file = reinterpret_cast<ZipFile*>(filePeer);
|
|
ZipFile::Entry* entry = reinterpret_cast<ZipFile::Entry*>(entryPeer);
|
|
if (file->region == 0) {
|
|
t->m->processor->invoke
|
|
(t, nativeInterceptOriginal
|
|
(t, methodRuntimeDataNative(t, getMethodRuntimeData(t, method))),
|
|
0, file->file, entry->entry);
|
|
}
|
|
|
|
t->m->heap->free(entry, sizeof(ZipFile::Entry));
|
|
}
|
|
|
|
int64_t JNICALL
|
|
readZipFileEntry(Thread* t, object method, uintptr_t* arguments)
|
|
{
|
|
int64_t filePeer; memcpy(&filePeer, arguments, 8);
|
|
int64_t entryPeer; memcpy(&entryPeer, arguments + 2, 8);
|
|
int64_t position; memcpy(&position, arguments + 4, 8);
|
|
object buffer = reinterpret_cast<object>(arguments[6]);
|
|
int offset = arguments[7];
|
|
int length = arguments[8];
|
|
|
|
ZipFile* file = reinterpret_cast<ZipFile*>(filePeer);
|
|
ZipFile::Entry* entry = reinterpret_cast<ZipFile::Entry*>(entryPeer);
|
|
if (file->region) {
|
|
unsigned size = uncompressedSize(entry->start);
|
|
if (position >= size) {
|
|
return -1;
|
|
}
|
|
|
|
if (position + length > size) {
|
|
length = size - position;
|
|
}
|
|
|
|
memcpy(&byteArrayBody(t, buffer, offset),
|
|
fileData(file->region->start() + localHeaderOffset(entry->start))
|
|
+ position,
|
|
length);
|
|
|
|
return length;
|
|
} else {
|
|
return intValue
|
|
(t, t->m->processor->invoke
|
|
(t, nativeInterceptOriginal
|
|
(t, methodRuntimeDataNative(t, getMethodRuntimeData(t, method))),
|
|
0, file->file, entry->entry, position, buffer, offset, length));
|
|
}
|
|
}
|
|
|
|
int64_t JNICALL
|
|
getZipMessage(Thread* t, object method, uintptr_t* arguments)
|
|
{
|
|
int64_t peer; memcpy(&peer, arguments, 8);
|
|
|
|
ZipFile* file = reinterpret_cast<ZipFile*>(peer);
|
|
if (file->region) {
|
|
return 0;
|
|
} else {
|
|
return reinterpret_cast<int64_t>
|
|
(t->m->processor->invoke
|
|
(t, nativeInterceptOriginal
|
|
(t, methodRuntimeDataNative(t, getMethodRuntimeData(t, method))),
|
|
0, file->file));
|
|
}
|
|
}
|
|
|
|
int64_t JNICALL
|
|
getJarFileMetaInfEntryNames(Thread* t, object method, uintptr_t* arguments)
|
|
{
|
|
object this_ = reinterpret_cast<object>(arguments[0]);
|
|
|
|
MyClasspath* cp = static_cast<MyClasspath*>(t->m->classpath);
|
|
|
|
int64_t peer = fieldAtOffset<int64_t>(this_, cp->zipFileJzfileField);
|
|
ZipFile* file = reinterpret_cast<ZipFile*>(peer);
|
|
if (file->region) {
|
|
return 0;
|
|
} else {
|
|
PROTECT(t, method);
|
|
|
|
// OpenJDK's Java_java_util_jar_JarFile_getMetaInfEntryNames
|
|
// implementation expects to find a pointer to an instance of its
|
|
// jzfile structure in the ZipFile.jzfile field of the object we
|
|
// pass in. However, we can't pass this_ in, because its
|
|
// ZipFile.jzfile field points to a ZipFile instance, not a
|
|
// jzfile. So we pass in a temporary object instead which has the
|
|
// desired pointer at the same offset. We assume here that
|
|
// ZipFile.jzfile is the first field in that class and that
|
|
// Java_java_util_jar_JarFile_getMetaInfEntryNames will not look
|
|
// for any other fields in the object.
|
|
object pseudoThis = makeLong(t, file->file);
|
|
|
|
return reinterpret_cast<int64_t>
|
|
(t->m->processor->invoke
|
|
(t, nativeInterceptOriginal
|
|
(t, methodRuntimeDataNative(t, getMethodRuntimeData(t, method))),
|
|
pseudoThis));
|
|
}
|
|
}
|
|
|
|
void JNICALL
|
|
closeZipFile(Thread* t, object method, uintptr_t* arguments)
|
|
{
|
|
int64_t peer; memcpy(&peer, arguments, 8);
|
|
|
|
ZipFile* file = reinterpret_cast<ZipFile*>(peer);
|
|
if (file->region) {
|
|
file->region->dispose();
|
|
t->m->heap->free(file, sizeof(ZipFile)
|
|
+ (sizeof(ZipFile::Entry) * file->entryCount));
|
|
} else {
|
|
t->m->processor->invoke
|
|
(t, nativeInterceptOriginal
|
|
(t, methodRuntimeDataNative(t, getMethodRuntimeData(t, method))),
|
|
0, file->file);
|
|
|
|
t->m->heap->free(file, sizeof(ZipFile));
|
|
}
|
|
}
|
|
|
|
int64_t JNICALL
|
|
getBootstrapResource(Thread* t, object, uintptr_t* arguments)
|
|
{
|
|
object name = reinterpret_cast<object>(arguments[0]);
|
|
PROTECT(t, name);
|
|
|
|
object m = findMethodOrNull
|
|
(t, type(t, Machine::SystemClassLoaderType),
|
|
"findResource", "(Ljava/lang/String;)Ljava/net/URL;");
|
|
|
|
if (m) {
|
|
return reinterpret_cast<int64_t>
|
|
(t->m->processor->invoke(t, m, root(t, Machine::BootLoader), name));
|
|
} else {
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
int64_t JNICALL
|
|
getBootstrapResources(Thread* t, object, uintptr_t* arguments)
|
|
{
|
|
object name = reinterpret_cast<object>(arguments[0]);
|
|
PROTECT(t, name);
|
|
|
|
object m = findMethodOrNull
|
|
(t, type(t, Machine::SystemClassLoaderType),
|
|
"findResources", "(Ljava/lang/String;)Ljava/util/Enumeration;");
|
|
|
|
if (m) {
|
|
return reinterpret_cast<int64_t>
|
|
(t->m->processor->invoke(t, m, root(t, Machine::BootLoader), name));
|
|
} else {
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jint JNICALL
|
|
net_JNI_OnLoad(JavaVM*, void*);
|
|
|
|
extern "C" AVIAN_EXPORT jint JNICALL
|
|
management_JNI_OnLoad(JavaVM*, void*);
|
|
|
|
void JNICALL
|
|
loadLibrary(Thread* t, object, uintptr_t* arguments)
|
|
{
|
|
object name = reinterpret_cast<object>(arguments[1]);
|
|
THREAD_RUNTIME_ARRAY(t, char, n, stringLength(t, name) + 1);
|
|
stringChars(t, name, RUNTIME_ARRAY_BODY(n));
|
|
|
|
bool absolute = arguments[2];
|
|
|
|
if (not absolute) {
|
|
if (strcmp(RUNTIME_ARRAY_BODY(n), "net") == 0) {
|
|
bool ran;
|
|
|
|
{ ACQUIRE(t, t->m->classLock);
|
|
|
|
local::MyClasspath* c = static_cast<local::MyClasspath*>
|
|
(t->m->classpath);
|
|
|
|
ran = c->ranNetOnLoad;
|
|
c->ranNetOnLoad = true;
|
|
}
|
|
|
|
if (not ran) {
|
|
net_JNI_OnLoad(t->m, 0);
|
|
}
|
|
|
|
return;
|
|
} else if (strcmp(RUNTIME_ARRAY_BODY(n), "management") == 0) {
|
|
bool ran;
|
|
|
|
{ ACQUIRE(t, t->m->classLock);
|
|
|
|
local::MyClasspath* c = static_cast<local::MyClasspath*>
|
|
(t->m->classpath);
|
|
|
|
ran = c->ranManagementOnLoad;
|
|
c->ranManagementOnLoad = true;
|
|
}
|
|
|
|
if (not ran) {
|
|
management_JNI_OnLoad(t->m, 0);
|
|
}
|
|
|
|
return;
|
|
} else if (strcmp(RUNTIME_ARRAY_BODY(n), "zip") == 0
|
|
or strcmp(RUNTIME_ARRAY_BODY(n), "nio") == 0)
|
|
{
|
|
return;
|
|
}
|
|
}
|
|
|
|
loadLibrary
|
|
(t, static_cast<local::MyClasspath*>(t->m->classpath)->libraryPath,
|
|
RUNTIME_ARRAY_BODY(n), not absolute, true);
|
|
}
|
|
|
|
void
|
|
interceptFileOperations(Thread* t, bool updateRuntimeData)
|
|
{
|
|
MyClasspath* cp = static_cast<MyClasspath*>(t->m->classpath);
|
|
|
|
{ object fileClass = resolveClass
|
|
(t, root(t, Machine::BootLoader), "java/io/File", false);
|
|
|
|
if (fileClass) {
|
|
object filePathField = findFieldInClass2
|
|
(t, fileClass, "path", "Ljava/lang/String;");
|
|
|
|
if (filePathField) {
|
|
cp->filePathField = fieldOffset(t, filePathField);
|
|
}
|
|
}
|
|
}
|
|
|
|
{ object fileDescriptorClass = resolveClass
|
|
(t, root(t, Machine::BootLoader), "java/io/FileDescriptor", false);
|
|
|
|
if (fileDescriptorClass) {
|
|
object fileDescriptorFdField = findFieldInClass2
|
|
(t, fileDescriptorClass, "fd", "I");
|
|
|
|
if (fileDescriptorFdField) {
|
|
cp->fileDescriptorFdField = fieldOffset(t, fileDescriptorFdField);
|
|
}
|
|
}
|
|
}
|
|
|
|
{ object fileInputStreamClass = resolveClass
|
|
(t, root(t, Machine::BootLoader), "java/io/FileInputStream", false);
|
|
|
|
if (fileInputStreamClass) {
|
|
PROTECT(t, fileInputStreamClass);
|
|
|
|
object fileInputStreamFdField = findFieldInClass2
|
|
(t, fileInputStreamClass, "fd", "Ljava/io/FileDescriptor;");
|
|
|
|
if (fileInputStreamFdField) {
|
|
cp->fileInputStreamFdField = fieldOffset(t, fileInputStreamFdField);
|
|
|
|
intercept(t, fileInputStreamClass, "open", "(Ljava/lang/String;)V",
|
|
voidPointer(openFile), updateRuntimeData);
|
|
|
|
if(findMethodOrNull(t, fileInputStreamClass, "read0", "()I") != 0) {
|
|
intercept(t, fileInputStreamClass, "read0", "()I",
|
|
voidPointer(readByteFromFile), updateRuntimeData);
|
|
} else {
|
|
intercept(t, fileInputStreamClass, "read", "()I",
|
|
voidPointer(readByteFromFile), updateRuntimeData);
|
|
}
|
|
|
|
intercept(t, fileInputStreamClass, "readBytes", "([BII)I",
|
|
voidPointer(readBytesFromFile), updateRuntimeData);
|
|
|
|
intercept(t, fileInputStreamClass, "skip", "(J)J",
|
|
voidPointer(skipBytesInFile), updateRuntimeData);
|
|
|
|
intercept(t, fileInputStreamClass, "available", "()I",
|
|
voidPointer(availableBytesInFile), updateRuntimeData);
|
|
|
|
intercept(t, fileInputStreamClass, "close0", "()V",
|
|
voidPointer(closeFile), updateRuntimeData);
|
|
}
|
|
}
|
|
}
|
|
|
|
{ object zipFileClass = resolveClass
|
|
(t, root(t, Machine::BootLoader), "java/util/zip/ZipFile", false);
|
|
|
|
if (zipFileClass) {
|
|
PROTECT(t, zipFileClass);
|
|
|
|
object zipFileJzfileField = findFieldInClass2
|
|
(t, zipFileClass, "jzfile", "J");
|
|
|
|
if (zipFileJzfileField) {
|
|
cp->zipFileJzfileField = fieldOffset(t, zipFileJzfileField);
|
|
|
|
intercept(t, zipFileClass, "open", "(Ljava/lang/String;IJZ)J",
|
|
voidPointer(openZipFile), updateRuntimeData);
|
|
|
|
intercept(t, zipFileClass, "getTotal", "(J)I",
|
|
voidPointer(getZipFileEntryCount), updateRuntimeData);
|
|
|
|
intercept(t, zipFileClass, "getEntry", "(J[BZ)J",
|
|
voidPointer(getZipFileEntry), updateRuntimeData);
|
|
|
|
intercept(t, zipFileClass, "getEntryBytes", "(JI)[B",
|
|
voidPointer(getZipFileEntryBytes), updateRuntimeData);
|
|
|
|
intercept(t, zipFileClass, "getNextEntry", "(JI)J",
|
|
voidPointer(getNextZipFileEntry), updateRuntimeData);
|
|
|
|
intercept(t, zipFileClass, "getEntryMethod", "(J)I",
|
|
voidPointer(getZipFileEntryMethod), updateRuntimeData);
|
|
|
|
intercept(t, zipFileClass, "freeEntry", "(JJ)V",
|
|
voidPointer(freeZipFileEntry), updateRuntimeData);
|
|
|
|
intercept(t, zipFileClass, "read", "(JJJ[BII)I",
|
|
voidPointer(readZipFileEntry), updateRuntimeData);
|
|
|
|
intercept(t, zipFileClass, "getEntryCSize", "(J)J",
|
|
voidPointer(getZipFileEntryCompressedSize),
|
|
updateRuntimeData);
|
|
|
|
intercept(t, zipFileClass, "getEntrySize", "(J)J",
|
|
voidPointer(getZipFileEntryUncompressedSize),
|
|
updateRuntimeData);
|
|
|
|
intercept(t, zipFileClass, "getZipMessage", "(J)Ljava/lang/String;",
|
|
voidPointer(getZipMessage), updateRuntimeData);
|
|
|
|
intercept(t, zipFileClass, "close", "(J)V",
|
|
voidPointer(closeZipFile), updateRuntimeData);
|
|
}
|
|
}
|
|
}
|
|
|
|
{ object jarFileClass = resolveClass
|
|
(t, root(t, Machine::BootLoader), "java/util/jar/JarFile", false);
|
|
|
|
if (jarFileClass) {
|
|
intercept(t, jarFileClass, "getMetaInfEntryNames",
|
|
"()[Ljava/lang/String;",
|
|
voidPointer(getJarFileMetaInfEntryNames), updateRuntimeData);
|
|
}
|
|
}
|
|
|
|
{
|
|
#ifdef PLATFORM_WINDOWS
|
|
const char* const fsClassName = "java/io/WinNTFileSystem";
|
|
const char* const gbaMethodName = "getBooleanAttributes";
|
|
#else
|
|
const char* const fsClassName = "java/io/UnixFileSystem";
|
|
const char* const gbaMethodName = "getBooleanAttributes0";
|
|
#endif
|
|
|
|
object fsClass = resolveClass
|
|
(t, root(t, Machine::BootLoader), fsClassName, false);
|
|
|
|
if (fsClass) {
|
|
PROTECT(t, fsClass);
|
|
|
|
intercept(t, fsClass, gbaMethodName, "(Ljava/io/File;)I",
|
|
voidPointer(getFileAttributes), updateRuntimeData);
|
|
|
|
intercept(t, fsClass, "checkAccess", "(Ljava/io/File;I)Z",
|
|
voidPointer(checkFileAccess), updateRuntimeData);
|
|
|
|
intercept(t, fsClass, "getLength", "(Ljava/io/File;)J",
|
|
voidPointer(getFileLength), updateRuntimeData);
|
|
}
|
|
}
|
|
|
|
intercept(t, type(t, Machine::ClassLoaderType), "loadLibrary",
|
|
"(Ljava/lang/Class;Ljava/lang/String;Z)V",
|
|
voidPointer(loadLibrary), updateRuntimeData);
|
|
|
|
intercept(t, type(t, Machine::ClassLoaderType), "getBootstrapResource",
|
|
"(Ljava/lang/String;)Ljava/net/URL;",
|
|
voidPointer(getBootstrapResource), updateRuntimeData);
|
|
|
|
intercept(t, type(t, Machine::ClassLoaderType), "getBootstrapResources",
|
|
"(Ljava/lang/String;)Ljava/util/Enumeration;",
|
|
voidPointer(getBootstrapResources), updateRuntimeData);
|
|
}
|
|
#endif // AVIAN_OPENJDK_SRC
|
|
|
|
unsigned
|
|
classDeclaredMethodCount(Thread* t, object c)
|
|
{
|
|
object addendum = classAddendum(t, c);
|
|
if (addendum) {
|
|
int count = classAddendumDeclaredMethodCount(t, addendum);
|
|
if (count >= 0) {
|
|
return count;
|
|
}
|
|
}
|
|
object table = classMethodTable(t, c);
|
|
return table == 0 ? 0 : arrayLength(t, table);
|
|
}
|
|
|
|
unsigned
|
|
countMethods(Thread* t, object c, bool publicOnly)
|
|
{
|
|
object table = classMethodTable(t, c);
|
|
unsigned count = 0;
|
|
for (unsigned i = 0, j = classDeclaredMethodCount(t, c); i < j; ++i) {
|
|
object vmMethod = arrayBody(t, table, i);
|
|
if (((not publicOnly) or (methodFlags(t, vmMethod) & ACC_PUBLIC))
|
|
and byteArrayBody(t, methodName(t, vmMethod), 0) != '<')
|
|
{
|
|
++ count;
|
|
}
|
|
}
|
|
return count;
|
|
}
|
|
|
|
unsigned
|
|
countFields(Thread* t, object c, bool publicOnly)
|
|
{
|
|
object table = classFieldTable(t, c);
|
|
if (publicOnly) {
|
|
unsigned count = 0;
|
|
for (unsigned i = 0; i < arrayLength(t, table); ++i) {
|
|
object vmField = arrayBody(t, table, i);
|
|
if (fieldFlags(t, vmField) & ACC_PUBLIC) {
|
|
++ count;
|
|
}
|
|
}
|
|
return count;
|
|
} else {
|
|
return objectArrayLength(t, table);
|
|
}
|
|
}
|
|
|
|
unsigned
|
|
countConstructors(Thread* t, object c, bool publicOnly)
|
|
{
|
|
object table = classMethodTable(t, c);
|
|
unsigned count = 0;
|
|
for (unsigned i = 0, j = classDeclaredMethodCount(t, c); i < j; ++i) {
|
|
object vmMethod = arrayBody(t, table, i);
|
|
if (((not publicOnly) or (methodFlags(t, vmMethod) & ACC_PUBLIC))
|
|
and strcmp(reinterpret_cast<char*>
|
|
(&byteArrayBody(t, methodName(t, vmMethod), 0)),
|
|
"<init>") == 0)
|
|
{
|
|
++ count;
|
|
}
|
|
}
|
|
return count;
|
|
}
|
|
|
|
#ifdef HAVE_JexecutableHasRealParameterData
|
|
object
|
|
makeJmethod(Thread* t,
|
|
uint8_t override,
|
|
object securityCheckCache,
|
|
object clazz,
|
|
uint32_t slot,
|
|
object name,
|
|
object returnType,
|
|
object parameterTypes,
|
|
object exceptionTypes,
|
|
uint32_t modifiers,
|
|
object signature,
|
|
object genericInfo,
|
|
object annotations,
|
|
object parameterAnnotations,
|
|
object annotationDefault,
|
|
object methodAccessor,
|
|
object root,
|
|
object declaredAnnotations)
|
|
{
|
|
return makeJmethod
|
|
(t, override, securityCheckCache, 0, 0, declaredAnnotations, clazz, slot,
|
|
name, returnType, parameterTypes, exceptionTypes, modifiers, signature,
|
|
genericInfo, annotations, parameterAnnotations, annotationDefault,
|
|
methodAccessor, root);
|
|
}
|
|
|
|
object
|
|
makeJconstructor(Thread* t,
|
|
uint8_t override,
|
|
object securityCheckCache,
|
|
object clazz,
|
|
uint32_t slot,
|
|
object parameterTypes,
|
|
object exceptionTypes,
|
|
uint32_t modifiers,
|
|
object signature,
|
|
object genericInfo,
|
|
object annotations,
|
|
object parameterAnnotations,
|
|
object constructorAccessor,
|
|
object root,
|
|
object declaredAnnotations)
|
|
{
|
|
return makeJconstructor
|
|
(t, override, securityCheckCache, 0, 0, declaredAnnotations, clazz, slot,
|
|
parameterTypes, exceptionTypes, modifiers, signature, genericInfo,
|
|
annotations, parameterAnnotations, constructorAccessor, root);
|
|
}
|
|
#endif // HAVE_JexecutableHasRealParameterData
|
|
|
|
object
|
|
makeJmethod(Thread* t, object vmMethod, int index)
|
|
{
|
|
PROTECT(t, vmMethod);
|
|
|
|
object name = intern
|
|
(t, t->m->classpath->makeString
|
|
(t, methodName(t, vmMethod), 0, byteArrayLength
|
|
(t, methodName(t, vmMethod)) - 1));
|
|
PROTECT(t, name);
|
|
|
|
unsigned parameterCount;
|
|
unsigned returnTypeSpec;
|
|
object parameterTypes = resolveParameterJTypes
|
|
(t, classLoader(t, methodClass(t, vmMethod)), methodSpec(t, vmMethod),
|
|
¶meterCount, &returnTypeSpec);
|
|
PROTECT(t, parameterTypes);
|
|
|
|
object returnType = resolveJType
|
|
(t, classLoader(t, methodClass(t, vmMethod)), reinterpret_cast<char*>
|
|
(&byteArrayBody(t, methodSpec(t, vmMethod), returnTypeSpec)),
|
|
byteArrayLength(t, methodSpec(t, vmMethod)) - 1 - returnTypeSpec);
|
|
PROTECT(t, returnType);
|
|
|
|
object exceptionTypes = resolveExceptionJTypes
|
|
(t, classLoader(t, methodClass(t, vmMethod)),
|
|
methodAddendum(t, vmMethod));
|
|
PROTECT(t, exceptionTypes);
|
|
|
|
object signature;
|
|
object annotationTable;
|
|
object parameterAnnotationTable;
|
|
object annotationDefault;
|
|
object addendum = methodAddendum(t, vmMethod);
|
|
if (addendum) {
|
|
signature = addendumSignature(t, addendum);
|
|
if (signature) {
|
|
PROTECT(t, addendum);
|
|
|
|
signature = t->m->classpath->makeString
|
|
(t, signature, 0, byteArrayLength(t, signature) - 1);
|
|
}
|
|
|
|
annotationTable = addendumAnnotationTable(t, addendum);
|
|
|
|
parameterAnnotationTable = methodAddendumParameterAnnotationTable
|
|
(t, addendum);
|
|
|
|
annotationDefault = methodAddendumAnnotationDefault(t, addendum);
|
|
} else {
|
|
signature = 0;
|
|
annotationTable = 0;
|
|
parameterAnnotationTable = 0;
|
|
annotationDefault = 0;
|
|
}
|
|
|
|
PROTECT(t, signature);
|
|
PROTECT(t, annotationTable);
|
|
PROTECT(t, parameterAnnotationTable);
|
|
PROTECT(t, annotationDefault);
|
|
|
|
if (annotationTable or parameterAnnotationTable or annotationDefault) {
|
|
object runtimeData = getClassRuntimeData(t, methodClass(t, vmMethod));
|
|
|
|
set(t, runtimeData, ClassRuntimeDataPool,
|
|
addendumPool(t, methodAddendum(t, vmMethod)));
|
|
}
|
|
|
|
if (index == -1) {
|
|
object table = classMethodTable(t, methodClass(t, vmMethod));
|
|
for (unsigned i = 0; i < arrayLength(t, table); ++i) {
|
|
if (vmMethod == arrayBody(t, table, i)) {
|
|
index = i;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
expect(t, index != -1);
|
|
|
|
object jclass = getJClass(t, methodClass(t, vmMethod));
|
|
|
|
return makeJmethod
|
|
(t, true, 0, jclass, index, name, returnType, parameterTypes,
|
|
exceptionTypes, methodFlags(t, vmMethod), signature, 0, annotationTable,
|
|
parameterAnnotationTable, annotationDefault, 0, 0, 0);
|
|
}
|
|
|
|
object
|
|
makeJconstructor(Thread* t, object vmMethod, int index)
|
|
{
|
|
PROTECT(t, vmMethod);
|
|
|
|
unsigned parameterCount;
|
|
unsigned returnTypeSpec;
|
|
object parameterTypes = resolveParameterJTypes
|
|
(t, classLoader(t, methodClass(t, vmMethod)), methodSpec(t, vmMethod),
|
|
¶meterCount, &returnTypeSpec);
|
|
PROTECT(t, parameterTypes);
|
|
|
|
object exceptionTypes = resolveExceptionJTypes
|
|
(t, classLoader(t, methodClass(t, vmMethod)),
|
|
methodAddendum(t, vmMethod));
|
|
PROTECT(t, exceptionTypes);
|
|
|
|
object signature;
|
|
object annotationTable;
|
|
object parameterAnnotationTable;
|
|
object addendum = methodAddendum(t, vmMethod);
|
|
if (addendum) {
|
|
signature = addendumSignature(t, addendum);
|
|
if (signature) {
|
|
PROTECT(t, addendum);
|
|
|
|
signature = t->m->classpath->makeString
|
|
(t, signature, 0, byteArrayLength(t, signature) - 1);
|
|
}
|
|
|
|
annotationTable = addendumAnnotationTable(t, addendum);
|
|
parameterAnnotationTable = methodAddendumParameterAnnotationTable
|
|
(t, addendum);
|
|
} else {
|
|
signature = 0;
|
|
annotationTable = 0;
|
|
parameterAnnotationTable = 0;
|
|
}
|
|
|
|
PROTECT(t, signature);
|
|
PROTECT(t, annotationTable);
|
|
PROTECT(t, parameterAnnotationTable);
|
|
|
|
if (annotationTable or parameterAnnotationTable) {
|
|
object runtimeData = getClassRuntimeData(t, methodClass(t, vmMethod));
|
|
|
|
set(t, runtimeData, ClassRuntimeDataPool,
|
|
addendumPool(t, methodAddendum(t, vmMethod)));
|
|
}
|
|
|
|
if (index == -1) {
|
|
object table = classMethodTable(t, methodClass(t, vmMethod));
|
|
for (unsigned i = 0; i < arrayLength(t, table); ++i) {
|
|
if (vmMethod == arrayBody(t, table, i)) {
|
|
index = i;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
expect(t, index != -1);
|
|
|
|
object jclass = getJClass(t, methodClass(t, vmMethod));
|
|
|
|
return makeJconstructor
|
|
(t, true, 0, jclass, index, parameterTypes, exceptionTypes, methodFlags
|
|
(t, vmMethod), signature, 0, annotationTable, parameterAnnotationTable,
|
|
0, 0, 0);
|
|
}
|
|
|
|
object
|
|
makeJfield(Thread* t, object vmField, int index)
|
|
{
|
|
PROTECT(t, vmField);
|
|
|
|
object name = intern
|
|
(t, t->m->classpath->makeString
|
|
(t, fieldName(t, vmField), 0, byteArrayLength
|
|
(t, fieldName(t, vmField)) - 1));
|
|
PROTECT(t, name);
|
|
|
|
object type = resolveClassBySpec
|
|
(t, classLoader(t, fieldClass(t, vmField)),
|
|
reinterpret_cast<char*>
|
|
(&byteArrayBody(t, fieldSpec(t, vmField), 0)),
|
|
byteArrayLength(t, fieldSpec(t, vmField)) - 1);
|
|
PROTECT(t, type);
|
|
|
|
type = getJClass(t, type);
|
|
|
|
object signature;
|
|
object annotationTable;
|
|
object addendum = fieldAddendum(t, vmField);
|
|
if (addendum) {
|
|
signature = addendumSignature(t, addendum);
|
|
if (signature) {
|
|
PROTECT(t, addendum);
|
|
|
|
signature = t->m->classpath->makeString
|
|
(t, signature, 0, byteArrayLength(t, signature) - 1);
|
|
}
|
|
|
|
annotationTable = addendumAnnotationTable(t, addendum);
|
|
} else {
|
|
signature = 0;
|
|
annotationTable = 0;
|
|
}
|
|
|
|
PROTECT(t, signature);
|
|
PROTECT(t, annotationTable);
|
|
|
|
if (annotationTable) {
|
|
object runtimeData = getClassRuntimeData(t, fieldClass(t, vmField));
|
|
|
|
set(t, runtimeData, ClassRuntimeDataPool,
|
|
addendumPool(t, fieldAddendum(t, vmField)));
|
|
}
|
|
|
|
if (index == -1) {
|
|
object table = classFieldTable(t, fieldClass(t, vmField));
|
|
for (unsigned i = 0; i < arrayLength(t, table); ++i) {
|
|
if (vmField == arrayBody(t, table, i)) {
|
|
index = i;
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
|
|
expect(t, index != -1);
|
|
|
|
object jclass = getJClass(t, fieldClass(t, vmField));
|
|
|
|
return makeJfield
|
|
(t, true, 0, jclass, index, name, type, fieldFlags
|
|
(t, vmField), signature, 0, annotationTable, 0, 0, 0, 0);
|
|
}
|
|
|
|
void
|
|
setProperty(Thread* t, object method, object properties,
|
|
const char* name, const void* value, const char* format = "%s")
|
|
{
|
|
PROTECT(t, method);
|
|
PROTECT(t, properties);
|
|
|
|
object n = makeString(t, "%s", name);
|
|
PROTECT(t, n);
|
|
|
|
object v = makeString(t, format, value);
|
|
|
|
t->m->processor->invoke(t, method, properties, n, v);
|
|
}
|
|
|
|
bool
|
|
pipeAvailable(int fd, int* available)
|
|
{
|
|
#ifdef PLATFORM_WINDOWS
|
|
HANDLE h = reinterpret_cast<HANDLE>(_get_osfhandle(fd));
|
|
if (h == INVALID_HANDLE_VALUE) {
|
|
return false;
|
|
}
|
|
|
|
DWORD n;
|
|
if (PeekNamedPipe(h, 0,0, 0, &n, 0)) {
|
|
*available = n;
|
|
} else {
|
|
if (GetLastError() != ERROR_BROKEN_PIPE) {
|
|
return false;
|
|
}
|
|
*available = 0;
|
|
}
|
|
|
|
return true;
|
|
#else
|
|
return ioctl(fd, FIONREAD, available) >= 0;
|
|
#endif
|
|
}
|
|
|
|
} // namespace local
|
|
|
|
} // namespace
|
|
|
|
namespace vm {
|
|
|
|
Classpath*
|
|
makeClasspath(System* s, Allocator* allocator, const char* javaHome,
|
|
const char* embedPrefix)
|
|
{
|
|
return new (allocator->allocate(sizeof(local::MyClasspath)))
|
|
local::MyClasspath(s, allocator, javaHome, embedPrefix);
|
|
}
|
|
|
|
} // namespace vm
|
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
|
Avian_java_lang_Class_getSuperclass
|
|
(Thread* t, object, uintptr_t* arguments)
|
|
{
|
|
object class_ = jclassVmClass(t, reinterpret_cast<object>(arguments[0]));
|
|
if (classFlags(t, class_) & ACC_INTERFACE) {
|
|
return 0;
|
|
} else {
|
|
object super = classSuper(t, class_);
|
|
return super ? reinterpret_cast<int64_t>(getJClass(t, super)) : 0;
|
|
}
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT void
|
|
Avian_sun_misc_Unsafe_registerNatives
|
|
(Thread*, object, uintptr_t*)
|
|
{
|
|
// ignore
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT void
|
|
Avian_sun_misc_Perf_registerNatives
|
|
(Thread*, object, uintptr_t*)
|
|
{
|
|
// ignore
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT int64_t
|
|
Avian_sun_misc_Perf_createLong
|
|
(Thread* t, object, uintptr_t*)
|
|
{
|
|
return reinterpret_cast<int64_t>
|
|
(t->m->processor->invoke
|
|
(t, resolveMethod
|
|
(t, root(t, Machine::BootLoader), "java/nio/ByteBuffer", "allocate",
|
|
"(I)Ljava/nio/ByteBuffer;"), 0, 8));
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT int64_t
|
|
Avian_sun_misc_Unsafe_addressSize
|
|
(Thread*, object, uintptr_t*)
|
|
{
|
|
return BytesPerWord;
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT int64_t
|
|
Avian_sun_misc_Unsafe_defineClass__Ljava_lang_String_2_3BIILjava_lang_ClassLoader_2Ljava_security_ProtectionDomain_2
|
|
(Thread* t, object, uintptr_t* arguments)
|
|
{
|
|
//object name = reinterpret_cast<object>(arguments[1]);
|
|
object data = reinterpret_cast<object>(arguments[2]);
|
|
int32_t offset = arguments[3];
|
|
int32_t length = arguments[4];
|
|
object loader = reinterpret_cast<object>(arguments[5]);
|
|
//object domain = reinterpret_cast<object>(arguments[6]);
|
|
|
|
uint8_t* buffer = static_cast<uint8_t*>(t->m->heap->allocate(length));
|
|
|
|
THREAD_RESOURCE2(t, uint8_t*, buffer, int, length,
|
|
t->m->heap->free(buffer, length));
|
|
|
|
memcpy(buffer, &byteArrayBody(t, data, offset), length);
|
|
|
|
return reinterpret_cast<int64_t>
|
|
(getJClass(t, defineClass(t, loader, buffer, length)));
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT int64_t
|
|
Avian_sun_misc_Unsafe_allocateInstance
|
|
(Thread* t, object, uintptr_t* arguments)
|
|
{
|
|
object c = jclassVmClass(t, reinterpret_cast<object>(arguments[1]));
|
|
PROTECT(t, c);
|
|
|
|
initClass(t, c);
|
|
|
|
return reinterpret_cast<int64_t>(make(t, c));
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
|
Avian_sun_misc_Unsafe_staticFieldOffset
|
|
(Thread* t, object, uintptr_t* arguments)
|
|
{
|
|
object jfield = reinterpret_cast<object>(arguments[1]);
|
|
return fieldOffset
|
|
(t, arrayBody
|
|
(t, classFieldTable
|
|
(t, jclassVmClass(t, jfieldClazz(t, jfield))), jfieldSlot(t, jfield)));
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
|
Avian_sun_misc_Unsafe_staticFieldBase
|
|
(Thread* t, object, uintptr_t* arguments)
|
|
{
|
|
return reinterpret_cast<int64_t>
|
|
(classStaticTable
|
|
(t, jclassVmClass
|
|
(t, jfieldClazz(t, reinterpret_cast<object>(arguments[1])))));
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
|
Avian_sun_misc_Unsafe_objectFieldOffset
|
|
(Thread* t, object, uintptr_t* arguments)
|
|
{
|
|
object jfield = reinterpret_cast<object>(arguments[1]);
|
|
return fieldOffset
|
|
(t, arrayBody
|
|
(t, classFieldTable
|
|
(t, jclassVmClass(t, jfieldClazz(t, jfield))), jfieldSlot(t, jfield)));
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
|
Avian_sun_misc_Unsafe_getShort__Ljava_lang_Object_2J
|
|
(Thread*, object, uintptr_t* arguments)
|
|
{
|
|
object o = reinterpret_cast<object>(arguments[1]);
|
|
int64_t offset; memcpy(&offset, arguments + 2, 8);
|
|
|
|
return fieldAtOffset<int16_t>(o, offset);
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
|
Avian_sun_misc_Unsafe_getChar__Ljava_lang_Object_2J
|
|
(Thread*, object, uintptr_t* arguments)
|
|
{
|
|
object o = reinterpret_cast<object>(arguments[1]);
|
|
int64_t offset; memcpy(&offset, arguments + 2, 8);
|
|
|
|
return fieldAtOffset<uint16_t>(o, offset);
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
|
Avian_sun_misc_Unsafe_getInt__Ljava_lang_Object_2J
|
|
(Thread*, object, uintptr_t* arguments)
|
|
{
|
|
object o = reinterpret_cast<object>(arguments[1]);
|
|
int64_t offset; memcpy(&offset, arguments + 2, 8);
|
|
|
|
return fieldAtOffset<int32_t>(o, offset);
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
|
Avian_sun_misc_Unsafe_getFloat__Ljava_lang_Object_2J
|
|
(Thread*, object, uintptr_t* arguments)
|
|
{
|
|
object o = reinterpret_cast<object>(arguments[1]);
|
|
int64_t offset; memcpy(&offset, arguments + 2, 8);
|
|
|
|
return fieldAtOffset<int32_t>(o, offset);
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
|
Avian_sun_misc_Unsafe_getLong__Ljava_lang_Object_2J
|
|
(Thread*, object, uintptr_t* arguments)
|
|
{
|
|
object o = reinterpret_cast<object>(arguments[1]);
|
|
int64_t offset; memcpy(&offset, arguments + 2, 8);
|
|
|
|
return fieldAtOffset<int64_t>(o, offset);
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
|
Avian_sun_misc_Unsafe_getDouble__Ljava_lang_Object_2J
|
|
(Thread* t, object method, uintptr_t* arguments)
|
|
{
|
|
return Avian_sun_misc_Unsafe_getLong__Ljava_lang_Object_2J
|
|
(t, method, arguments);
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
|
Avian_sun_misc_Unsafe_putByte__Ljava_lang_Object_2JB
|
|
(Thread*, object, uintptr_t* arguments)
|
|
{
|
|
object o = reinterpret_cast<object>(arguments[1]);
|
|
int64_t offset; memcpy(&offset, arguments + 2, 8);
|
|
int8_t value = arguments[4];
|
|
|
|
fieldAtOffset<int8_t>(o, offset) = value;
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
|
Avian_sun_misc_Unsafe_putShort__Ljava_lang_Object_2JS
|
|
(Thread*, object, uintptr_t* arguments)
|
|
{
|
|
object o = reinterpret_cast<object>(arguments[1]);
|
|
int64_t offset; memcpy(&offset, arguments + 2, 8);
|
|
int16_t value = arguments[4];
|
|
|
|
fieldAtOffset<int16_t>(o, offset) = value;
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
|
Avian_sun_misc_Unsafe_putChar__Ljava_lang_Object_2JC
|
|
(Thread*, object, uintptr_t* arguments)
|
|
{
|
|
object o = reinterpret_cast<object>(arguments[1]);
|
|
int64_t offset; memcpy(&offset, arguments + 2, 8);
|
|
uint16_t value = arguments[4];
|
|
|
|
fieldAtOffset<uint16_t>(o, offset) = value;
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
|
Avian_sun_misc_Unsafe_putInt__Ljava_lang_Object_2JI
|
|
(Thread*, object, uintptr_t* arguments)
|
|
{
|
|
object o = reinterpret_cast<object>(arguments[1]);
|
|
int64_t offset; memcpy(&offset, arguments + 2, 8);
|
|
int32_t value = arguments[4];
|
|
|
|
fieldAtOffset<int32_t>(o, offset) = value;
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
|
Avian_sun_misc_Unsafe_putFloat__Ljava_lang_Object_2JF
|
|
(Thread*, object, uintptr_t* arguments)
|
|
{
|
|
object o = reinterpret_cast<object>(arguments[1]);
|
|
int64_t offset; memcpy(&offset, arguments + 2, 8);
|
|
int32_t value = arguments[4];
|
|
|
|
fieldAtOffset<int32_t>(o, offset) = value;
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
|
Avian_sun_misc_Unsafe_getByte__Ljava_lang_Object_2J
|
|
(Thread*, object, uintptr_t* arguments)
|
|
{
|
|
object o = reinterpret_cast<object>(arguments[1]);
|
|
int64_t offset; memcpy(&offset, arguments + 2, 8);
|
|
|
|
return fieldAtOffset<int8_t>(o, offset);
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
|
Avian_sun_misc_Unsafe_getBoolean__Ljava_lang_Object_2J
|
|
(Thread* t, object method, uintptr_t* arguments)
|
|
{
|
|
return Avian_sun_misc_Unsafe_getByte__Ljava_lang_Object_2J
|
|
(t, method, arguments);
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
|
Avian_sun_misc_Unsafe_putBoolean__Ljava_lang_Object_2JZ
|
|
(Thread*, object, uintptr_t* arguments)
|
|
{
|
|
object o = reinterpret_cast<object>(arguments[1]);
|
|
int64_t offset; memcpy(&offset, arguments + 2, 8);
|
|
uint8_t value = arguments[4];
|
|
|
|
fieldAtOffset<uint8_t>(o, offset) = value;
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
|
Avian_sun_misc_Unsafe_putLong__Ljava_lang_Object_2JJ
|
|
(Thread*, object, uintptr_t* arguments)
|
|
{
|
|
object o = reinterpret_cast<object>(arguments[1]);
|
|
int64_t offset; memcpy(&offset, arguments + 2, 8);
|
|
int64_t value; memcpy(&value, arguments + 4, 8);
|
|
|
|
fieldAtOffset<int64_t>(o, offset) = value;
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
|
Avian_sun_misc_Unsafe_pageSize
|
|
(Thread*, object, uintptr_t*)
|
|
{
|
|
return local::PageSize;
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
|
Avian_sun_misc_Unsafe_ensureClassInitialized
|
|
(Thread* t, object, uintptr_t* arguments)
|
|
{
|
|
initClass(t, jclassVmClass(t, reinterpret_cast<object>(arguments[1])));
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
|
Avian_sun_misc_Unsafe_monitorEnter
|
|
(Thread* t, object, uintptr_t* arguments)
|
|
{
|
|
acquire(t, reinterpret_cast<object>(arguments[1]));
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
|
Avian_sun_misc_Unsafe_monitorExit
|
|
(Thread* t, object, uintptr_t* arguments)
|
|
{
|
|
release(t, reinterpret_cast<object>(arguments[1]));
|
|
}
|
|
|
|
namespace {
|
|
|
|
namespace local {
|
|
|
|
extern "C" AVIAN_EXPORT jint JNICALL
|
|
EXPORT(JVM_GetInterfaceVersion)()
|
|
{
|
|
return local::InterfaceVersion;
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jint JNICALL
|
|
EXPORT(JVM_IHashCode)(Thread* t, jobject o)
|
|
{
|
|
ENTER(t, Thread::ActiveState);
|
|
|
|
return o ? objectHash(t, *o) : 0;
|
|
}
|
|
|
|
uint64_t
|
|
jvmWait(Thread* t, uintptr_t* arguments)
|
|
{
|
|
jobject o = reinterpret_cast<jobject>(arguments[0]);
|
|
jlong milliseconds; memcpy(&milliseconds, arguments + 1, sizeof(jlong));
|
|
|
|
vm::wait(t, *o, milliseconds);
|
|
|
|
return 1;
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
|
EXPORT(JVM_MonitorWait)(Thread* t, jobject o, jlong milliseconds)
|
|
{
|
|
uintptr_t arguments[1 + (sizeof(jlong) / BytesPerWord)];
|
|
arguments[0] = reinterpret_cast<uintptr_t>(o);
|
|
memcpy(arguments + 1, &milliseconds, sizeof(jlong));
|
|
|
|
run(t, jvmWait, arguments);
|
|
}
|
|
|
|
uint64_t
|
|
jvmNotify(Thread* t, uintptr_t* arguments)
|
|
{
|
|
jobject o = reinterpret_cast<jobject>(arguments[0]);
|
|
|
|
notify(t, *o);
|
|
|
|
return 1;
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
|
EXPORT(JVM_MonitorNotify)(Thread* t, jobject o)
|
|
{
|
|
uintptr_t arguments[] = { reinterpret_cast<uintptr_t>(o) };
|
|
|
|
run(t, jvmNotify, arguments);
|
|
}
|
|
|
|
uint64_t
|
|
jvmNotifyAll(Thread* t, uintptr_t* arguments)
|
|
{
|
|
jobject o = reinterpret_cast<jobject>(arguments[0]);
|
|
|
|
notifyAll(t, *o);
|
|
|
|
return 1;
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
|
EXPORT(JVM_MonitorNotifyAll)(Thread* t, jobject o)
|
|
{
|
|
uintptr_t arguments[] = { reinterpret_cast<uintptr_t>(o) };
|
|
|
|
run(t, jvmNotifyAll, arguments);
|
|
}
|
|
|
|
uint64_t
|
|
jvmClone(Thread* t, uintptr_t* arguments)
|
|
{
|
|
jobject o = reinterpret_cast<jobject>(arguments[0]);
|
|
|
|
return reinterpret_cast<uint64_t>(makeLocalReference(t, clone(t, *o)));
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jobject JNICALL
|
|
EXPORT(JVM_Clone)(Thread* t, jobject o)
|
|
{
|
|
uintptr_t arguments[] = { reinterpret_cast<uintptr_t>(o) };
|
|
|
|
return reinterpret_cast<jobject>(run(t, jvmClone, arguments));
|
|
}
|
|
|
|
uint64_t
|
|
jvmInternString(Thread* t, uintptr_t* arguments)
|
|
{
|
|
jobject o = reinterpret_cast<jobject>(arguments[0]);
|
|
|
|
return reinterpret_cast<uint64_t>(makeLocalReference(t, intern(t, *o)));
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jstring JNICALL
|
|
EXPORT(JVM_InternString)(Thread* t, jstring s)
|
|
{
|
|
uintptr_t arguments[] = { reinterpret_cast<uintptr_t>(s) };
|
|
|
|
return reinterpret_cast<jobject>(run(t, jvmInternString, arguments));
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jlong JNICALL
|
|
EXPORT(JVM_CurrentTimeMillis)(Thread* t, jclass)
|
|
{
|
|
return t->m->system->now();
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jlong JNICALL
|
|
EXPORT(JVM_NanoTime)(Thread* t, jclass)
|
|
{
|
|
return t->m->system->now() * 1000 * 1000;
|
|
}
|
|
|
|
uint64_t
|
|
jvmArrayCopy(Thread* t, uintptr_t* arguments)
|
|
{
|
|
jobject src = reinterpret_cast<jobject>(arguments[0]);
|
|
jint srcOffset = arguments[1];
|
|
jobject dst = reinterpret_cast<jobject>(arguments[2]);
|
|
jint dstOffset = arguments[3];
|
|
jint length = arguments[4];
|
|
|
|
arrayCopy(t, *src, srcOffset, *dst, dstOffset, length);
|
|
|
|
return 1;
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
|
EXPORT(JVM_ArrayCopy)(Thread* t, jclass, jobject src, jint srcOffset,
|
|
jobject dst, jint dstOffset, jint length)
|
|
{
|
|
uintptr_t arguments[] = { reinterpret_cast<uintptr_t>(src),
|
|
static_cast<uintptr_t>(srcOffset),
|
|
reinterpret_cast<uintptr_t>(dst),
|
|
static_cast<uintptr_t>(dstOffset),
|
|
static_cast<uintptr_t>(length) };
|
|
|
|
run(t, jvmArrayCopy, arguments);
|
|
}
|
|
|
|
uint64_t
|
|
jvmInitProperties(Thread* t, uintptr_t* arguments)
|
|
{
|
|
jobject properties = reinterpret_cast<jobject>(arguments[0]);
|
|
|
|
object method = resolveMethod
|
|
(t, root(t, Machine::BootLoader), "java/util/Properties", "setProperty",
|
|
"(Ljava/lang/String;Ljava/lang/String;)Ljava/lang/Object;");
|
|
|
|
PROTECT(t, method);
|
|
|
|
#ifdef PLATFORM_WINDOWS
|
|
local::setProperty(t, method, *properties, "line.separator", "\r\n");
|
|
local::setProperty(t, method, *properties, "file.separator", "\\");
|
|
local::setProperty(t, method, *properties, "path.separator", ";");
|
|
local::setProperty(t, method, *properties, "os.name", "Windows");
|
|
|
|
TCHAR buffer[MAX_PATH];
|
|
GetTempPath(MAX_PATH, buffer);
|
|
|
|
local::setProperty(t, method, *properties, "java.io.tmpdir", buffer);
|
|
local::setProperty(t, method, *properties, "java.home", buffer);
|
|
local::setProperty(t, method, *properties, "user.home",
|
|
_wgetenv(L"USERPROFILE"), "%ls");
|
|
|
|
GetCurrentDirectory(MAX_PATH, buffer);
|
|
|
|
local::setProperty(t, method, *properties, "user.dir", buffer);
|
|
#else // not PLATFORM_WINDOWS
|
|
local::setProperty(t, method, *properties, "line.separator", "\n");
|
|
local::setProperty(t, method, *properties, "file.separator", "/");
|
|
local::setProperty(t, method, *properties, "path.separator", ":");
|
|
# ifdef __APPLE__
|
|
local::setProperty(t, method, *properties, "os.name", "Mac OS X");
|
|
# elif defined __FreeBSD__
|
|
local::setProperty(t, method, *properties, "os.name", "FreeBSD");
|
|
# else // not __APPLE__
|
|
local::setProperty(t, method, *properties, "os.name", "Linux");
|
|
# endif // not __APPLE__
|
|
local::setProperty(t, method, *properties, "java.io.tmpdir", "/tmp");
|
|
local::setProperty(t, method, *properties, "user.home", getenv("HOME"));
|
|
|
|
char buffer[PATH_MAX];
|
|
local::setProperty(t, method, *properties, "user.dir",
|
|
getcwd(buffer, PATH_MAX));
|
|
#endif // not PLATFORM_WINDOWS
|
|
|
|
local::setProperty(t, method, *properties, "java.protocol.handler.pkgs",
|
|
"avian");
|
|
|
|
local::setProperty(t, method, *properties, "java.vm.vendor",
|
|
"Avian Contributors");
|
|
|
|
local::setProperty(t, method, *properties, "java.vm.name","Avian");
|
|
#ifdef AVIAN_VERSION
|
|
local::setProperty(t, method, *properties, "java.vm.version",AVIAN_VERSION);
|
|
#endif
|
|
#ifdef AVIAN_INFO
|
|
local::setProperty(t, method, *properties, "java.vm.info",AVIAN_INFO);
|
|
#endif
|
|
|
|
local::setProperty
|
|
(t, method, *properties, "java.home",
|
|
static_cast<local::MyClasspath*>(t->m->classpath)->javaHome);
|
|
|
|
local::setProperty
|
|
(t, method, *properties, "sun.boot.library.path",
|
|
static_cast<local::MyClasspath*>(t->m->classpath)->libraryPath);
|
|
|
|
local::setProperty
|
|
(t, method, *properties, "sun.boot.class.path",
|
|
static_cast<Finder*>
|
|
(systemClassLoaderFinder(t, root(t, Machine::BootLoader)))->path());
|
|
|
|
local::setProperty(t, method, *properties, "file.encoding", "UTF-8");
|
|
#ifdef ARCH_x86_32
|
|
local::setProperty(t, method, *properties, "os.arch", "x86");
|
|
#elif defined ARCH_x86_64
|
|
local::setProperty(t, method, *properties, "os.arch", "x86_64");
|
|
#elif defined ARCH_powerpc
|
|
local::setProperty(t, method, *properties, "os.arch", "ppc");
|
|
#elif defined ARCH_arm
|
|
local::setProperty(t, method, *properties, "os.arch", "arm");
|
|
#else
|
|
local::setProperty(t, method, *properties, "os.arch", "unknown");
|
|
#endif
|
|
|
|
for (unsigned i = 0; i < t->m->propertyCount; ++i) {
|
|
const char* start = t->m->properties[i];
|
|
const char* p = start;
|
|
while (*p and *p != '=') ++p;
|
|
|
|
if (*p == '=') {
|
|
THREAD_RUNTIME_ARRAY(t, char, name, (p - start) + 1);
|
|
memcpy(RUNTIME_ARRAY_BODY(name), start, p - start);
|
|
RUNTIME_ARRAY_BODY(name)[p - start] = 0;
|
|
local::setProperty
|
|
(t, method, *properties, RUNTIME_ARRAY_BODY(name), p + 1);
|
|
}
|
|
}
|
|
|
|
return reinterpret_cast<uint64_t>(properties);
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jobject JNICALL
|
|
EXPORT(JVM_InitProperties)(Thread* t, jobject properties)
|
|
{
|
|
uintptr_t arguments[] = { reinterpret_cast<uintptr_t>(properties) };
|
|
|
|
return reinterpret_cast<jobject>(run(t, jvmInitProperties, arguments));
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
|
EXPORT(JVM_OnExit)(void (*)(void)) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
|
EXPORT(JVM_Exit)(jint code)
|
|
{
|
|
exit(code);
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
|
EXPORT(JVM_Halt)(jint code)
|
|
{
|
|
exit(code);
|
|
}
|
|
|
|
uint64_t
|
|
jvmGC(Thread* t, uintptr_t*)
|
|
{
|
|
collect(t, Heap::MajorCollection);
|
|
|
|
return 1;
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
|
EXPORT(JVM_GC)()
|
|
{
|
|
Thread* t = static_cast<Thread*>(local::globalMachine->localThread->get());
|
|
|
|
run(t, jvmGC, 0);
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jlong JNICALL
|
|
EXPORT(JVM_MaxObjectInspectionAge)(void)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
|
EXPORT(JVM_TraceInstructions)(jboolean) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
|
EXPORT(JVM_TraceMethodCalls)(jboolean) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT jlong JNICALL
|
|
EXPORT(JVM_TotalMemory)()
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jlong JNICALL
|
|
EXPORT(JVM_FreeMemory)()
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jlong JNICALL
|
|
EXPORT(JVM_MaxMemory)()
|
|
{
|
|
return local::globalMachine->heap->limit();
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jint JNICALL
|
|
EXPORT(JVM_ActiveProcessorCount)()
|
|
{
|
|
#ifdef PLATFORM_WINDOWS
|
|
SYSTEM_INFO si;
|
|
GetSystemInfo(&si);
|
|
return si.dwNumberOfProcessors;
|
|
#else
|
|
return sysconf(_SC_NPROCESSORS_ONLN);
|
|
#endif
|
|
}
|
|
|
|
uint64_t
|
|
jvmLoadLibrary(Thread* t, uintptr_t* arguments)
|
|
{
|
|
const char* path = reinterpret_cast<const char*>(arguments[0]);
|
|
|
|
THREAD_RUNTIME_ARRAY(t, char, p, strlen(path) + 1);
|
|
replace('\\', '/', RUNTIME_ARRAY_BODY(p), path);
|
|
|
|
return reinterpret_cast<uint64_t>
|
|
(loadLibrary
|
|
(t, static_cast<local::MyClasspath*>(t->m->classpath)->libraryPath,
|
|
RUNTIME_ARRAY_BODY(p), false, false));
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT void* JNICALL
|
|
EXPORT(JVM_LoadLibrary)(const char* path)
|
|
{
|
|
Thread* t = static_cast<Thread*>(local::globalMachine->localThread->get());
|
|
|
|
uintptr_t arguments[] = { reinterpret_cast<uintptr_t>(path) };
|
|
|
|
return reinterpret_cast<void*>(run(t, jvmLoadLibrary, arguments));
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
|
EXPORT(JVM_UnloadLibrary)(void*)
|
|
{
|
|
// todo: implement this correctly for POSIX and Windows
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT void* JNICALL
|
|
EXPORT(JVM_FindLibraryEntry)(void* library, const char* name)
|
|
{
|
|
Thread* t = static_cast<Thread*>(local::globalMachine->localThread->get());
|
|
|
|
ENTER(t, Thread::ActiveState);
|
|
|
|
if (library == RTLD_DEFAULT) {
|
|
library = t->m->libraries;
|
|
}
|
|
|
|
for (System::Library* lib = t->m->libraries; lib; lib = lib->next()) {
|
|
if (library == lib) {
|
|
return lib->resolve(name);
|
|
}
|
|
}
|
|
|
|
return 0;
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jboolean JNICALL
|
|
EXPORT(JVM_IsSupportedJNIVersion)(jint version)
|
|
{
|
|
return version <= JNI_VERSION_1_6;
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jboolean JNICALL
|
|
EXPORT(JVM_IsNaN)(jdouble) { abort(); }
|
|
|
|
uint64_t
|
|
jvmFillInStackTrace(Thread* t, uintptr_t* arguments)
|
|
{
|
|
jobject throwable = reinterpret_cast<jobject>(arguments[0]);
|
|
|
|
object trace = getTrace(t, 2);
|
|
set(t, *throwable, ThrowableTrace, trace);
|
|
|
|
return 1;
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
|
EXPORT(JVM_FillInStackTrace)(Thread* t, jobject throwable)
|
|
{
|
|
uintptr_t arguments[] = { reinterpret_cast<uintptr_t>(throwable) };
|
|
|
|
run(t, jvmFillInStackTrace, arguments);
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
|
EXPORT(JVM_PrintStackTrace)(Thread*, jobject, jobject) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT jint JNICALL
|
|
EXPORT(JVM_GetStackTraceDepth)(Thread* t, jobject throwable)
|
|
{
|
|
ENTER(t, Thread::ActiveState);
|
|
|
|
return objectArrayLength(t, throwableTrace(t, *throwable));
|
|
}
|
|
|
|
uint64_t
|
|
jvmGetStackTraceElement(Thread* t, uintptr_t* arguments)
|
|
{
|
|
jobject throwable = reinterpret_cast<jobject>(arguments[0]);
|
|
jint index = arguments[1];
|
|
|
|
return reinterpret_cast<uint64_t>
|
|
(makeLocalReference
|
|
(t, makeStackTraceElement
|
|
(t, objectArrayBody(t, throwableTrace(t, *throwable), index))));
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jobject JNICALL
|
|
EXPORT(JVM_GetStackTraceElement)(Thread* t, jobject throwable, jint index)
|
|
{
|
|
uintptr_t arguments[] = { reinterpret_cast<uintptr_t>(throwable),
|
|
static_cast<uintptr_t>(index) };
|
|
|
|
return reinterpret_cast<jobject>(run(t, jvmGetStackTraceElement, arguments));
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
|
EXPORT(JVM_InitializeCompiler) (Thread*, jclass) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT jboolean JNICALL
|
|
EXPORT(JVM_IsSilentCompiler)(Thread*, jclass) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT jboolean JNICALL
|
|
EXPORT(JVM_CompileClass)(Thread*, jclass, jclass)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jboolean JNICALL
|
|
EXPORT(JVM_CompileClasses)(Thread*, jclass, jstring)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jobject JNICALL
|
|
EXPORT(JVM_CompilerCommand)(Thread*, jclass, jobject) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
|
EXPORT(JVM_EnableCompiler)(Thread*, jclass)
|
|
{
|
|
// ignore
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
|
EXPORT(JVM_DisableCompiler)(Thread*, jclass)
|
|
{
|
|
// ignore
|
|
}
|
|
|
|
uint64_t
|
|
jvmStartThread(Thread* t, uintptr_t* arguments)
|
|
{
|
|
jobject thread = reinterpret_cast<jobject>(arguments[0]);
|
|
|
|
return startThread(t, *thread) != 0;
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
|
EXPORT(JVM_StartThread)(Thread* t, jobject thread)
|
|
{
|
|
uintptr_t arguments[] = { reinterpret_cast<uintptr_t>(thread) };
|
|
|
|
run(t, jvmStartThread, arguments);
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
|
EXPORT(JVM_StopThread)(Thread*, jobject, jobject) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT jboolean JNICALL
|
|
EXPORT(JVM_IsThreadAlive)(Thread* t, jobject thread)
|
|
{
|
|
ENTER(t, Thread::ActiveState);
|
|
|
|
Thread* p = reinterpret_cast<Thread*>(threadPeer(t, *thread));
|
|
return p and (p->flags & Thread::ActiveFlag) != 0;
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
|
EXPORT(JVM_SuspendThread)(Thread*, jobject) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
|
EXPORT(JVM_ResumeThread)(Thread*, jobject) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
|
EXPORT(JVM_SetThreadPriority)(Thread*, jobject, jint)
|
|
{
|
|
// ignore
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
|
EXPORT(JVM_Yield)(Thread* t, jclass)
|
|
{
|
|
t->m->system->yield();
|
|
}
|
|
|
|
uint64_t
|
|
jvmSleep(Thread* t, uintptr_t* arguments)
|
|
{
|
|
jlong milliseconds; memcpy(&milliseconds, arguments, sizeof(jlong));
|
|
|
|
if (milliseconds <= 0) {
|
|
milliseconds = 1;
|
|
}
|
|
|
|
if (threadSleepLock(t, t->javaThread) == 0) {
|
|
object lock = makeJobject(t);
|
|
set(t, t->javaThread, ThreadSleepLock, lock);
|
|
}
|
|
|
|
acquire(t, threadSleepLock(t, t->javaThread));
|
|
vm::wait(t, threadSleepLock(t, t->javaThread), milliseconds);
|
|
release(t, threadSleepLock(t, t->javaThread));
|
|
|
|
return 1;
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
|
EXPORT(JVM_Sleep)(Thread* t, jclass, jlong milliseconds)
|
|
{
|
|
uintptr_t arguments[sizeof(jlong) / BytesPerWord];
|
|
memcpy(arguments, &milliseconds, sizeof(jlong));
|
|
|
|
run(t, jvmSleep, arguments);
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jobject JNICALL
|
|
EXPORT(JVM_CurrentThread)(Thread* t, jclass)
|
|
{
|
|
ENTER(t, Thread::ActiveState);
|
|
|
|
return makeLocalReference(t, t->javaThread);
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jint JNICALL
|
|
EXPORT(JVM_CountStackFrames)(Thread*, jobject) { abort(); }
|
|
|
|
uint64_t
|
|
jvmInterrupt(Thread* t, uintptr_t* arguments)
|
|
{
|
|
threadInterrupt(t, *reinterpret_cast<jobject>(arguments[0]));
|
|
|
|
return 1;
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
|
EXPORT(JVM_Interrupt)(Thread* t, jobject thread)
|
|
{
|
|
uintptr_t arguments[] = { reinterpret_cast<uintptr_t>(thread) };
|
|
|
|
run(t, jvmInterrupt, arguments);
|
|
}
|
|
|
|
uint64_t
|
|
jvmIsInterrupted(Thread* t, uintptr_t* arguments)
|
|
{
|
|
jobject thread = reinterpret_cast<jobject>(arguments[0]);
|
|
jboolean clear = arguments[1];
|
|
|
|
return threadIsInterrupted(t, *thread, clear);
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jboolean JNICALL
|
|
EXPORT(JVM_IsInterrupted)(Thread* t, jobject thread, jboolean clear)
|
|
{
|
|
uintptr_t arguments[] = { reinterpret_cast<uintptr_t>(thread), clear };
|
|
|
|
return run(t, jvmIsInterrupted, arguments);
|
|
}
|
|
|
|
uint64_t
|
|
jvmHoldsLock(Thread* t, uintptr_t* arguments)
|
|
{
|
|
object m = objectMonitor(t, *reinterpret_cast<jobject>(arguments[0]), false);
|
|
|
|
return m and monitorOwner(t, m) == t;
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jboolean JNICALL
|
|
EXPORT(JVM_HoldsLock)(Thread* t, jclass, jobject o)
|
|
{
|
|
uintptr_t arguments[] = { reinterpret_cast<uintptr_t>(o) };
|
|
|
|
return run(t, jvmHoldsLock, arguments);
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
|
EXPORT(JVM_DumpAllStacks)(Thread*, jclass) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT jobjectArray JNICALL
|
|
EXPORT(JVM_GetAllThreads)(Thread*, jclass) { abort(); }
|
|
|
|
uint64_t
|
|
jvmDumpThreads(Thread* t, uintptr_t* arguments)
|
|
{
|
|
jobjectArray threads = reinterpret_cast<jobjectArray>(arguments[0]);
|
|
|
|
unsigned threadsLength = objectArrayLength(t, *threads);
|
|
object arrayClass = resolveObjectArrayClass
|
|
(t, classLoader(t, type(t, Machine::StackTraceElementType)),
|
|
type(t, Machine::StackTraceElementType));
|
|
object result = makeObjectArray(t, arrayClass, threadsLength);
|
|
PROTECT(t, result);
|
|
|
|
for (unsigned threadsIndex = 0; threadsIndex < threadsLength;
|
|
++ threadsIndex)
|
|
{
|
|
Thread* peer = reinterpret_cast<Thread*>
|
|
(threadPeer(t, objectArrayBody(t, *threads, threadsIndex)));
|
|
|
|
if (peer) {
|
|
object trace = t->m->processor->getStackTrace(t, peer);
|
|
PROTECT(t, trace);
|
|
|
|
unsigned traceLength = objectArrayLength(t, trace);
|
|
object array = makeObjectArray
|
|
(t, type(t, Machine::StackTraceElementType), traceLength);
|
|
PROTECT(t, array);
|
|
|
|
for (unsigned traceIndex = 0; traceIndex < traceLength; ++ traceIndex) {
|
|
object ste = makeStackTraceElement
|
|
(t, objectArrayBody(t, trace, traceIndex));
|
|
set(t, array, ArrayBody + (traceIndex * BytesPerWord), ste);
|
|
}
|
|
|
|
set(t, result, ArrayBody + (threadsIndex * BytesPerWord), array);
|
|
}
|
|
}
|
|
|
|
return reinterpret_cast<uint64_t>(makeLocalReference(t, result));
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jobjectArray JNICALL
|
|
EXPORT(JVM_DumpThreads)(Thread* t, jclass, jobjectArray threads)
|
|
{
|
|
uintptr_t arguments[] = { reinterpret_cast<uintptr_t>(threads) };
|
|
|
|
return reinterpret_cast<jobjectArray>(run(t, jvmDumpThreads, arguments));
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jclass JNICALL
|
|
EXPORT(JVM_CurrentLoadedClass)(Thread*) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT jobject JNICALL
|
|
EXPORT(JVM_CurrentClassLoader)(Thread*)
|
|
{
|
|
// just return null, i.e. tell SecurityManager.currentClassLoader
|
|
// all permissions are granted, since Avian doesn't do any internal
|
|
// security checks:
|
|
return 0;
|
|
}
|
|
|
|
uint64_t
|
|
jvmGetClassContext(Thread* t, uintptr_t*)
|
|
{
|
|
object trace = getTrace(t, 1);
|
|
PROTECT(t, trace);
|
|
|
|
object context = makeObjectArray
|
|
(t, type(t, Machine::JclassType), objectArrayLength(t, trace));
|
|
PROTECT(t, context);
|
|
|
|
for (unsigned i = 0; i < objectArrayLength(t, trace); ++i) {
|
|
object c = getJClass
|
|
(t, methodClass(t, traceElementMethod(t, objectArrayBody(t, trace, i))));
|
|
|
|
set(t, context, ArrayBody + (i * BytesPerWord), c);
|
|
}
|
|
|
|
return reinterpret_cast<uint64_t>(makeLocalReference(t, context));
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jobjectArray JNICALL
|
|
EXPORT(JVM_GetClassContext)(Thread* t)
|
|
{
|
|
return reinterpret_cast<jobjectArray>(run(t, jvmGetClassContext, 0));
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jint JNICALL
|
|
EXPORT(JVM_ClassDepth)(Thread*, jstring) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT jint JNICALL
|
|
EXPORT(JVM_ClassLoaderDepth)(Thread*) { abort(); }
|
|
|
|
uint64_t
|
|
jvmGetSystemPackage(Thread* t, uintptr_t* arguments)
|
|
{
|
|
jstring s = reinterpret_cast<jstring>(arguments[0]);
|
|
|
|
ACQUIRE(t, t->m->classLock);
|
|
|
|
THREAD_RUNTIME_ARRAY(t, char, chars, stringLength(t, *s) + 1);
|
|
stringChars(t, *s, RUNTIME_ARRAY_BODY(chars));
|
|
|
|
object key = makeByteArray(t, RUNTIME_ARRAY_BODY(chars));
|
|
|
|
object array = hashMapFind
|
|
(t, root(t, Machine::PackageMap), key, byteArrayHash, byteArrayEqual);
|
|
|
|
if (array) {
|
|
return reinterpret_cast<uintptr_t>
|
|
(makeLocalReference
|
|
(t, t->m->classpath->makeString
|
|
(t, array, 0, byteArrayLength(t, array))));
|
|
} else {
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jstring JNICALL
|
|
EXPORT(JVM_GetSystemPackage)(Thread* t, jstring s)
|
|
{
|
|
uintptr_t arguments[] = { reinterpret_cast<uintptr_t>(s) };
|
|
|
|
return reinterpret_cast<jstring>(run(t, jvmGetSystemPackage, arguments));
|
|
}
|
|
|
|
uint64_t
|
|
jvmGetSystemPackages(Thread* t, uintptr_t*)
|
|
{
|
|
return reinterpret_cast<uintptr_t>
|
|
(makeLocalReference
|
|
(t, makeObjectArray
|
|
(t, resolveClass
|
|
(t, root(t, Machine::BootLoader), "java/lang/Package"), 0)));
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jobjectArray JNICALL
|
|
EXPORT(JVM_GetSystemPackages)(Thread* t)
|
|
{
|
|
return reinterpret_cast<jobjectArray>(run(t, jvmGetSystemPackages, 0));
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jobject JNICALL
|
|
EXPORT(JVM_AllocateNewObject)(Thread*, jobject, jclass,
|
|
jclass) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT jobject JNICALL
|
|
EXPORT(JVM_AllocateNewArray)(Thread*, jobject, jclass,
|
|
jint) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT jobject JNICALL
|
|
EXPORT(JVM_LatestUserDefinedLoader)(Thread* t)
|
|
{
|
|
ENTER(t, Thread::ActiveState);
|
|
|
|
class Visitor: public Processor::StackVisitor {
|
|
public:
|
|
Visitor(Thread* t):
|
|
t(t), loader(0)
|
|
{ }
|
|
|
|
virtual bool visit(Processor::StackWalker* walker) {
|
|
object loader = classLoader(t, methodClass(t, walker->method()));
|
|
if (loader
|
|
and loader != root(t, Machine::BootLoader)
|
|
and strcmp
|
|
(&byteArrayBody(t, className(t, objectClass(t, loader)), 0),
|
|
reinterpret_cast<const int8_t*>
|
|
("sun/reflect/DelegatingClassLoader")))
|
|
{
|
|
this->loader = loader;
|
|
return false;
|
|
} else {
|
|
return true;
|
|
}
|
|
}
|
|
|
|
Thread* t;
|
|
object loader;
|
|
} v(t);
|
|
|
|
t->m->processor->walkStack(t, &v);
|
|
|
|
return makeLocalReference(t, v.loader);
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jclass JNICALL
|
|
EXPORT(JVM_LoadClass0)(Thread*, jobject, jclass,
|
|
jstring) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT jint JNICALL
|
|
EXPORT(JVM_GetArrayLength)(Thread* t, jobject array)
|
|
{
|
|
ENTER(t, Thread::ActiveState);
|
|
|
|
return fieldAtOffset<uintptr_t>(*array, BytesPerWord);
|
|
}
|
|
|
|
uint64_t
|
|
jvmGetArrayElement(Thread* t, uintptr_t* arguments)
|
|
{
|
|
jobject array = reinterpret_cast<jobject>(arguments[0]);
|
|
jint index = arguments[1];
|
|
|
|
switch (byteArrayBody(t, className(t, objectClass(t, *array)), 1)) {
|
|
case 'Z':
|
|
return reinterpret_cast<intptr_t>
|
|
(makeLocalReference
|
|
(t, makeBoolean(t, fieldAtOffset<int8_t>(*array, ArrayBody + index))));
|
|
case 'B':
|
|
return reinterpret_cast<intptr_t>
|
|
(makeLocalReference
|
|
(t, makeByte(t, fieldAtOffset<int8_t>(*array, ArrayBody + index))));
|
|
case 'C':
|
|
return reinterpret_cast<intptr_t>
|
|
(makeLocalReference
|
|
(t, makeChar(t, fieldAtOffset<int16_t>(*array, ArrayBody + (index * 2)))));
|
|
case 'S':
|
|
return reinterpret_cast<intptr_t>
|
|
(makeLocalReference
|
|
(t, makeShort(t, fieldAtOffset<int16_t>(*array, ArrayBody + (index * 2)))));
|
|
case 'I':
|
|
return reinterpret_cast<intptr_t>
|
|
(makeLocalReference
|
|
(t, makeInt(t, fieldAtOffset<int32_t>(*array, ArrayBody + (index * 4)))));
|
|
case 'F':
|
|
return reinterpret_cast<intptr_t>
|
|
(makeLocalReference
|
|
(t, makeFloat(t, fieldAtOffset<int32_t>(*array, ArrayBody + (index * 4)))));
|
|
case 'J':
|
|
return reinterpret_cast<intptr_t>
|
|
(makeLocalReference
|
|
(t, makeLong(t, fieldAtOffset<int64_t>(*array, ArrayBody + (index * 8)))));
|
|
case 'D':
|
|
return reinterpret_cast<intptr_t>
|
|
(makeLocalReference
|
|
(t, makeDouble(t, fieldAtOffset<int64_t>(*array, ArrayBody + (index * 8)))));
|
|
case 'L':
|
|
case '[':
|
|
return reinterpret_cast<intptr_t>
|
|
(makeLocalReference
|
|
(t, fieldAtOffset<object>(*array, ArrayBody + (index * BytesPerWord))));
|
|
default:
|
|
abort(t);
|
|
}
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jobject JNICALL
|
|
EXPORT(JVM_GetArrayElement)(Thread* t, jobject array, jint index)
|
|
{
|
|
uintptr_t arguments[] = { reinterpret_cast<uintptr_t>(array),
|
|
static_cast<uintptr_t>(index) };
|
|
|
|
return reinterpret_cast<jobject>(run(t, jvmGetArrayElement, arguments));
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jvalue JNICALL
|
|
EXPORT(JVM_GetPrimitiveArrayElement)(Thread*, jobject, jint, jint) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
|
EXPORT(JVM_SetArrayElement)(Thread* t, jobject array, jint index,
|
|
jobject value)
|
|
{
|
|
ENTER(t, Thread::ActiveState);
|
|
|
|
switch (byteArrayBody(t, className(t, objectClass(t, *array)), 1)) {
|
|
case 'Z':
|
|
fieldAtOffset<int8_t>(*array, ArrayBody + index) = booleanValue(t, *value);
|
|
break;
|
|
case 'B':
|
|
fieldAtOffset<int8_t>(*array, ArrayBody + index) = byteValue(t, *value);
|
|
break;
|
|
case 'C':
|
|
fieldAtOffset<int16_t>(*array, ArrayBody + (index * 2)) = charValue(t, *value);
|
|
break;
|
|
case 'S':
|
|
fieldAtOffset<int16_t>(*array, ArrayBody + (index * 2)) = shortValue(t, *value);
|
|
break;
|
|
case 'I':
|
|
fieldAtOffset<int32_t>(*array, ArrayBody + (index * 4)) = intValue(t, *value);
|
|
break;
|
|
case 'F':
|
|
fieldAtOffset<int32_t>(*array, ArrayBody + (index * 4)) = floatValue(t, *value);
|
|
break;
|
|
case 'J':
|
|
fieldAtOffset<int64_t>(*array, ArrayBody + (index * 8)) = longValue(t, *value);
|
|
break;
|
|
case 'D':
|
|
fieldAtOffset<int64_t>(*array, ArrayBody + (index * 8)) = doubleValue(t, *value);
|
|
break;
|
|
case 'L':
|
|
case '[':
|
|
set(t, *array, ArrayBody + (index * BytesPerWord), (value ? *value : 0));
|
|
break;
|
|
default:
|
|
abort(t);
|
|
}
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
|
EXPORT(JVM_SetPrimitiveArrayElement)(Thread*, jobject, jint, jvalue,
|
|
unsigned char) { abort(); }
|
|
|
|
object
|
|
makeNewArray(Thread* t, object c, unsigned length)
|
|
{
|
|
if (classVmFlags(t, c) & PrimitiveFlag) {
|
|
const char* name = reinterpret_cast<char*>
|
|
(&byteArrayBody(t, local::getClassName(t, c), 0));
|
|
|
|
switch (*name) {
|
|
case 'b':
|
|
if (name[1] == 'o') {
|
|
return makeBooleanArray(t, length);
|
|
} else {
|
|
return makeByteArray(t, length);
|
|
}
|
|
case 'c': return makeCharArray(t, length);
|
|
case 'd': return makeDoubleArray(t, length);
|
|
case 'f': return makeFloatArray(t, length);
|
|
case 'i': return makeIntArray(t, length);
|
|
case 'l': return makeLongArray(t, length);
|
|
case 's': return makeShortArray(t, length);
|
|
default: abort(t);
|
|
}
|
|
} else {
|
|
return makeObjectArray(t, c, length);
|
|
}
|
|
}
|
|
|
|
uint64_t
|
|
jvmNewArray(Thread* t, uintptr_t* arguments)
|
|
{
|
|
jclass elementClass = reinterpret_cast<jclass>(arguments[0]);
|
|
jint length = arguments[1];
|
|
|
|
return reinterpret_cast<uint64_t>
|
|
(makeLocalReference
|
|
(t, makeNewArray(t, jclassVmClass(t, *elementClass), length)));
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jobject JNICALL
|
|
EXPORT(JVM_NewArray)(Thread* t, jclass elementClass, jint length)
|
|
{
|
|
uintptr_t arguments[] = { reinterpret_cast<uintptr_t>(elementClass),
|
|
static_cast<uintptr_t>(length) };
|
|
|
|
return reinterpret_cast<jobject>(run(t, jvmNewArray, arguments));
|
|
}
|
|
|
|
uint64_t
|
|
jvmNewMultiArray(Thread* t, uintptr_t* arguments)
|
|
{
|
|
jclass elementClass = reinterpret_cast<jclass>(arguments[0]);
|
|
jintArray dimensions = reinterpret_cast<jintArray>(arguments[1]);
|
|
|
|
THREAD_RUNTIME_ARRAY(t, int32_t, counts, intArrayLength(t, *dimensions));
|
|
for (int i = intArrayLength(t, *dimensions) - 1; i >= 0; --i) {
|
|
RUNTIME_ARRAY_BODY(counts)[i] = intArrayBody(t, *dimensions, i);
|
|
if (UNLIKELY(RUNTIME_ARRAY_BODY(counts)[i] < 0)) {
|
|
throwNew(t, Machine::NegativeArraySizeExceptionType, "%d",
|
|
RUNTIME_ARRAY_BODY(counts)[i]);
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
object array = makeNewArray
|
|
(t, jclassVmClass(t, *elementClass), RUNTIME_ARRAY_BODY(counts)[0]);
|
|
PROTECT(t, array);
|
|
|
|
populateMultiArray(t, array, RUNTIME_ARRAY_BODY(counts), 0,
|
|
intArrayLength(t, *dimensions));
|
|
|
|
return reinterpret_cast<uint64_t>(makeLocalReference(t, array));
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jobject JNICALL
|
|
EXPORT(JVM_NewMultiArray)(Thread* t, jclass elementClass,
|
|
jintArray dimensions)
|
|
{
|
|
uintptr_t arguments[] = { reinterpret_cast<uintptr_t>(elementClass),
|
|
reinterpret_cast<uintptr_t>(dimensions) };
|
|
|
|
return reinterpret_cast<jobject>(run(t, jvmNewMultiArray, arguments));
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jclass JNICALL
|
|
EXPORT(JVM_GetCallerClass)(Thread* t, int target)
|
|
{
|
|
ENTER(t, Thread::ActiveState);
|
|
|
|
object method = getCaller(t, target, true);
|
|
|
|
return method ? makeLocalReference
|
|
(t, getJClass(t, methodClass(t, method))) : 0;
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jclass JNICALL
|
|
EXPORT(JVM_FindPrimitiveClass)(Thread* t, const char* name)
|
|
{
|
|
ENTER(t, Thread::ActiveState);
|
|
|
|
switch (*name) {
|
|
case 'b':
|
|
if (name[1] == 'o') {
|
|
return makeLocalReference
|
|
(t, getJClass(t, type(t, Machine::JbooleanType)));
|
|
} else {
|
|
return makeLocalReference
|
|
(t, getJClass(t, type(t, Machine::JbyteType)));
|
|
}
|
|
case 'c':
|
|
return makeLocalReference
|
|
(t, getJClass(t, type(t, Machine::JcharType)));
|
|
case 'd':
|
|
return makeLocalReference
|
|
(t, getJClass(t, type(t, Machine::JdoubleType)));
|
|
case 'f':
|
|
return makeLocalReference
|
|
(t, getJClass(t, type(t, Machine::JfloatType)));
|
|
case 'i':
|
|
return makeLocalReference
|
|
(t, getJClass(t, type(t, Machine::JintType)));
|
|
case 'l':
|
|
return makeLocalReference
|
|
(t, getJClass(t, type(t, Machine::JlongType)));
|
|
case 's':
|
|
return makeLocalReference
|
|
(t, getJClass(t, type(t, Machine::JshortType)));
|
|
case 'v':
|
|
return makeLocalReference
|
|
(t, getJClass(t, type(t, Machine::JvoidType)));
|
|
default:
|
|
throwNew(t, Machine::IllegalArgumentExceptionType);
|
|
}
|
|
}
|
|
|
|
uint64_t
|
|
jvmResolveClass(Thread* t, uintptr_t* arguments)
|
|
{
|
|
jclass c = reinterpret_cast<jclass>(arguments[0]);
|
|
|
|
object method = resolveMethod
|
|
(t, root(t, Machine::BootLoader), "avian/Classes", "link",
|
|
"(Lavian/VMClass;)V");
|
|
|
|
t->m->processor->invoke(t, method, 0, jclassVmClass(t, *c));
|
|
|
|
return 1;
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
|
EXPORT(JVM_ResolveClass)(Thread* t, jclass c)
|
|
{
|
|
uintptr_t arguments[] = { reinterpret_cast<uintptr_t>(c) };
|
|
|
|
run(t, jvmResolveClass, arguments);
|
|
}
|
|
|
|
uint64_t
|
|
jvmFindClassFromClassLoader(Thread* t, uintptr_t* arguments)
|
|
{
|
|
const char* name = reinterpret_cast<const char*>(arguments[0]);
|
|
jboolean init = arguments[1];
|
|
jobject loader = reinterpret_cast<jobject>(arguments[2]);
|
|
jboolean throwError = arguments[3];
|
|
|
|
object c = resolveClass
|
|
(t, loader ? *loader : root(t, Machine::BootLoader), name, true,
|
|
throwError ? Machine::NoClassDefFoundErrorType
|
|
: Machine::ClassNotFoundExceptionType);
|
|
|
|
if (init) {
|
|
PROTECT(t, c);
|
|
|
|
initClass(t, c);
|
|
}
|
|
|
|
return reinterpret_cast<uint64_t>(makeLocalReference(t, getJClass(t, c)));
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jclass JNICALL
|
|
EXPORT(JVM_FindClassFromClassLoader)(Thread* t, const char* name,
|
|
jboolean init, jobject loader,
|
|
jboolean throwError)
|
|
{
|
|
uintptr_t arguments[] = { reinterpret_cast<uintptr_t>(name),
|
|
init,
|
|
reinterpret_cast<uintptr_t>(loader),
|
|
throwError };
|
|
|
|
return reinterpret_cast<jclass>
|
|
(run(t, jvmFindClassFromClassLoader, arguments));
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jclass JNICALL
|
|
JVM_FindClassFromBootLoader(Thread* t, const char* name)
|
|
{
|
|
return EXPORT(JVM_FindClassFromClassLoader)(t, name, false, 0, false);
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jclass JNICALL
|
|
EXPORT(JVM_FindClassFromClass)(Thread*, const char*, jboolean, jclass)
|
|
{ abort(); }
|
|
|
|
uint64_t
|
|
jvmFindLoadedClass(Thread* t, uintptr_t* arguments)
|
|
{
|
|
jobject loader = reinterpret_cast<jobject>(arguments[0]);
|
|
jstring name = reinterpret_cast<jstring>(arguments[1]);
|
|
|
|
object spec = makeByteArray(t, stringLength(t, *name) + 1);
|
|
|
|
{ char* s = reinterpret_cast<char*>(&byteArrayBody(t, spec, 0));
|
|
stringChars(t, *name, s);
|
|
replace('.', '/', s);
|
|
}
|
|
|
|
object c = findLoadedClass(t, *loader, spec);
|
|
|
|
return reinterpret_cast<uint64_t>
|
|
(c ? makeLocalReference(t, getJClass(t, c)) : 0);
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jclass JNICALL
|
|
EXPORT(JVM_FindLoadedClass)(Thread* t, jobject loader, jstring name)
|
|
{
|
|
uintptr_t arguments[] = { reinterpret_cast<uintptr_t>(loader),
|
|
reinterpret_cast<uintptr_t>(name) };
|
|
|
|
return reinterpret_cast<jclass>(run(t, jvmFindLoadedClass, arguments));
|
|
}
|
|
|
|
uint64_t
|
|
jvmDefineClass(Thread* t, uintptr_t* arguments)
|
|
{
|
|
jobject loader = reinterpret_cast<jobject>(arguments[0]);
|
|
const uint8_t* data = reinterpret_cast<const uint8_t*>(arguments[1]);
|
|
jsize length = arguments[2];
|
|
|
|
return reinterpret_cast<uint64_t>
|
|
(makeLocalReference
|
|
(t, getJClass(t, defineClass(t, *loader, data, length))));
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jclass JNICALL
|
|
EXPORT(JVM_DefineClass)(Thread* t, const char*, jobject loader,
|
|
const uint8_t* data, jsize length, jobject)
|
|
{
|
|
uintptr_t arguments[] = { reinterpret_cast<uintptr_t>(loader),
|
|
reinterpret_cast<uintptr_t>(data),
|
|
static_cast<uintptr_t>(length) };
|
|
|
|
return reinterpret_cast<jclass>(run(t, jvmDefineClass, arguments));
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jclass JNICALL
|
|
EXPORT(JVM_DefineClassWithSource)(Thread* t, const char*, jobject loader,
|
|
const uint8_t* data, jsize length, jobject,
|
|
const char*)
|
|
{
|
|
return EXPORT(JVM_DefineClass)(t, 0, loader, data, length, 0);
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jclass JNICALL
|
|
EXPORT(JVM_DefineClassWithSourceCond)(Thread* t, const char*, jobject loader,
|
|
const uint8_t* data, jsize length,
|
|
jobject, const char*, jboolean)
|
|
{
|
|
return EXPORT(JVM_DefineClass)(t, 0, loader, data, length, 0);
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jstring JNICALL
|
|
EXPORT(JVM_GetClassName)(Thread* t, jclass c)
|
|
{
|
|
ENTER(t, Thread::ActiveState);
|
|
|
|
return makeLocalReference(t, jclassName(t, *c));
|
|
}
|
|
|
|
uint64_t
|
|
jvmGetClassInterfaces(Thread* t, uintptr_t* arguments)
|
|
{
|
|
jclass c = reinterpret_cast<jclass>(arguments[0]);
|
|
|
|
object addendum = classAddendum(t, jclassVmClass(t, *c));
|
|
if (addendum) {
|
|
object table = classAddendumInterfaceTable(t, addendum);
|
|
if (table) {
|
|
PROTECT(t, table);
|
|
|
|
object array = makeObjectArray
|
|
(t, type(t, Machine::JclassType), arrayLength(t, table));
|
|
PROTECT(t, array);
|
|
|
|
for (unsigned i = 0; i < arrayLength(t, table); ++i) {
|
|
object c = getJClass(t, arrayBody(t, table, i));
|
|
set(t, array, ArrayBody + (i * BytesPerWord), c);
|
|
}
|
|
|
|
return reinterpret_cast<uint64_t>(makeLocalReference(t, array));
|
|
}
|
|
}
|
|
|
|
return reinterpret_cast<uint64_t>
|
|
(makeLocalReference
|
|
(t, makeObjectArray(t, type(t, Machine::JclassType), 0)));
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jobjectArray JNICALL
|
|
EXPORT(JVM_GetClassInterfaces)(Thread* t, jclass c)
|
|
{
|
|
uintptr_t arguments[] = { reinterpret_cast<uintptr_t>(c) };
|
|
|
|
return reinterpret_cast<jclass>(run(t, jvmGetClassInterfaces, arguments));
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jobject JNICALL
|
|
EXPORT(JVM_GetClassLoader)(Thread* t, jclass c)
|
|
{
|
|
ENTER(t, Thread::ActiveState);
|
|
|
|
object loader = classLoader(t, jclassVmClass(t, *c));
|
|
|
|
if (loader == root(t, Machine::BootLoader)) {
|
|
// sun.misc.Unsafe.getUnsafe expects a null result if the class
|
|
// loader is the boot classloader and will throw a
|
|
// SecurityException otherwise.
|
|
object caller = getCaller(t, 2);
|
|
if (caller and strcmp
|
|
(reinterpret_cast<const char*>
|
|
(&byteArrayBody(t, className(t, methodClass(t, caller)), 0)),
|
|
"sun/misc/Unsafe") == 0)
|
|
{
|
|
return 0;
|
|
} else {
|
|
return makeLocalReference(t, root(t, Machine::BootLoader));
|
|
}
|
|
} else {
|
|
return makeLocalReference(t, loader);
|
|
}
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jboolean JNICALL
|
|
EXPORT(JVM_IsInterface)(Thread* t, jclass c)
|
|
{
|
|
ENTER(t, Thread::ActiveState);
|
|
|
|
return (classFlags(t, jclassVmClass(t, *c)) & ACC_INTERFACE) != 0;
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jobjectArray JNICALL
|
|
EXPORT(JVM_GetClassSigners)(Thread* t, jclass c)
|
|
{
|
|
ENTER(t, Thread::ActiveState);
|
|
|
|
object runtimeData = getClassRuntimeDataIfExists(t, jclassVmClass(t, *c));
|
|
|
|
return runtimeData ? makeLocalReference
|
|
(t, classRuntimeDataSigners(t, runtimeData)) : 0;
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jbyteArray JNICALL
|
|
EXPORT(JVM_GetClassTypeAnnotations)(Thread*, jclass)
|
|
{
|
|
abort();
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jbyteArray JNICALL
|
|
EXPORT(JVM_GetFieldTypeAnnotations)(Thread*, jobject)
|
|
{
|
|
abort();
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jbyteArray JNICALL
|
|
EXPORT(JVM_GetMethodTypeAnnotations)(Thread*, jobject)
|
|
{
|
|
abort();
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
|
EXPORT(JVM_SetClassSigners)(Thread* t, jclass c, jobjectArray signers)
|
|
{
|
|
ENTER(t, Thread::ActiveState);
|
|
|
|
object runtimeData = getClassRuntimeData(t, jclassVmClass(t, *c));
|
|
|
|
set(t, runtimeData, ClassRuntimeDataSigners, *signers);
|
|
}
|
|
|
|
uint64_t
|
|
jvmGetProtectionDomain(Thread* t, uintptr_t* arguments)
|
|
{
|
|
jclass c = reinterpret_cast<jclass>(arguments[0]);
|
|
|
|
object method = resolveMethod
|
|
(t, root(t, Machine::BootLoader), "avian/Classes", "getProtectionDomain",
|
|
"(Lavian/VMClass;)Ljava/security/ProtectionDomain;");
|
|
|
|
return reinterpret_cast<uint64_t>
|
|
(makeLocalReference
|
|
(t, t->m->processor->invoke(t, method, 0, jclassVmClass(t, *c))));
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jobject JNICALL
|
|
EXPORT(JVM_GetProtectionDomain)(Thread* t, jclass c)
|
|
{
|
|
uintptr_t arguments[] = { reinterpret_cast<uintptr_t>(c) };
|
|
|
|
return reinterpret_cast<jobject>(run(t, jvmGetProtectionDomain, arguments));
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
|
EXPORT(JVM_SetProtectionDomain)(Thread*, jclass, jobject) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT jboolean JNICALL
|
|
EXPORT(JVM_IsArrayClass)(Thread* t, jclass c)
|
|
{
|
|
ENTER(t, Thread::ActiveState);
|
|
|
|
return classArrayDimensions(t, jclassVmClass(t, *c)) != 0;
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jboolean JNICALL
|
|
EXPORT(JVM_IsPrimitiveClass)(Thread* t, jclass c)
|
|
{
|
|
ENTER(t, Thread::ActiveState);
|
|
|
|
return (classVmFlags(t, jclassVmClass(t, *c)) & PrimitiveFlag) != 0;
|
|
}
|
|
|
|
uint64_t
|
|
jvmGetComponentType(Thread* t, uintptr_t* arguments)
|
|
{
|
|
jclass c = reinterpret_cast<jobject>(arguments[0]);
|
|
|
|
if (classArrayDimensions(t, jclassVmClass(t, *c))) {
|
|
uint8_t n = byteArrayBody(t, className(t, jclassVmClass(t, *c)), 1);
|
|
if (n != 'L' and n != '[') {
|
|
return reinterpret_cast<uintptr_t>
|
|
(makeLocalReference(t, getJClass(t, primitiveClass(t, n))));
|
|
} else {
|
|
return reinterpret_cast<uintptr_t>
|
|
(makeLocalReference
|
|
(t, getJClass(t, classStaticTable(t, jclassVmClass(t, *c)))));
|
|
}
|
|
} else {
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jclass JNICALL
|
|
EXPORT(JVM_GetComponentType)(Thread* t, jclass c)
|
|
{
|
|
uintptr_t arguments[] = { reinterpret_cast<uintptr_t>(c) };
|
|
|
|
return reinterpret_cast<jclass>(run(t, jvmGetComponentType, arguments));
|
|
}
|
|
|
|
uint64_t
|
|
jvmGetClassModifiers(Thread* t, uintptr_t* arguments)
|
|
{
|
|
return classModifiers
|
|
(t, jclassVmClass(t, *reinterpret_cast<jobject>(arguments[0])));
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jint JNICALL
|
|
EXPORT(JVM_GetClassModifiers)(Thread* t, jclass c)
|
|
{
|
|
uintptr_t arguments[] = { reinterpret_cast<uintptr_t>(c) };
|
|
|
|
return run(t, jvmGetClassModifiers, arguments);
|
|
}
|
|
|
|
uint64_t
|
|
jvmGetDeclaredClasses(Thread* t, uintptr_t* arguments)
|
|
{
|
|
return reinterpret_cast<uintptr_t>
|
|
(makeLocalReference
|
|
(t, getDeclaredClasses
|
|
(t, jclassVmClass(t, *reinterpret_cast<jobject>(arguments[0])), false)));
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jobjectArray JNICALL
|
|
EXPORT(JVM_GetDeclaredClasses)(Thread* t, jclass c)
|
|
{
|
|
uintptr_t arguments[] = { reinterpret_cast<uintptr_t>(c) };
|
|
|
|
return reinterpret_cast<jclass>(run(t, jvmGetDeclaredClasses, arguments));
|
|
}
|
|
|
|
uint64_t
|
|
jvmGetDeclaringClass(Thread* t, uintptr_t* arguments)
|
|
{
|
|
return reinterpret_cast<uintptr_t>
|
|
(makeLocalReference
|
|
(t, getDeclaringClass
|
|
(t, jclassVmClass(t, *reinterpret_cast<jobject>(arguments[0])))));
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jclass JNICALL
|
|
EXPORT(JVM_GetDeclaringClass)(Thread* t, jclass c)
|
|
{
|
|
uintptr_t arguments[] = { reinterpret_cast<uintptr_t>(c) };
|
|
|
|
return reinterpret_cast<jclass>(run(t, jvmGetDeclaringClass, arguments));
|
|
}
|
|
|
|
uint64_t
|
|
jvmGetClassSignature(Thread* t, uintptr_t* arguments)
|
|
{
|
|
jclass c = reinterpret_cast<jobject>(arguments[0]);
|
|
|
|
object addendum = classAddendum(t, jclassVmClass(t, *c));
|
|
if (addendum) {
|
|
object signature = addendumSignature(t, addendum);
|
|
if (signature) {
|
|
return reinterpret_cast<uintptr_t>
|
|
(makeLocalReference
|
|
(t, t->m->classpath->makeString
|
|
(t, signature, 0, byteArrayLength(t, signature) - 1)));
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jstring JNICALL
|
|
EXPORT(JVM_GetClassSignature)(Thread* t, jclass c)
|
|
{
|
|
uintptr_t arguments[] = { reinterpret_cast<uintptr_t>(c) };
|
|
|
|
return reinterpret_cast<jclass>(run(t, jvmGetClassSignature, arguments));
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jbyteArray JNICALL
|
|
EXPORT(JVM_GetClassAnnotations)(Thread* t, jclass c)
|
|
{
|
|
ENTER(t, Thread::ActiveState);
|
|
|
|
object addendum = classAddendum(t, jclassVmClass(t, *c));
|
|
return addendum
|
|
? makeLocalReference(t, addendumAnnotationTable(t, addendum)) : 0;
|
|
}
|
|
|
|
uint64_t
|
|
jvmGetClassDeclaredMethods(Thread* t, uintptr_t* arguments)
|
|
{
|
|
jclass c = reinterpret_cast<jclass>(arguments[0]);
|
|
jboolean publicOnly = arguments[1];
|
|
|
|
object table = classMethodTable(t, jclassVmClass(t, *c));
|
|
if (table) {
|
|
PROTECT(t, table);
|
|
|
|
object array = makeObjectArray
|
|
(t, type(t, Machine::JmethodType),
|
|
local::countMethods(t, jclassVmClass(t, *c), publicOnly));
|
|
PROTECT(t, array);
|
|
|
|
unsigned ai = 0;
|
|
for (unsigned i = 0, j = classDeclaredMethodCount(t, jclassVmClass(t, *c));
|
|
i < j; ++i)
|
|
{
|
|
object vmMethod = arrayBody(t, table, i);
|
|
PROTECT(t, vmMethod);
|
|
|
|
if (((not publicOnly) or (methodFlags(t, vmMethod) & ACC_PUBLIC))
|
|
and byteArrayBody(t, methodName(t, vmMethod), 0) != '<')
|
|
{
|
|
object method = makeJmethod(t, vmMethod, i);
|
|
|
|
assert(t, ai < objectArrayLength(t, array));
|
|
|
|
set(t, array, ArrayBody + ((ai++) * BytesPerWord), method);
|
|
}
|
|
}
|
|
|
|
return reinterpret_cast<uint64_t>(makeLocalReference(t, array));
|
|
} else {
|
|
return reinterpret_cast<uint64_t>
|
|
(makeLocalReference
|
|
(t, makeObjectArray(t, type(t, Machine::JmethodType), 0)));
|
|
}
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jobjectArray JNICALL
|
|
EXPORT(JVM_GetClassDeclaredMethods)(Thread* t, jclass c, jboolean publicOnly)
|
|
{
|
|
uintptr_t arguments[] = { reinterpret_cast<uintptr_t>(c), publicOnly };
|
|
|
|
return reinterpret_cast<jobjectArray>
|
|
(run(t, jvmGetClassDeclaredMethods, arguments));
|
|
}
|
|
|
|
uint64_t
|
|
jvmGetClassDeclaredFields(Thread* t, uintptr_t* arguments)
|
|
{
|
|
jclass c = reinterpret_cast<jclass>(arguments[0]);
|
|
jboolean publicOnly = arguments[1];
|
|
object table = classFieldTable(t, jclassVmClass(t, *c));
|
|
if (table) {
|
|
PROTECT(t, table);
|
|
|
|
object array = makeObjectArray
|
|
(t, type(t, Machine::JfieldType),
|
|
local::countFields(t, jclassVmClass(t, *c), publicOnly));
|
|
PROTECT(t, array);
|
|
|
|
unsigned ai = 0;
|
|
for (unsigned i = 0; i < arrayLength(t, table); ++i) {
|
|
object vmField = arrayBody(t, table, i);
|
|
PROTECT(t, vmField);
|
|
|
|
if ((not publicOnly) or (fieldFlags(t, vmField) & ACC_PUBLIC)) {
|
|
object field = makeJfield(t, vmField, i);
|
|
|
|
assert(t, ai < objectArrayLength(t, array));
|
|
|
|
set(t, array, ArrayBody + ((ai++) * BytesPerWord), field);
|
|
}
|
|
}
|
|
assert(t, ai == objectArrayLength(t, array));
|
|
|
|
return reinterpret_cast<uint64_t>(makeLocalReference(t, array));
|
|
} else {
|
|
return reinterpret_cast<uint64_t>
|
|
(makeLocalReference
|
|
(t, makeObjectArray(t, type(t, Machine::JfieldType), 0)));
|
|
}
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jobjectArray JNICALL
|
|
EXPORT(JVM_GetClassDeclaredFields)(Thread* t, jclass c, jboolean publicOnly)
|
|
{
|
|
uintptr_t arguments[] = { reinterpret_cast<uintptr_t>(c), publicOnly };
|
|
|
|
return reinterpret_cast<jobjectArray>
|
|
(run(t, jvmGetClassDeclaredFields, arguments));
|
|
}
|
|
|
|
uint64_t
|
|
jvmGetClassDeclaredConstructors(Thread* t, uintptr_t* arguments)
|
|
{
|
|
jclass c = reinterpret_cast<jclass>(arguments[0]);
|
|
jboolean publicOnly = arguments[1];
|
|
|
|
object table = classMethodTable(t, jclassVmClass(t, *c));
|
|
if (table) {
|
|
PROTECT(t, table);
|
|
|
|
object array = makeObjectArray
|
|
(t, type(t, Machine::JconstructorType),
|
|
local::countConstructors(t, jclassVmClass(t, *c), publicOnly));
|
|
PROTECT(t, array);
|
|
|
|
unsigned ai = 0;
|
|
for (unsigned i = 0, j = classDeclaredMethodCount(t, jclassVmClass(t, *c));
|
|
i < j; ++i)
|
|
{
|
|
object vmMethod = arrayBody(t, table, i);
|
|
PROTECT(t, vmMethod);
|
|
|
|
if (((not publicOnly) or (methodFlags(t, vmMethod) & ACC_PUBLIC))
|
|
and strcmp(reinterpret_cast<char*>
|
|
(&byteArrayBody(t, methodName(t, vmMethod), 0)),
|
|
"<init>") == 0)
|
|
{
|
|
object method = makeJconstructor(t, vmMethod, i);
|
|
|
|
assert(t, ai < objectArrayLength(t, array));
|
|
|
|
set(t, array, ArrayBody + ((ai++) * BytesPerWord), method);
|
|
}
|
|
}
|
|
|
|
return reinterpret_cast<uint64_t>(makeLocalReference(t, array));
|
|
} else {
|
|
return reinterpret_cast<uint64_t>
|
|
(makeLocalReference
|
|
(t, makeObjectArray(t, type(t, Machine::JconstructorType), 0)));
|
|
}
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jobjectArray JNICALL
|
|
EXPORT(JVM_GetClassDeclaredConstructors)(Thread* t, jclass c,
|
|
jboolean publicOnly)
|
|
{
|
|
uintptr_t arguments[] = { reinterpret_cast<uintptr_t>(c), publicOnly };
|
|
|
|
return reinterpret_cast<jobjectArray>
|
|
(run(t, jvmGetClassDeclaredConstructors, arguments));
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jint JNICALL
|
|
EXPORT(JVM_GetClassAccessFlags)(Thread* t, jclass c)
|
|
{
|
|
ENTER(t, Thread::ActiveState);
|
|
|
|
return classFlags(t, jclassVmClass(t, *c));
|
|
}
|
|
|
|
uint64_t
|
|
jvmInvokeMethod(Thread* t, uintptr_t* arguments)
|
|
{
|
|
jobject method = reinterpret_cast<jobject>(arguments[0]);
|
|
jobject instance = reinterpret_cast<jobject>(arguments[1]);
|
|
jobjectArray args = reinterpret_cast<jobjectArray>(arguments[2]);
|
|
|
|
object vmMethod = arrayBody
|
|
(t, classMethodTable
|
|
(t, jclassVmClass(t, jmethodClazz(t, *method))),
|
|
jmethodSlot(t, *method));
|
|
|
|
if (methodFlags(t, vmMethod) & ACC_STATIC) {
|
|
instance = 0;
|
|
}
|
|
|
|
if (instance and not instanceOf(t, methodClass(t, vmMethod), *instance)) {
|
|
throwNew(t, Machine::IllegalArgumentExceptionType);
|
|
}
|
|
|
|
return reinterpret_cast<uint64_t>
|
|
(makeLocalReference
|
|
(t, invoke
|
|
(t, vmMethod, instance ? *instance : 0, args ? *args : 0)));
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jobject JNICALL
|
|
EXPORT(JVM_InvokeMethod)(Thread* t, jobject method, jobject instance,
|
|
jobjectArray args)
|
|
{
|
|
uintptr_t arguments[] = { reinterpret_cast<uintptr_t>(method),
|
|
reinterpret_cast<uintptr_t>(instance),
|
|
reinterpret_cast<uintptr_t>(args) };
|
|
|
|
return reinterpret_cast<jobject>(run(t, jvmInvokeMethod, arguments));
|
|
}
|
|
|
|
uint64_t
|
|
jvmNewInstanceFromConstructor(Thread* t, uintptr_t* arguments)
|
|
{
|
|
jobject constructor = reinterpret_cast<jobject>(arguments[0]);
|
|
jobjectArray args = reinterpret_cast<jobjectArray>(arguments[1]);
|
|
|
|
object instance = make
|
|
(t, jclassVmClass(t, jconstructorClazz(t, *constructor)));
|
|
PROTECT(t, instance);
|
|
|
|
object method = arrayBody
|
|
(t, classMethodTable
|
|
(t, jclassVmClass(t, jconstructorClazz(t, *constructor))),
|
|
jconstructorSlot(t, *constructor));
|
|
|
|
invoke(t, method, instance, args ? *args : 0);
|
|
|
|
return reinterpret_cast<uint64_t>(makeLocalReference(t, instance));
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jobject JNICALL
|
|
EXPORT(JVM_NewInstanceFromConstructor)(Thread* t, jobject constructor,
|
|
jobjectArray args)
|
|
{
|
|
uintptr_t arguments[] = { reinterpret_cast<uintptr_t>(constructor),
|
|
reinterpret_cast<uintptr_t>(args) };
|
|
|
|
return reinterpret_cast<jobject>
|
|
(run(t, jvmNewInstanceFromConstructor, arguments));
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jobject JNICALL
|
|
EXPORT(JVM_GetClassConstantPool)(Thread* t, jclass c)
|
|
{
|
|
ENTER(t, Thread::ActiveState);
|
|
|
|
object vmClass = jclassVmClass(t, *c);
|
|
object addendum = classAddendum(t, vmClass);
|
|
object pool;
|
|
if (addendum) {
|
|
pool = addendumPool(t, addendum);
|
|
} else {
|
|
pool = 0;
|
|
}
|
|
|
|
if (pool == 0) {
|
|
pool = classRuntimeDataPool(t, getClassRuntimeData(t, vmClass));
|
|
}
|
|
|
|
return makeLocalReference(t, makeConstantPool(t, pool));
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jint JNICALL
|
|
EXPORT(JVM_ConstantPoolGetSize)(Thread* t, jobject, jobject pool)
|
|
{
|
|
if (pool == 0) return 0;
|
|
|
|
ENTER(t, Thread::ActiveState);
|
|
|
|
return singletonCount(t, *pool);
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jclass JNICALL
|
|
EXPORT(JVM_ConstantPoolGetClassAt)(Thread*, jobject, jobject, jint)
|
|
{ abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT jclass JNICALL
|
|
EXPORT(JVM_ConstantPoolGetClassAtIfLoaded)(Thread*, jobject, jobject, jint)
|
|
{ abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT jobject JNICALL
|
|
EXPORT(JVM_ConstantPoolGetMethodAt)(Thread*, jobject, jobject, jint)
|
|
{ abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT jobject JNICALL
|
|
EXPORT(JVM_ConstantPoolGetMethodAtIfLoaded)(Thread*, jobject, jobject, jint)
|
|
{ abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT jobject JNICALL
|
|
EXPORT(JVM_ConstantPoolGetFieldAt)(Thread*, jobject, jobject, jint)
|
|
{ abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT jobject JNICALL
|
|
EXPORT(JVM_ConstantPoolGetFieldAtIfLoaded)(Thread*, jobject, jobject, jint)
|
|
{ abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT jobjectArray JNICALL
|
|
EXPORT(JVM_ConstantPoolGetMemberRefInfoAt)(Thread*, jobject, jobject, jint)
|
|
{ abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT jint JNICALL
|
|
EXPORT(JVM_ConstantPoolGetIntAt)(Thread* t, jobject, jobject pool, jint index)
|
|
{
|
|
ENTER(t, Thread::ActiveState);
|
|
|
|
return singletonValue(t, *pool, index - 1);
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jlong JNICALL
|
|
EXPORT(JVM_ConstantPoolGetLongAt)(Thread* t, jobject, jobject pool, jint index)
|
|
{
|
|
ENTER(t, Thread::ActiveState);
|
|
|
|
uint64_t v;
|
|
memcpy(&v, &singletonValue(t, *pool, index - 1), 8);
|
|
|
|
return v;
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jfloat JNICALL
|
|
EXPORT(JVM_ConstantPoolGetFloatAt)(Thread* t, jobject, jobject pool,
|
|
jint index)
|
|
{
|
|
ENTER(t, Thread::ActiveState);
|
|
|
|
return bitsToFloat(singletonValue(t, *pool, index - 1));
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jdouble JNICALL
|
|
EXPORT(JVM_ConstantPoolGetDoubleAt)(Thread* t, jobject, jobject pool,
|
|
jint index)
|
|
{
|
|
ENTER(t, Thread::ActiveState);
|
|
|
|
double v;
|
|
memcpy(&v, &singletonValue(t, *pool, index - 1), 8);
|
|
|
|
return v;
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jstring JNICALL
|
|
EXPORT(JVM_ConstantPoolGetStringAt)(Thread*, jobject, jobject, jint)
|
|
{ abort(); }
|
|
|
|
uint64_t
|
|
jvmConstantPoolGetUTF8At(Thread* t, uintptr_t* arguments)
|
|
{
|
|
jobject pool = reinterpret_cast<jobject>(arguments[0]);
|
|
jint index = arguments[1];
|
|
|
|
object array = parseUtf8(t, singletonObject(t, *pool, index - 1));
|
|
|
|
return reinterpret_cast<uint64_t>
|
|
(makeLocalReference
|
|
(t, t->m->classpath->makeString
|
|
(t, array, 0, fieldAtOffset<uintptr_t>(array, BytesPerWord) - 1)));
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jstring JNICALL
|
|
EXPORT(JVM_ConstantPoolGetUTF8At)(Thread* t, jobject, jobject pool, jint index)
|
|
{
|
|
uintptr_t arguments[] = { reinterpret_cast<uintptr_t>(pool),
|
|
static_cast<uintptr_t>(index) };
|
|
|
|
return reinterpret_cast<jstring>
|
|
(run(t, jvmConstantPoolGetUTF8At, arguments));
|
|
}
|
|
|
|
void
|
|
maybeWrap(Thread* t, bool wrapException)
|
|
{
|
|
if (t->exception
|
|
and wrapException
|
|
and not (instanceOf(t, type(t, Machine::ErrorType), t->exception)
|
|
or instanceOf
|
|
(t, type(t, Machine::RuntimeExceptionType), t->exception)))
|
|
{
|
|
object exception = t->exception;
|
|
t->exception = 0;
|
|
|
|
PROTECT(t, exception);
|
|
|
|
object paeClass = resolveClass
|
|
(t, root(t, Machine::BootLoader),
|
|
"java/security/PrivilegedActionException");
|
|
PROTECT(t, paeClass);
|
|
|
|
object paeConstructor = resolveMethod
|
|
(t, paeClass, "<init>", "(Ljava/lang/Exception;)V");
|
|
PROTECT(t, paeConstructor);
|
|
|
|
object result = make(t, paeClass);
|
|
PROTECT(t, result);
|
|
|
|
t->m->processor->invoke(t, paeConstructor, result, exception);
|
|
|
|
t->exception = result;
|
|
}
|
|
}
|
|
|
|
uint64_t
|
|
jvmDoPrivileged(Thread* t, uintptr_t* arguments)
|
|
{
|
|
jobject action = reinterpret_cast<jobject>(arguments[0]);
|
|
jboolean wrapException = arguments[1];
|
|
|
|
// todo: cache these class and method lookups in the t->m->classpath
|
|
// object:
|
|
|
|
object privilegedAction = resolveClass
|
|
(t, root(t, Machine::BootLoader), "java/security/PrivilegedAction");
|
|
|
|
object method;
|
|
if (instanceOf(t, privilegedAction, *action)) {
|
|
method = resolveMethod
|
|
(t, privilegedAction, "run", "()Ljava/lang/Object;");
|
|
} else {
|
|
object privilegedExceptionAction = resolveClass
|
|
(t, root(t, Machine::BootLoader),
|
|
"java/security/PrivilegedExceptionAction");
|
|
|
|
method = resolveMethod
|
|
(t, privilegedExceptionAction, "run", "()Ljava/lang/Object;");
|
|
}
|
|
|
|
THREAD_RESOURCE(t, jboolean, wrapException, maybeWrap(t, wrapException));
|
|
|
|
return reinterpret_cast<uint64_t>
|
|
(makeLocalReference(t, t->m->processor->invoke(t, method, *action)));
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jobject JNICALL
|
|
EXPORT(JVM_DoPrivileged)
|
|
(Thread* t, jclass, jobject action, jobject, jboolean wrapException)
|
|
{
|
|
uintptr_t arguments[] = { reinterpret_cast<uintptr_t>(action),
|
|
wrapException };
|
|
|
|
return reinterpret_cast<jobject>(run(t, jvmDoPrivileged, arguments));
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jobject JNICALL
|
|
EXPORT(JVM_GetInheritedAccessControlContext)(Thread*, jclass) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT jobject JNICALL
|
|
EXPORT(JVM_GetStackAccessControlContext)(Thread*, jclass)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT void* JNICALL
|
|
EXPORT(JVM_RegisterSignal)(jint, void*) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT jboolean JNICALL
|
|
EXPORT(JVM_RaiseSignal)(jint) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT jint JNICALL
|
|
EXPORT(JVM_FindSignal)(const char*)
|
|
{
|
|
return -1;
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jboolean JNICALL
|
|
EXPORT(JVM_DesiredAssertionStatus)(Thread*, jclass, jclass)
|
|
{
|
|
return false;
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jobject JNICALL
|
|
EXPORT(JVM_AssertionStatusDirectives)(Thread*, jclass) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT jboolean JNICALL
|
|
EXPORT(JVM_SupportsCX8)()
|
|
{
|
|
return true;
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT const char* JNICALL
|
|
EXPORT(JVM_GetClassNameUTF)(Thread*, jclass) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
|
EXPORT(JVM_GetClassCPTypes)(Thread*, jclass, unsigned char*) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT jint JNICALL
|
|
EXPORT(JVM_GetClassCPEntriesCount)(Thread*, jclass) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT jint JNICALL
|
|
EXPORT(JVM_GetClassFieldsCount)(Thread*, jclass) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT jint JNICALL
|
|
EXPORT(JVM_GetClassMethodsCount)(Thread*, jclass) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
|
EXPORT(JVM_GetMethodIxExceptionIndexes)(Thread*, jclass, jint,
|
|
unsigned short*) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT jint JNICALL
|
|
EXPORT(JVM_GetMethodIxExceptionsCount)(Thread*, jclass, jint) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
|
EXPORT(JVM_GetMethodIxByteCode)(Thread*, jclass, jint,
|
|
unsigned char*) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT jint JNICALL
|
|
EXPORT(JVM_GetMethodIxByteCodeLength)(Thread*, jclass, jint) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
|
EXPORT(JVM_GetMethodIxExceptionTableEntry)(Thread*, jclass, jint,
|
|
jint,
|
|
local::JVM_ExceptionTableEntryType*)
|
|
{ abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT jint JNICALL
|
|
EXPORT(JVM_GetMethodIxExceptionTableLength)(Thread*, jclass, int) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT jint JNICALL
|
|
EXPORT(JVM_GetFieldIxModifiers)(Thread*, jclass, int) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT jint JNICALL
|
|
EXPORT(JVM_GetMethodIxModifiers)(Thread*, jclass, int) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT jint JNICALL
|
|
EXPORT(JVM_GetMethodIxLocalsCount)(Thread*, jclass, int) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT jint JNICALL
|
|
EXPORT(JVM_GetMethodIxArgsSize)(Thread*, jclass, int) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT jint JNICALL
|
|
EXPORT(JVM_GetMethodIxMaxStack)(Thread*, jclass, int) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT jboolean JNICALL
|
|
EXPORT(JVM_IsConstructorIx)(Thread*, jclass, int) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT jboolean JNICALL
|
|
EXPORT(JVM_IsVMGeneratedMethodIx)(Thread*, jclass, int) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT const char* JNICALL
|
|
EXPORT(JVM_GetMethodIxNameUTF)(Thread*, jclass, jint) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT const char* JNICALL
|
|
EXPORT(JVM_GetMethodIxSignatureUTF)(Thread*, jclass, jint) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT const char* JNICALL
|
|
EXPORT(JVM_GetCPFieldNameUTF)(Thread*, jclass, jint) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT const char* JNICALL
|
|
EXPORT(JVM_GetCPMethodNameUTF)(Thread*, jclass, jint) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT const char* JNICALL
|
|
EXPORT(JVM_GetCPMethodSignatureUTF)(Thread*, jclass, jint) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT const char* JNICALL
|
|
EXPORT(JVM_GetCPFieldSignatureUTF)(Thread*, jclass, jint) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT const char* JNICALL
|
|
EXPORT(JVM_GetCPClassNameUTF)(Thread*, jclass, jint) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT const char* JNICALL
|
|
EXPORT(JVM_GetCPFieldClassNameUTF)(Thread*, jclass, jint) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT const char* JNICALL
|
|
EXPORT(JVM_GetCPMethodClassNameUTF)(Thread*, jclass, jint) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT jint JNICALL
|
|
EXPORT(JVM_GetCPFieldModifiers)(Thread*, jclass, int, jclass) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT jint JNICALL
|
|
EXPORT(JVM_GetCPMethodModifiers)(Thread*, jclass, int, jclass) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
|
EXPORT(JVM_ReleaseUTF)(const char*) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT jboolean JNICALL
|
|
EXPORT(JVM_IsSameClassPackage)(Thread*, jclass, jclass) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT jint JNICALL
|
|
EXPORT(JVM_GetLastErrorString)(char* dst, int length)
|
|
{
|
|
strncpy(dst, strerror(errno), length);
|
|
return strlen(dst);
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT char* JNICALL
|
|
EXPORT(JVM_NativePath)(char* path)
|
|
{
|
|
return path;
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jint JNICALL
|
|
EXPORT(JVM_Open)(const char* path, jint flags, jint mode)
|
|
{
|
|
int r = OPEN(path, flags & 0xFFFF, mode);
|
|
if (r == -1) {
|
|
return errno == EEXIST ? JVM_EEXIST : -1;
|
|
} else {
|
|
return r;
|
|
}
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jint JNICALL
|
|
EXPORT(JVM_Close)(jint fd)
|
|
{
|
|
return CLOSE(fd);
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jint JNICALL
|
|
EXPORT(JVM_Read)(jint fd, char* dst, jint length)
|
|
{
|
|
return READ(fd, dst, length);
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jint JNICALL
|
|
EXPORT(JVM_Write)(jint fd, char* src, jint length)
|
|
{
|
|
return WRITE(fd, src, length);
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jint JNICALL
|
|
EXPORT(JVM_Available)(jint fd, jlong* result)
|
|
{
|
|
struct STAT buffer;
|
|
int n;
|
|
if (FSTAT(fd, &buffer) >= 0
|
|
and (S_ISCHR(buffer.st_mode)
|
|
or S_ISFIFO(buffer.st_mode)
|
|
or S_ISSOCK(buffer.st_mode))
|
|
and local::pipeAvailable(fd, &n))
|
|
{
|
|
*result = n;
|
|
return 1;
|
|
}
|
|
|
|
int current = LSEEK(fd, 0, SEEK_CUR);
|
|
if (current == -1) return 0;
|
|
|
|
int end = LSEEK(fd, 0, SEEK_END);
|
|
if (end == -1) return 0;
|
|
|
|
if (LSEEK(fd, current, SEEK_SET) == -1) return 0;
|
|
|
|
*result = end - current;
|
|
return 1;
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jlong JNICALL
|
|
EXPORT(JVM_Lseek)(jint fd, jlong offset, jint seek)
|
|
{
|
|
return LSEEK(fd, offset, seek);
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jint JNICALL
|
|
EXPORT(JVM_SetLength)(jint fd, jlong length)
|
|
{
|
|
#ifdef PLATFORM_WINDOWS
|
|
HANDLE h = reinterpret_cast<HANDLE>(_get_osfhandle(fd));
|
|
if (h == INVALID_HANDLE_VALUE) {
|
|
errno = EBADF;
|
|
return -1;
|
|
}
|
|
|
|
long high = length >> 32;
|
|
DWORD r = SetFilePointer(h, static_cast<long>(length), &high, FILE_BEGIN);
|
|
if (r == 0xFFFFFFFF and GetLastError() != NO_ERROR) {
|
|
errno = EIO;
|
|
return -1;
|
|
}
|
|
|
|
if (SetEndOfFile(h)) {
|
|
return 0;
|
|
} else {
|
|
errno = EIO;
|
|
return -1;
|
|
}
|
|
#else
|
|
return ftruncate(fd, length);
|
|
#endif
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jint JNICALL
|
|
EXPORT(JVM_Sync)(jint fd)
|
|
{
|
|
#ifdef PLATFORM_WINDOWS
|
|
HANDLE h = reinterpret_cast<HANDLE>(_get_osfhandle(fd));
|
|
if (h == INVALID_HANDLE_VALUE) {
|
|
errno = EBADF;
|
|
return -1;
|
|
}
|
|
|
|
if (FlushFileBuffers(h)) {
|
|
return 0;
|
|
} else {
|
|
errno = EIO;
|
|
return -1;
|
|
}
|
|
#else
|
|
return fsync(fd);
|
|
#endif
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jint JNICALL
|
|
EXPORT(JVM_InitializeSocketLibrary)()
|
|
{
|
|
#ifdef PLATFORM_WINDOWS
|
|
static bool wsaInitialized = false;
|
|
if (not wsaInitialized) {
|
|
WSADATA data;
|
|
int r = WSAStartup(MAKEWORD(2, 2), &data);
|
|
if (r or LOBYTE(data.wVersion) != 2 or HIBYTE(data.wVersion) != 2) {
|
|
return -1;
|
|
} else {
|
|
wsaInitialized = true;
|
|
}
|
|
}
|
|
#endif
|
|
return 0;
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jint JNICALL
|
|
EXPORT(JVM_Socket)(jint domain, jint type, jint protocol)
|
|
{
|
|
return socket(domain, type, protocol);
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jint JNICALL
|
|
EXPORT(JVM_SocketClose)(jint socket)
|
|
{
|
|
#ifdef PLATFORM_WINDOWS
|
|
return closesocket(socket);
|
|
#else
|
|
return close(socket);
|
|
#endif
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jint JNICALL
|
|
EXPORT(JVM_SocketShutdown)(jint socket, jint how)
|
|
{
|
|
return shutdown(socket, how);
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jint JNICALL
|
|
EXPORT(JVM_Recv)(jint socket, char* dst, jint count, jint flags)
|
|
{
|
|
return recv(socket, dst, count, flags);
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jint JNICALL
|
|
EXPORT(JVM_Send)(jint socket, char* src, jint count, jint flags)
|
|
{
|
|
return send(socket, src, count, flags);
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jint JNICALL
|
|
EXPORT(JVM_Timeout)(int, long) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT jint JNICALL
|
|
EXPORT(JVM_Listen)(jint socket, jint count)
|
|
{
|
|
return listen(socket, count);
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jint JNICALL
|
|
EXPORT(JVM_Connect)(jint socket, sockaddr* address, jint addressLength)
|
|
{
|
|
return connect(socket, address, addressLength);
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jint JNICALL
|
|
EXPORT(JVM_Bind)(jint, struct sockaddr*, jint) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT jint JNICALL
|
|
EXPORT(JVM_Accept)(jint socket, struct sockaddr* address, jint* addressLength)
|
|
{
|
|
socklen_t length = *addressLength;
|
|
int r = accept(socket, address, &length);
|
|
*addressLength = length;
|
|
return r;
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jint JNICALL
|
|
EXPORT(JVM_RecvFrom)(jint, char*, int,
|
|
int, struct sockaddr*, int*) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT jint JNICALL
|
|
EXPORT(JVM_SendTo)(jint, char*, int,
|
|
int, struct sockaddr*, int) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT jint JNICALL
|
|
EXPORT(JVM_SocketAvailable)(jint socket, jint* count)
|
|
{
|
|
#ifdef PLATFORM_WINDOWS
|
|
unsigned long c = *count;
|
|
int r = ioctlsocket(socket, FIONREAD, &c);
|
|
*count = c;
|
|
return r;
|
|
#else
|
|
return ioctl(socket, FIONREAD, count) < 0 ? 0 : 1;
|
|
#endif
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jint JNICALL
|
|
EXPORT(JVM_GetSockName)(jint socket, struct sockaddr* address,
|
|
int* addressLength)
|
|
{
|
|
socklen_t length = *addressLength;
|
|
int r = getsockname(socket, address, &length);
|
|
*addressLength = length;
|
|
return r;
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jint JNICALL
|
|
EXPORT(JVM_GetSockOpt)(jint socket, int level, int optionName,
|
|
char* optionValue, int* optionLength)
|
|
{
|
|
socklen_t length = *optionLength;
|
|
int rv = getsockopt(socket, level, optionName, optionValue, &length);
|
|
*optionLength = length;
|
|
return rv;
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jint JNICALL
|
|
EXPORT(JVM_SetSockOpt)(jint socket, int level, int optionName,
|
|
const char* optionValue, int optionLength)
|
|
{
|
|
return setsockopt(socket, level, optionName, optionValue, optionLength);
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT struct protoent* JNICALL
|
|
EXPORT(JVM_GetProtoByName)(char*) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT struct hostent* JNICALL
|
|
EXPORT(JVM_GetHostByAddr)(const char*, int, int) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT struct hostent* JNICALL
|
|
EXPORT(JVM_GetHostByName)(char*) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT int JNICALL
|
|
EXPORT(JVM_GetHostName)(char* name, int length)
|
|
{
|
|
return gethostname(name, length);
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT void* JNICALL
|
|
EXPORT(JVM_RawMonitorCreate)(void)
|
|
{
|
|
System* s = local::globalMachine->system;
|
|
System::Monitor* lock;
|
|
if (s->success(s->make(&lock))) {
|
|
return lock;
|
|
} else {
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
|
EXPORT(JVM_RawMonitorDestroy)(void* lock)
|
|
{
|
|
static_cast<System::Monitor*>(lock)->dispose();
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jint JNICALL
|
|
EXPORT(JVM_RawMonitorEnter)(void* lock)
|
|
{
|
|
static_cast<System::Monitor*>(lock)->acquire
|
|
(static_cast<Thread*>
|
|
(local::globalMachine->localThread->get())->systemThread);
|
|
|
|
return 0;
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
|
EXPORT(JVM_RawMonitorExit)(void* lock)
|
|
{
|
|
static_cast<System::Monitor*>(lock)->release
|
|
(static_cast<Thread*>
|
|
(local::globalMachine->localThread->get())->systemThread);
|
|
}
|
|
|
|
int JNICALL
|
|
GetVersion(Thread*)
|
|
{
|
|
return JMM_VERSION_1_0;
|
|
}
|
|
|
|
uint64_t
|
|
getInputArgumentArray(Thread* t, uintptr_t*)
|
|
{
|
|
object array = makeObjectArray
|
|
(t, type(t, Machine::StringType), t->m->argumentCount);
|
|
PROTECT(t, array);
|
|
|
|
for (unsigned i = 0; i < t->m->argumentCount; ++i) {
|
|
object argument = makeString(t, t->m->arguments[i]);
|
|
set(t, array, ArrayBody + (i * BytesPerWord), argument);
|
|
}
|
|
|
|
return reinterpret_cast<uintptr_t>(makeLocalReference(t, array));
|
|
}
|
|
|
|
jobjectArray JNICALL
|
|
GetInputArgumentArray(Thread* t)
|
|
{
|
|
return reinterpret_cast<jobjectArray>(run(t, getInputArgumentArray, 0));
|
|
}
|
|
|
|
jint JNICALL
|
|
GetOptionalSupport(Thread*, jmmOptionalSupport* support)
|
|
{
|
|
memset(support, 0, sizeof(jmmOptionalSupport));
|
|
return 0;
|
|
}
|
|
|
|
jlong JNICALL
|
|
GetLongAttribute(Thread* t, jobject, jmmLongAttribute attribute)
|
|
{
|
|
const unsigned JMM_JVM_INIT_DONE_TIME_MS = 7;
|
|
|
|
switch (attribute) {
|
|
case JMM_JVM_INIT_DONE_TIME_MS:
|
|
return 0;
|
|
|
|
default:
|
|
abort(t);
|
|
}
|
|
}
|
|
|
|
jboolean JNICALL
|
|
GetBoolAttribute(Thread* t, jmmBoolAttribute attribute)
|
|
{
|
|
const unsigned JMM_THREAD_CPU_TIME = 24;
|
|
const unsigned JMM_THREAD_ALLOCATED_MEMORY = 25;
|
|
|
|
switch (attribute) {
|
|
case JMM_THREAD_CPU_TIME:
|
|
case JMM_THREAD_ALLOCATED_MEMORY:
|
|
return false;
|
|
|
|
default:
|
|
abort(t);
|
|
}
|
|
}
|
|
|
|
uint64_t
|
|
getMemoryManagers(Thread* t, uintptr_t*)
|
|
{
|
|
return reinterpret_cast<uintptr_t>
|
|
(makeLocalReference
|
|
(t, makeObjectArray
|
|
(t, resolveClass
|
|
(t, root(t, Machine::BootLoader),
|
|
"java/lang/management/MemoryManagerMXBean"), 0)));
|
|
}
|
|
|
|
jobjectArray JNICALL
|
|
GetMemoryManagers(Thread* t, jobject)
|
|
{
|
|
return reinterpret_cast<jobjectArray>(run(t, getMemoryManagers, 0));
|
|
}
|
|
|
|
uint64_t
|
|
getMemoryPools(Thread* t, uintptr_t*)
|
|
{
|
|
return reinterpret_cast<uintptr_t>
|
|
(makeLocalReference
|
|
(t, makeObjectArray
|
|
(t, resolveClass
|
|
(t, root(t, Machine::BootLoader),
|
|
"java/lang/management/MemoryPoolMXBean"), 0)));
|
|
}
|
|
|
|
jobjectArray JNICALL
|
|
GetMemoryPools(Thread* t, jobject)
|
|
{
|
|
return reinterpret_cast<jobjectArray>(run(t, getMemoryPools, 0));
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT void* JNICALL
|
|
EXPORT(JVM_GetManagement)(jint version)
|
|
{
|
|
if (version == JMM_VERSION_1_0) {
|
|
JmmInterface* interface
|
|
= &(static_cast<MyClasspath*>
|
|
(local::globalMachine->classpath)->jmmInterface);
|
|
|
|
memset(interface, 0, sizeof(JmmInterface));
|
|
|
|
interface->GetVersion = GetVersion;
|
|
interface->GetOptionalSupport = GetOptionalSupport;
|
|
interface->GetLongAttribute = GetLongAttribute;
|
|
interface->GetBoolAttribute = GetBoolAttribute;
|
|
interface->GetMemoryManagers = GetMemoryManagers;
|
|
interface->GetMemoryPools = GetMemoryPools;
|
|
interface->GetInputArgumentArray = GetInputArgumentArray;
|
|
|
|
return interface;
|
|
} else {
|
|
return 0;
|
|
}
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jobject JNICALL
|
|
EXPORT(JVM_InitAgentProperties)(Thread*, jobject) { abort(); }
|
|
|
|
uint64_t
|
|
getEnclosingMethodInfo(Thread* t, uintptr_t* arguments)
|
|
{
|
|
jclass c = reinterpret_cast<jclass>(arguments[0]);
|
|
object class_ = jclassVmClass(t, *c);
|
|
PROTECT(t, class_);
|
|
|
|
object addendum = classAddendum(t, class_);
|
|
if (addendum) {
|
|
object enclosingClass = classAddendumEnclosingClass(t, addendum);
|
|
if (enclosingClass) {
|
|
PROTECT(t, enclosingClass);
|
|
|
|
object array = makeObjectArray(t, type(t, Machine::JobjectType), 3);
|
|
PROTECT(t, array);
|
|
|
|
enclosingClass = getJClass
|
|
(t, resolveClass(t, classLoader(t, class_), enclosingClass));
|
|
|
|
set(t, array, ArrayBody, enclosingClass);
|
|
|
|
object enclosingMethod = classAddendumEnclosingMethod(t, addendum);
|
|
|
|
if (enclosingMethod) {
|
|
PROTECT(t, enclosingMethod);
|
|
|
|
object name = t->m->classpath->makeString
|
|
(t, pairFirst(t, enclosingMethod), 0,
|
|
byteArrayLength(t, pairFirst(t, enclosingMethod)) - 1);
|
|
|
|
set(t, array, ArrayBody + BytesPerWord, name);
|
|
|
|
object spec = t->m->classpath->makeString
|
|
(t, pairSecond(t, enclosingMethod), 0,
|
|
byteArrayLength(t, pairSecond(t, enclosingMethod)) - 1);
|
|
|
|
set(t, array, ArrayBody + (2 * BytesPerWord), spec);
|
|
}
|
|
|
|
return reinterpret_cast<uintptr_t>(makeLocalReference(t, array));
|
|
}
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
|
|
extern "C" AVIAN_EXPORT jobjectArray JNICALL
|
|
EXPORT(JVM_GetEnclosingMethodInfo)(Thread* t, jclass c)
|
|
{
|
|
uintptr_t arguments[] = { reinterpret_cast<uintptr_t>(c) };
|
|
|
|
return reinterpret_cast<jobjectArray>
|
|
(run(t, getEnclosingMethodInfo, arguments));
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jintArray JNICALL
|
|
EXPORT(JVM_GetThreadStateValues)(JNIEnv*, jint) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT jobjectArray JNICALL
|
|
EXPORT(JVM_GetThreadStateNames)(JNIEnv*, jint, jintArray) { abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
|
EXPORT(JVM_GetVersionInfo)(JNIEnv*, local::jvm_version_info* info, size_t size)
|
|
{
|
|
memset(info, 0, size);
|
|
info->jvm_version = 0x01070000;
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT jboolean JNICALL
|
|
EXPORT(JVM_CX8Field)(JNIEnv*, jobject*, jfieldID*, jlong, jlong)
|
|
{ abort(); }
|
|
|
|
extern "C" AVIAN_EXPORT void JNICALL
|
|
EXPORT(JVM_SetNativeThreadName)(JNIEnv*, jobject, jstring) { abort(); }
|
|
|
|
} // namespace local
|
|
|
|
} // namespace
|
|
|
|
extern "C" AVIAN_EXPORT int
|
|
jio_vsnprintf(char* dst, size_t size, const char* format, va_list a)
|
|
{
|
|
return vm::vsnprintf(dst, size, format, a);
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT int
|
|
jio_vfprintf(FILE* stream, const char* format, va_list a)
|
|
{
|
|
return vfprintf(stream, format, a);
|
|
}
|
|
|
|
#ifdef PLATFORM_WINDOWS
|
|
extern "C" AVIAN_EXPORT void* JNICALL
|
|
EXPORT(JVM_GetThreadInterruptEvent)()
|
|
{
|
|
// hack: We don't want to expose thread interruption implementation
|
|
// details, so we give the class library a fake event to play with.
|
|
// This means that threads won't be interruptable when blocked in
|
|
// Process.waitFor.
|
|
static HANDLE fake = 0;
|
|
if (fake == 0) {
|
|
fake = CreateEvent(0, true, false, 0);
|
|
}
|
|
return fake;
|
|
}
|
|
|
|
namespace { HMODULE jvmHandle = 0; }
|
|
|
|
extern "C" int JDK_InitJvmHandle()
|
|
{
|
|
jvmHandle = GetModuleHandle(0);
|
|
return jvmHandle != 0;
|
|
}
|
|
|
|
extern "C" void* JDK_FindJvmEntry(const char* name)
|
|
{
|
|
return voidPointer(GetProcAddress(jvmHandle, name));
|
|
}
|
|
|
|
# ifdef AVIAN_OPENJDK_SRC
|
|
|
|
extern "C" char* findJavaTZ_md(const char*, const char*);
|
|
|
|
extern "C" AVIAN_EXPORT int64_t JNICALL
|
|
Avian_java_util_TimeZone_getSystemTimeZoneID
|
|
(Thread* t, object, uintptr_t* arguments)
|
|
{
|
|
// On Windows, findJavaTZ_md loads tzmappings from the filesystem
|
|
// using fopen, so we have no opportunity to make it read straight
|
|
// from the embedded JAR file as with files read from Java code.
|
|
// Therefore, we must extract tzmappings to a temporary location
|
|
// before calling findJavaTZ_md. We could avoid this by
|
|
// implementing findJavaTZ_md ourselves from scratch, but that would
|
|
// be a lot of code to implement and maintain.
|
|
|
|
object country = reinterpret_cast<object>(arguments[1]);
|
|
|
|
THREAD_RUNTIME_ARRAY(t, char, countryChars, stringLength(t, country) + 1);
|
|
stringChars(t, country, RUNTIME_ARRAY_BODY(countryChars));
|
|
|
|
local::MyClasspath* cp = static_cast<local::MyClasspath*>(t->m->classpath);
|
|
|
|
local::EmbeddedFile ef(cp, cp->tzMappings, cp->tzMappingsLength);
|
|
if (ef.jar == 0 or ef.jarLength == 0 or ef.pathLength == 0) {
|
|
return 0;
|
|
}
|
|
|
|
Finder* finder = getFinder(t, ef.jar, ef.jarLength);
|
|
if (finder == 0) {
|
|
return 0;
|
|
}
|
|
|
|
System::Region* r = finder->find(ef.path);
|
|
if (r == 0) {
|
|
return 0;
|
|
}
|
|
|
|
THREAD_RESOURCE(t, System::Region*, r, r->dispose());
|
|
|
|
char tmpPath[MAX_PATH + 1];
|
|
GetTempPathA(MAX_PATH, tmpPath);
|
|
|
|
char tmpDir[MAX_PATH + 1];
|
|
vm::snprintf(tmpDir, MAX_PATH, "%s/avian-tmp", tmpPath);
|
|
if (_mkdir(tmpDir) != 0 and errno != EEXIST) {
|
|
return 0;
|
|
}
|
|
|
|
THREAD_RESOURCE(t, char*, tmpDir, rmdir(tmpDir));
|
|
|
|
char libDir[MAX_PATH + 1];
|
|
vm::snprintf(libDir, MAX_PATH, "%s/lib", tmpDir);
|
|
if (mkdir(libDir) != 0 and errno != EEXIST) {
|
|
return 0;
|
|
}
|
|
|
|
THREAD_RESOURCE(t, char*, libDir, rmdir(libDir));
|
|
|
|
char file[MAX_PATH + 1];
|
|
vm::snprintf(file, MAX_PATH, "%s/tzmappings", libDir);
|
|
FILE* out = vm::fopen(file, "wb");
|
|
if (out == 0) {
|
|
return 0;
|
|
}
|
|
|
|
THREAD_RESOURCE(t, char*, file, unlink(file));
|
|
THREAD_RESOURCE(t, FILE*, out, fclose(out));
|
|
|
|
if (fwrite(r->start(), 1, r->length(), out) != r->length()
|
|
or fflush(out) != 0)
|
|
{
|
|
return 0;
|
|
}
|
|
|
|
char* javaTZ = findJavaTZ_md(tmpDir, RUNTIME_ARRAY_BODY(countryChars));
|
|
if (javaTZ) {
|
|
THREAD_RESOURCE(t, char*, javaTZ, free(javaTZ));
|
|
|
|
return reinterpret_cast<int64_t>(makeString(t, "%s", javaTZ));
|
|
} else {
|
|
return 0;
|
|
}
|
|
}
|
|
# else // not AVIAN_OPENJDK_SRC
|
|
extern "C" AVIAN_EXPORT int
|
|
jio_snprintf(char* dst, size_t size, const char* format, ...)
|
|
{
|
|
va_list a;
|
|
va_start(a, format);
|
|
|
|
int r = jio_vsnprintf(dst, size, format, a);
|
|
|
|
va_end(a);
|
|
|
|
return r;
|
|
}
|
|
|
|
extern "C" AVIAN_EXPORT int
|
|
jio_fprintf(FILE* stream, const char* format, ...)
|
|
{
|
|
va_list a;
|
|
va_start(a, format);
|
|
|
|
int r = jio_vfprintf(stream, format, a);
|
|
|
|
va_end(a);
|
|
|
|
return r;
|
|
}
|
|
# endif // not AVIAN_OPENJDK_SRC
|
|
#endif // PLATFORM_WINDOWS
|