corda/src/avian/machine.h

3859 lines
102 KiB
C++

/* Copyright (c) 2008-2015, 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. */
#ifndef MACHINE_H
#define MACHINE_H
#include "avian/common.h"
#include "java-common.h"
#include <avian/system/system.h>
#include <avian/system/signal.h>
#include <avian/heap/heap.h>
#include <avian/util/hash.h>
#include "avian/finder.h"
#include "avian/processor.h"
#include "avian/constants.h"
#include "avian/arch.h"
using namespace avian::util;
#ifdef PLATFORM_WINDOWS
#define JNICALL __stdcall
#else
#define JNICALL
#endif
#define PROTECT(thread, name) \
Thread::SingleProtector MAKE_NAME(protector_)(thread, &name);
#define ACQUIRE(t, x) MonitorResource MAKE_NAME(monitorResource_)(t, x)
#define ACQUIRE_OBJECT(t, x) \
ObjectMonitorResource MAKE_NAME(monitorResource_)(t, x)
#define ACQUIRE_FIELD_FOR_READ(t, field) \
FieldReadResource MAKE_NAME(monitorResource_)(t, field)
#define ACQUIRE_FIELD_FOR_WRITE(t, field) \
FieldWriteResource MAKE_NAME(monitorResource_)(t, field)
#define ACQUIRE_RAW(t, x) RawMonitorResource MAKE_NAME(monitorResource_)(t, x)
#define ENTER(t, state) StateResource MAKE_NAME(stateResource_)(t, state)
#define THREAD_RESOURCE0(t, releaseBody) \
class MAKE_NAME(Resource_) : public Thread::AutoResource { \
public: \
MAKE_NAME(Resource_)(Thread * t) : AutoResource(t) \
{ \
} \
~MAKE_NAME(Resource_)() \
{ \
releaseBody; \
} \
virtual void release() \
{ \
this->MAKE_NAME(Resource_)::~MAKE_NAME(Resource_)(); \
} \
} MAKE_NAME(resource_)(t);
#define OBJECT_RESOURCE(t, name, releaseBody) \
class MAKE_NAME(Resource_) : public Thread::AutoResource { \
public: \
MAKE_NAME(Resource_)(Thread * t, object name) \
: AutoResource(t), name(name), protector(t, &(this->name)) \
{ \
} \
~MAKE_NAME(Resource_)() \
{ \
releaseBody; \
} \
virtual void release() \
{ \
this->MAKE_NAME(Resource_)::~MAKE_NAME(Resource_)(); \
} \
\
private: \
object name; \
Thread::SingleProtector protector; \
} MAKE_NAME(resource_)(t, name);
#define THREAD_RESOURCE(t, type, name, releaseBody) \
class MAKE_NAME(Resource_) : public Thread::AutoResource { \
public: \
MAKE_NAME(Resource_)(Thread * t, type name) : AutoResource(t), name(name) \
{ \
} \
~MAKE_NAME(Resource_)() \
{ \
releaseBody; \
} \
virtual void release() \
{ \
this->MAKE_NAME(Resource_)::~MAKE_NAME(Resource_)(); \
} \
\
private: \
type name; \
} MAKE_NAME(resource_)(t, name);
#define THREAD_RESOURCE2(t, type1, name1, type2, name2, releaseBody) \
class MAKE_NAME(Resource_) : public Thread::AutoResource { \
public: \
MAKE_NAME(Resource_)(Thread * t, type1 name1, type2 name2) \
: AutoResource(t), name1(name1), name2(name2) \
{ \
} \
~MAKE_NAME(Resource_)() \
{ \
releaseBody; \
} \
virtual void release() \
{ \
this->MAKE_NAME(Resource_)::~MAKE_NAME(Resource_)(); \
} \
\
private: \
type1 name1; \
type2 name2; \
} MAKE_NAME(resource_)(t, name1, name2);
namespace vm {
const bool Verbose = false;
const bool DebugRun = false;
const bool DebugStack = false;
const bool DebugMonitors = false;
const bool DebugReferences = false;
const bool AbortOnOutOfMemoryError = false;
const uintptr_t HashTakenMark = 1;
const uintptr_t ExtendedMark = 2;
const uintptr_t FixedMark = 3;
const unsigned ThreadHeapSizeInBytes = 64 * 1024;
const unsigned ThreadHeapSizeInWords = ThreadHeapSizeInBytes / BytesPerWord;
const unsigned ThreadBackupHeapSizeInBytes = 2 * 1024;
const unsigned ThreadBackupHeapSizeInWords = ThreadBackupHeapSizeInBytes
/ BytesPerWord;
const unsigned ThreadHeapPoolSize = 64;
const unsigned FixedFootprintThresholdInBytes = ThreadHeapPoolSize
* ThreadHeapSizeInBytes;
// number of zombie threads which may accumulate before we force a GC
// to clean them up:
const unsigned ZombieCollectionThreshold = 16;
enum FieldCode {
VoidField,
ByteField,
CharField,
DoubleField,
FloatField,
IntField,
LongField,
ShortField,
BooleanField,
ObjectField
};
enum StackTag {
IntTag, // must be zero
ObjectTag
};
const int NativeLine = -2;
const int UnknownLine = -1;
// class vmFlags:
const unsigned ReferenceFlag = 1 << 0;
const unsigned WeakReferenceFlag = 1 << 1;
const unsigned NeedInitFlag = 1 << 2;
const unsigned InitFlag = 1 << 3;
const unsigned InitErrorFlag = 1 << 4;
const unsigned PrimitiveFlag = 1 << 5;
const unsigned BootstrapFlag = 1 << 6;
const unsigned HasFinalizerFlag = 1 << 7;
const unsigned LinkFlag = 1 << 8;
const unsigned HasFinalMemberFlag = 1 << 9;
const unsigned SingletonFlag = 1 << 10;
const unsigned ContinuationFlag = 1 << 11;
// method vmFlags:
const unsigned ClassInitFlag = 1 << 0;
const unsigned ConstructorFlag = 1 << 1;
#ifndef JNI_VERSION_1_6
#define JNI_VERSION_1_6 0x00010006
#endif
#ifndef JNI_TRUE
#define JNI_TRUE 1
#endif
#ifndef JNI_OK
#define JNI_OK 0
#endif
typedef Machine JavaVM;
typedef Thread JNIEnv;
struct JNINativeMethod {
char* name;
char* signature;
void* function;
};
struct JavaVMOption {
char* optionString;
void* extraInfo;
};
struct JavaVMInitArgs {
jint version;
jint nOptions;
JavaVMOption* options;
jboolean ignoreUnrecognized;
};
struct JavaVMVTable {
void* reserved0;
void* reserved1;
void* reserved2;
jint(JNICALL* DestroyJavaVM)(JavaVM*);
jint(JNICALL* AttachCurrentThread)(JavaVM*, JNIEnv**, void*);
jint(JNICALL* DetachCurrentThread)(JavaVM*);
jint(JNICALL* GetEnv)(JavaVM*, JNIEnv**, jint);
jint(JNICALL* AttachCurrentThreadAsDaemon)(JavaVM*, JNIEnv**, void*);
};
struct JNIEnvVTable {
void* reserved0;
void* reserved1;
void* reserved2;
void* reserved3;
jint(JNICALL* GetVersion)(JNIEnv*);
jclass(
JNICALL* DefineClass)(JNIEnv*, const char*, jobject, const jbyte*, jsize);
jclass(JNICALL* FindClass)(JNIEnv*, const char*);
jmethodID(JNICALL* FromReflectedMethod)(JNIEnv*, jobject);
jfieldID(JNICALL* FromReflectedField)(JNIEnv*, jobject);
jobject(JNICALL* ToReflectedMethod)(JNIEnv*, jclass, jmethodID, jboolean);
jclass(JNICALL* GetSuperclass)(JNIEnv*, jclass);
jboolean(JNICALL* IsAssignableFrom)(JNIEnv*, jclass, jclass);
jobject(JNICALL* ToReflectedField)(JNIEnv*, jclass, jfieldID, jboolean);
jint(JNICALL* Throw)(JNIEnv*, jthrowable);
jint(JNICALL* ThrowNew)(JNIEnv*, jclass, const char*);
jthrowable(JNICALL* ExceptionOccurred)(JNIEnv*);
void(JNICALL* ExceptionDescribe)(JNIEnv*);
void(JNICALL* ExceptionClear)(JNIEnv*);
void(JNICALL* FatalError)(JNIEnv*, const char*);
jint(JNICALL* PushLocalFrame)(JNIEnv*, jint);
jobject(JNICALL* PopLocalFrame)(JNIEnv*, jobject);
jobject(JNICALL* NewGlobalRef)(JNIEnv*, jobject);
void(JNICALL* DeleteGlobalRef)(JNIEnv*, jobject);
void(JNICALL* DeleteLocalRef)(JNIEnv*, jobject);
jboolean(JNICALL* IsSameObject)(JNIEnv*, jobject, jobject);
jobject(JNICALL* NewLocalRef)(JNIEnv*, jobject);
jint(JNICALL* EnsureLocalCapacity)(JNIEnv*, jint);
jobject(JNICALL* AllocObject)(JNIEnv*, jclass);
jobject(JNICALL* NewObject)(JNIEnv*, jclass, jmethodID, ...);
jobject(JNICALL* NewObjectV)(JNIEnv*, jclass, jmethodID, va_list);
jobject(JNICALL* NewObjectA)(JNIEnv*, jclass, jmethodID, const jvalue*);
jclass(JNICALL* GetObjectClass)(JNIEnv*, jobject);
jboolean(JNICALL* IsInstanceOf)(JNIEnv*, jobject, jclass);
jmethodID(JNICALL* GetMethodID)(JNIEnv*, jclass, const char*, const char*);
jobject(JNICALL* CallObjectMethod)(JNIEnv*, jobject, jmethodID, ...);
jobject(JNICALL* CallObjectMethodV)(JNIEnv*, jobject, jmethodID, va_list);
jobject(JNICALL* CallObjectMethodA)(JNIEnv*,
jobject,
jmethodID,
const jvalue*);
jboolean(JNICALL* CallBooleanMethod)(JNIEnv*, jobject, jmethodID, ...);
jboolean(JNICALL* CallBooleanMethodV)(JNIEnv*, jobject, jmethodID, va_list);
jboolean(JNICALL* CallBooleanMethodA)(JNIEnv*,
jobject,
jmethodID,
const jvalue*);
jbyte(JNICALL* CallByteMethod)(JNIEnv*, jobject, jmethodID, ...);
jbyte(JNICALL* CallByteMethodV)(JNIEnv*, jobject, jmethodID, va_list);
jbyte(JNICALL* CallByteMethodA)(JNIEnv*, jobject, jmethodID, const jvalue*);
jchar(JNICALL* CallCharMethod)(JNIEnv*, jobject, jmethodID, ...);
jchar(JNICALL* CallCharMethodV)(JNIEnv*, jobject, jmethodID, va_list);
jchar(JNICALL* CallCharMethodA)(JNIEnv*, jobject, jmethodID, const jvalue*);
jshort(JNICALL* CallShortMethod)(JNIEnv*, jobject, jmethodID, ...);
jshort(JNICALL* CallShortMethodV)(JNIEnv*, jobject, jmethodID, va_list);
jshort(JNICALL* CallShortMethodA)(JNIEnv*, jobject, jmethodID, const jvalue*);
jint(JNICALL* CallIntMethod)(JNIEnv*, jobject, jmethodID, ...);
jint(JNICALL* CallIntMethodV)(JNIEnv*, jobject, jmethodID, va_list);
jint(JNICALL* CallIntMethodA)(JNIEnv*, jobject, jmethodID, const jvalue*);
jlong(JNICALL* CallLongMethod)(JNIEnv*, jobject, jmethodID, ...);
jlong(JNICALL* CallLongMethodV)(JNIEnv*, jobject, jmethodID, va_list);
jlong(JNICALL* CallLongMethodA)(JNIEnv*, jobject, jmethodID, const jvalue*);
jfloat(JNICALL* CallFloatMethod)(JNIEnv*, jobject, jmethodID, ...);
jfloat(JNICALL* CallFloatMethodV)(JNIEnv*, jobject, jmethodID, va_list);
jfloat(JNICALL* CallFloatMethodA)(JNIEnv*, jobject, jmethodID, const jvalue*);
jdouble(JNICALL* CallDoubleMethod)(JNIEnv*, jobject, jmethodID, ...);
jdouble(JNICALL* CallDoubleMethodV)(JNIEnv*, jobject, jmethodID, va_list);
jdouble(JNICALL* CallDoubleMethodA)(JNIEnv*,
jobject,
jmethodID,
const jvalue*);
void(JNICALL* CallVoidMethod)(JNIEnv*, jobject, jmethodID, ...);
void(JNICALL* CallVoidMethodV)(JNIEnv*, jobject, jmethodID, va_list);
void(JNICALL* CallVoidMethodA)(JNIEnv*, jobject, jmethodID, const jvalue*);
jobject(JNICALL* CallNonvirtualObjectMethod)(JNIEnv*,
jobject,
jclass,
jmethodID,
...);
jobject(JNICALL* CallNonvirtualObjectMethodV)(JNIEnv*,
jobject,
jclass,
jmethodID,
va_list);
jobject(JNICALL* CallNonvirtualObjectMethodA)(JNIEnv*,
jobject,
jclass,
jmethodID,
const jvalue*);
jboolean(JNICALL* CallNonvirtualBooleanMethod)(JNIEnv*,
jobject,
jclass,
jmethodID,
...);
jboolean(JNICALL* CallNonvirtualBooleanMethodV)(JNIEnv*,
jobject,
jclass,
jmethodID,
va_list);
jboolean(JNICALL* CallNonvirtualBooleanMethodA)(JNIEnv*,
jobject,
jclass,
jmethodID,
const jvalue*);
jbyte(JNICALL* CallNonvirtualByteMethod)(JNIEnv*,
jobject,
jclass,
jmethodID,
...);
jbyte(JNICALL* CallNonvirtualByteMethodV)(JNIEnv*,
jobject,
jclass,
jmethodID,
va_list);
jbyte(JNICALL* CallNonvirtualByteMethodA)(JNIEnv*,
jobject,
jclass,
jmethodID,
const jvalue*);
jchar(JNICALL* CallNonvirtualCharMethod)(JNIEnv*,
jobject,
jclass,
jmethodID,
...);
jchar(JNICALL* CallNonvirtualCharMethodV)(JNIEnv*,
jobject,
jclass,
jmethodID,
va_list);
jchar(JNICALL* CallNonvirtualCharMethodA)(JNIEnv*,
jobject,
jclass,
jmethodID,
const jvalue*);
jshort(JNICALL* CallNonvirtualShortMethod)(JNIEnv*,
jobject,
jclass,
jmethodID,
...);
jshort(JNICALL* CallNonvirtualShortMethodV)(JNIEnv*,
jobject,
jclass,
jmethodID,
va_list);
jshort(JNICALL* CallNonvirtualShortMethodA)(JNIEnv*,
jobject,
jclass,
jmethodID,
const jvalue*);
jint(JNICALL* CallNonvirtualIntMethod)(JNIEnv*,
jobject,
jclass,
jmethodID,
...);
jint(JNICALL* CallNonvirtualIntMethodV)(JNIEnv*,
jobject,
jclass,
jmethodID,
va_list);
jint(JNICALL* CallNonvirtualIntMethodA)(JNIEnv*,
jobject,
jclass,
jmethodID,
const jvalue*);
jlong(JNICALL* CallNonvirtualLongMethod)(JNIEnv*,
jobject,
jclass,
jmethodID,
...);
jlong(JNICALL* CallNonvirtualLongMethodV)(JNIEnv*,
jobject,
jclass,
jmethodID,
va_list);
jlong(JNICALL* CallNonvirtualLongMethodA)(JNIEnv*,
jobject,
jclass,
jmethodID,
const jvalue*);
jfloat(JNICALL* CallNonvirtualFloatMethod)(JNIEnv*,
jobject,
jclass,
jmethodID,
...);
jfloat(JNICALL* CallNonvirtualFloatMethodV)(JNIEnv*,
jobject,
jclass,
jmethodID,
va_list);
jfloat(JNICALL* CallNonvirtualFloatMethodA)(JNIEnv*,
jobject,
jclass,
jmethodID,
const jvalue*);
jdouble(JNICALL* CallNonvirtualDoubleMethod)(JNIEnv*,
jobject,
jclass,
jmethodID,
...);
jdouble(JNICALL* CallNonvirtualDoubleMethodV)(JNIEnv*,
jobject,
jclass,
jmethodID,
va_list);
jdouble(JNICALL* CallNonvirtualDoubleMethodA)(JNIEnv*,
jobject,
jclass,
jmethodID,
const jvalue*);
void(JNICALL* CallNonvirtualVoidMethod)(JNIEnv*,
jobject,
jclass,
jmethodID,
...);
void(JNICALL* CallNonvirtualVoidMethodV)(JNIEnv*,
jobject,
jclass,
jmethodID,
va_list);
void(JNICALL* CallNonvirtualVoidMethodA)(JNIEnv*,
jobject,
jclass,
jmethodID,
const jvalue*);
jfieldID(JNICALL* GetFieldID)(JNIEnv*, jclass, const char*, const char*);
jobject(JNICALL* GetObjectField)(JNIEnv*, jobject, jfieldID);
jboolean(JNICALL* GetBooleanField)(JNIEnv*, jobject, jfieldID);
jbyte(JNICALL* GetByteField)(JNIEnv*, jobject, jfieldID);
jchar(JNICALL* GetCharField)(JNIEnv*, jobject, jfieldID);
jshort(JNICALL* GetShortField)(JNIEnv*, jobject, jfieldID);
jint(JNICALL* GetIntField)(JNIEnv*, jobject, jfieldID);
jlong(JNICALL* GetLongField)(JNIEnv*, jobject, jfieldID);
jfloat(JNICALL* GetFloatField)(JNIEnv*, jobject, jfieldID);
jdouble(JNICALL* GetDoubleField)(JNIEnv*, jobject, jfieldID);
void(JNICALL* SetObjectField)(JNIEnv*, jobject, jfieldID, jobject);
void(JNICALL* SetBooleanField)(JNIEnv*, jobject, jfieldID, jboolean);
void(JNICALL* SetByteField)(JNIEnv*, jobject, jfieldID, jbyte);
void(JNICALL* SetCharField)(JNIEnv*, jobject, jfieldID, jchar);
void(JNICALL* SetShortField)(JNIEnv*, jobject, jfieldID, jshort);
void(JNICALL* SetIntField)(JNIEnv*, jobject, jfieldID, jint);
void(JNICALL* SetLongField)(JNIEnv*, jobject, jfieldID, jlong);
void(JNICALL* SetFloatField)(JNIEnv*, jobject, jfieldID, jfloat);
void(JNICALL* SetDoubleField)(JNIEnv*, jobject, jfieldID, jdouble);
jmethodID(JNICALL* GetStaticMethodID)(JNIEnv*,
jclass,
const char*,
const char*);
jobject(JNICALL* CallStaticObjectMethod)(JNIEnv*, jclass, jmethodID, ...);
jobject(JNICALL* CallStaticObjectMethodV)(JNIEnv*,
jclass,
jmethodID,
va_list);
jobject(JNICALL* CallStaticObjectMethodA)(JNIEnv*,
jclass,
jmethodID,
const jvalue*);
jboolean(JNICALL* CallStaticBooleanMethod)(JNIEnv*, jclass, jmethodID, ...);
jboolean(JNICALL* CallStaticBooleanMethodV)(JNIEnv*,
jclass,
jmethodID,
va_list);
jboolean(JNICALL* CallStaticBooleanMethodA)(JNIEnv*,
jclass,
jmethodID,
const jvalue*);
jbyte(JNICALL* CallStaticByteMethod)(JNIEnv*, jclass, jmethodID, ...);
jbyte(JNICALL* CallStaticByteMethodV)(JNIEnv*, jclass, jmethodID, va_list);
jbyte(JNICALL* CallStaticByteMethodA)(JNIEnv*,
jclass,
jmethodID,
const jvalue*);
jchar(JNICALL* CallStaticCharMethod)(JNIEnv*, jclass, jmethodID, ...);
jchar(JNICALL* CallStaticCharMethodV)(JNIEnv*, jclass, jmethodID, va_list);
jchar(JNICALL* CallStaticCharMethodA)(JNIEnv*,
jclass,
jmethodID,
const jvalue*);
jshort(JNICALL* CallStaticShortMethod)(JNIEnv*, jclass, jmethodID, ...);
jshort(JNICALL* CallStaticShortMethodV)(JNIEnv*, jclass, jmethodID, va_list);
jshort(JNICALL* CallStaticShortMethodA)(JNIEnv*,
jclass,
jmethodID,
const jvalue*);
jint(JNICALL* CallStaticIntMethod)(JNIEnv*, jclass, jmethodID, ...);
jint(JNICALL* CallStaticIntMethodV)(JNIEnv*, jclass, jmethodID, va_list);
jint(JNICALL* CallStaticIntMethodA)(JNIEnv*,
jclass,
jmethodID,
const jvalue*);
jlong(JNICALL* CallStaticLongMethod)(JNIEnv*, jclass, jmethodID, ...);
jlong(JNICALL* CallStaticLongMethodV)(JNIEnv*, jclass, jmethodID, va_list);
jlong(JNICALL* CallStaticLongMethodA)(JNIEnv*,
jclass,
jmethodID,
const jvalue*);
jfloat(JNICALL* CallStaticFloatMethod)(JNIEnv*, jclass, jmethodID, ...);
jfloat(JNICALL* CallStaticFloatMethodV)(JNIEnv*, jclass, jmethodID, va_list);
jfloat(JNICALL* CallStaticFloatMethodA)(JNIEnv*,
jclass,
jmethodID,
const jvalue*);
jdouble(JNICALL* CallStaticDoubleMethod)(JNIEnv*, jclass, jmethodID, ...);
jdouble(JNICALL* CallStaticDoubleMethodV)(JNIEnv*,
jclass,
jmethodID,
va_list);
jdouble(JNICALL* CallStaticDoubleMethodA)(JNIEnv*,
jclass,
jmethodID,
const jvalue*);
void(JNICALL* CallStaticVoidMethod)(JNIEnv*, jclass, jmethodID, ...);
void(JNICALL* CallStaticVoidMethodV)(JNIEnv*, jclass, jmethodID, va_list);
void(JNICALL* CallStaticVoidMethodA)(JNIEnv*,
jclass,
jmethodID,
const jvalue*);
jfieldID(JNICALL* GetStaticFieldID)(JNIEnv*,
jclass,
const char*,
const char*);
jobject(JNICALL* GetStaticObjectField)(JNIEnv*, jclass, jfieldID);
jboolean(JNICALL* GetStaticBooleanField)(JNIEnv*, jclass, jfieldID);
jbyte(JNICALL* GetStaticByteField)(JNIEnv*, jclass, jfieldID);
jchar(JNICALL* GetStaticCharField)(JNIEnv*, jclass, jfieldID);
jshort(JNICALL* GetStaticShortField)(JNIEnv*, jclass, jfieldID);
jint(JNICALL* GetStaticIntField)(JNIEnv*, jclass, jfieldID);
jlong(JNICALL* GetStaticLongField)(JNIEnv*, jclass, jfieldID);
jfloat(JNICALL* GetStaticFloatField)(JNIEnv*, jclass, jfieldID);
jdouble(JNICALL* GetStaticDoubleField)(JNIEnv*, jclass, jfieldID);
void(JNICALL* SetStaticObjectField)(JNIEnv*, jclass, jfieldID, jobject);
void(JNICALL* SetStaticBooleanField)(JNIEnv*, jclass, jfieldID, jboolean);
void(JNICALL* SetStaticByteField)(JNIEnv*, jclass, jfieldID, jbyte);
void(JNICALL* SetStaticCharField)(JNIEnv*, jclass, jfieldID, jchar);
void(JNICALL* SetStaticShortField)(JNIEnv*, jclass, jfieldID, jshort);
void(JNICALL* SetStaticIntField)(JNIEnv*, jclass, jfieldID, jint);
void(JNICALL* SetStaticLongField)(JNIEnv*, jclass, jfieldID, jlong);
void(JNICALL* SetStaticFloatField)(JNIEnv*, jclass, jfieldID, jfloat);
void(JNICALL* SetStaticDoubleField)(JNIEnv*, jclass, jfieldID, jdouble);
jstring(JNICALL* NewString)(JNIEnv*, const jchar*, jsize);
jsize(JNICALL* GetStringLength)(JNIEnv*, jstring);
const jchar*(JNICALL* GetStringChars)(JNIEnv*, jstring, jboolean*);
void(JNICALL* ReleaseStringChars)(JNIEnv*, jstring, const jchar*);
jstring(JNICALL* NewStringUTF)(JNIEnv*, const char*);
jsize(JNICALL* GetStringUTFLength)(JNIEnv*, jstring);
const char*(JNICALL* GetStringUTFChars)(JNIEnv*, jstring, jboolean*);
void(JNICALL* ReleaseStringUTFChars)(JNIEnv*, jstring, const char*);
jsize(JNICALL* GetArrayLength)(JNIEnv*, jarray);
jobjectArray(JNICALL* NewObjectArray)(JNIEnv*, jsize, jclass, jobject);
jobject(JNICALL* GetObjectArrayElement)(JNIEnv*, jobjectArray, jsize);
void(JNICALL* SetObjectArrayElement)(JNIEnv*, jobjectArray, jsize, jobject);
jbooleanArray(JNICALL* NewBooleanArray)(JNIEnv*, jsize);
jbyteArray(JNICALL* NewByteArray)(JNIEnv*, jsize);
jcharArray(JNICALL* NewCharArray)(JNIEnv*, jsize);
jshortArray(JNICALL* NewShortArray)(JNIEnv*, jsize);
jintArray(JNICALL* NewIntArray)(JNIEnv*, jsize);
jlongArray(JNICALL* NewLongArray)(JNIEnv*, jsize);
jfloatArray(JNICALL* NewFloatArray)(JNIEnv*, jsize);
jdoubleArray(JNICALL* NewDoubleArray)(JNIEnv*, jsize);
jboolean*(JNICALL* GetBooleanArrayElements)(JNIEnv*,
jbooleanArray,
jboolean*);
jbyte*(JNICALL* GetByteArrayElements)(JNIEnv*, jbyteArray, jboolean*);
jchar*(JNICALL* GetCharArrayElements)(JNIEnv*, jcharArray, jboolean*);
jshort*(JNICALL* GetShortArrayElements)(JNIEnv*, jshortArray, jboolean*);
jint*(JNICALL* GetIntArrayElements)(JNIEnv*, jintArray, jboolean*);
jlong*(JNICALL* GetLongArrayElements)(JNIEnv*, jlongArray, jboolean*);
jfloat*(JNICALL* GetFloatArrayElements)(JNIEnv*, jfloatArray, jboolean*);
jdouble*(JNICALL* GetDoubleArrayElements)(JNIEnv*, jdoubleArray, jboolean*);
void(JNICALL* ReleaseBooleanArrayElements)(JNIEnv*,
jbooleanArray,
jboolean*,
jint);
void(JNICALL* ReleaseByteArrayElements)(JNIEnv*, jbyteArray, jbyte*, jint);
void(JNICALL* ReleaseCharArrayElements)(JNIEnv*, jcharArray, jchar*, jint);
void(JNICALL* ReleaseShortArrayElements)(JNIEnv*, jshortArray, jshort*, jint);
void(JNICALL* ReleaseIntArrayElements)(JNIEnv*, jintArray, jint*, jint);
void(JNICALL* ReleaseLongArrayElements)(JNIEnv*, jlongArray, jlong*, jint);
void(JNICALL* ReleaseFloatArrayElements)(JNIEnv*, jfloatArray, jfloat*, jint);
void(JNICALL* ReleaseDoubleArrayElements)(JNIEnv*,
jdoubleArray,
jdouble*,
jint);
void(JNICALL* GetBooleanArrayRegion)(JNIEnv*,
jbooleanArray,
jsize,
jsize,
jboolean*);
void(JNICALL* GetByteArrayRegion)(JNIEnv*, jbyteArray, jsize, jsize, jbyte*);
void(JNICALL* GetCharArrayRegion)(JNIEnv*, jcharArray, jsize, jsize, jchar*);
void(JNICALL* GetShortArrayRegion)(JNIEnv*,
jshortArray,
jsize,
jsize,
jshort*);
void(JNICALL* GetIntArrayRegion)(JNIEnv*, jintArray, jsize, jsize, jint*);
void(JNICALL* GetLongArrayRegion)(JNIEnv*, jlongArray, jsize, jsize, jlong*);
void(JNICALL* GetFloatArrayRegion)(JNIEnv*,
jfloatArray,
jsize,
jsize,
jfloat*);
void(JNICALL* GetDoubleArrayRegion)(JNIEnv*,
jdoubleArray,
jsize,
jsize,
jdouble*);
void(JNICALL* SetBooleanArrayRegion)(JNIEnv*,
jbooleanArray,
jsize,
jsize,
const jboolean*);
void(JNICALL* SetByteArrayRegion)(JNIEnv*,
jbyteArray,
jsize,
jsize,
const jbyte*);
void(JNICALL* SetCharArrayRegion)(JNIEnv*,
jcharArray,
jsize,
jsize,
const jchar*);
void(JNICALL* SetShortArrayRegion)(JNIEnv*,
jshortArray,
jsize,
jsize,
const jshort*);
void(JNICALL* SetIntArrayRegion)(JNIEnv*,
jintArray,
jsize,
jsize,
const jint*);
void(JNICALL* SetLongArrayRegion)(JNIEnv*,
jlongArray,
jsize,
jsize,
const jlong*);
void(JNICALL* SetFloatArrayRegion)(JNIEnv*,
jfloatArray,
jsize,
jsize,
const jfloat*);
void(JNICALL* SetDoubleArrayRegion)(JNIEnv*,
jdoubleArray,
jsize,
jsize,
const jdouble*);
jint(JNICALL* RegisterNatives)(JNIEnv*, jclass, const JNINativeMethod*, jint);
jint(JNICALL* UnregisterNatives)(JNIEnv*, jclass);
jint(JNICALL* MonitorEnter)(JNIEnv*, jobject);
jint(JNICALL* MonitorExit)(JNIEnv*, jobject);
jint(JNICALL* GetJavaVM)(JNIEnv*, JavaVM**);
void(JNICALL* GetStringRegion)(JNIEnv*, jstring, jsize, jsize, jchar*);
void(JNICALL* GetStringUTFRegion)(JNIEnv*, jstring, jsize, jsize, char*);
void*(JNICALL* GetPrimitiveArrayCritical)(JNIEnv*, jarray, jboolean*);
void(JNICALL* ReleasePrimitiveArrayCritical)(JNIEnv*, jarray, void*, jint);
const jchar*(JNICALL* GetStringCritical)(JNIEnv*, jstring, jboolean*);
void(JNICALL* ReleaseStringCritical)(JNIEnv*, jstring, const jchar*);
jweak(JNICALL* NewWeakGlobalRef)(JNIEnv*, jobject);
void(JNICALL* DeleteWeakGlobalRef)(JNIEnv*, jweak);
jboolean(JNICALL* ExceptionCheck)(JNIEnv*);
jobject(JNICALL* NewDirectByteBuffer)(JNIEnv*, void*, jlong);
void*(JNICALL* GetDirectBufferAddress)(JNIEnv* env, jobject);
jlong(JNICALL* GetDirectBufferCapacity)(JNIEnv*, jobject);
};
inline void atomicOr(uint32_t* p, int v)
{
for (uint32_t old = *p; not atomicCompareAndSwap32(p, old, old | v);
old = *p) {
}
}
inline void atomicAnd(uint32_t* p, int v)
{
for (uint32_t old = *p; not atomicCompareAndSwap32(p, old, old & v);
old = *p) {
}
}
inline int strcmp(const int8_t* a, const int8_t* b)
{
return ::strcmp(reinterpret_cast<const char*>(a),
reinterpret_cast<const char*>(b));
}
void noop();
class Reference {
public:
Reference(object target, Reference** handle, bool weak)
: target(target), next(*handle), handle(handle), count(0), weak(weak)
{
if (next) {
next->handle = &next;
}
*handle = this;
}
object target;
Reference* next;
Reference** handle;
unsigned count;
bool weak;
};
class Classpath;
class Gc {
public:
enum Type {
#include "type-enums.cpp"
};
};
class GcObject {
public:
template <class T>
T* as(Thread* t);
template <class T>
bool isa(Thread* t);
protected:
template <class T>
T& field_at(size_t offset)
{
return *reinterpret_cast<T*>(reinterpret_cast<uint8_t*>(this) + offset);
}
};
class GcFinalizer;
class GcClassLoader;
class GcJreference;
class GcArray;
class GcThrowable;
class GcRoots;
class Machine {
public:
enum AllocationType {
MovableAllocation,
FixedAllocation,
ImmortalAllocation
};
Machine(System* system,
Heap* heap,
Finder* bootFinder,
Finder* appFinder,
Processor* processor,
Classpath* classpath,
const char** properties,
unsigned propertyCount,
const char** arguments,
unsigned argumentCount,
unsigned stackSizeInBytes);
~Machine()
{
dispose();
}
void dispose();
JavaVMVTable* vtable;
System* system;
Heap::Client* heapClient;
Heap* heap;
Finder* bootFinder;
Finder* appFinder;
Processor* processor;
Classpath* classpath;
Thread* rootThread;
Thread* exclusive;
Thread* finalizeThread;
Reference* jniReferences;
char** properties;
unsigned propertyCount;
const char** arguments;
unsigned argumentCount;
unsigned threadCount;
unsigned activeCount;
unsigned liveCount;
unsigned daemonCount;
unsigned fixedFootprint;
unsigned stackSizeInBytes;
System::Local* localThread;
System::Monitor* stateLock;
System::Monitor* heapLock;
System::Monitor* classLock;
System::Monitor* referenceLock;
System::Monitor* shutdownLock;
System::Library* libraries;
FILE* errorLog;
BootImage* bootimage;
GcArray* types;
GcRoots* roots;
GcFinalizer* finalizers;
GcFinalizer* tenuredFinalizers;
GcFinalizer* finalizeQueue;
GcJreference* weakReferences;
GcJreference* tenuredWeakReferences;
bool unsafe;
bool collecting;
bool triedBuiltinOnLoad;
bool dumpedHeapOnOOM;
bool alive;
JavaVMVTable javaVMVTable;
JNIEnvVTable jniEnvVTable;
uintptr_t* heapPool[ThreadHeapPoolSize];
unsigned heapPoolIndex;
size_t bootimageSize;
};
void printTrace(Thread* t, GcThrowable* exception);
void enterActiveState(Thread* t);
#ifdef VM_STRESS
inline void stress(Thread* t);
#else // not VM_STRESS
#define stress(t)
#endif // not VM_STRESS
uint64_t runThread(Thread*, uintptr_t*);
uint64_t run(Thread* t,
uint64_t (*function)(Thread*, uintptr_t*),
uintptr_t* arguments);
void checkDaemon(Thread* t);
GcRoots* roots(Thread* t);
extern "C" uint64_t vmRun(uint64_t (*function)(Thread*, uintptr_t*),
uintptr_t* arguments,
void* checkpoint);
extern "C" void vmRun_returnAddress();
class GcThread;
class GcThrowable;
class GcString;
class Thread {
public:
enum State {
NoState,
ActiveState,
IdleState,
ZombieState,
JoinedState,
ExclusiveState,
ExitState
};
enum Flag {
UseBackupHeapFlag = 1 << 0,
WaitingFlag = 1 << 1,
TracingFlag = 1 << 2,
DaemonFlag = 1 << 3,
StressFlag = 1 << 4,
ActiveFlag = 1 << 5,
SystemFlag = 1 << 6,
JoinFlag = 1 << 7,
TryNativeFlag = 1 << 8
};
class Protector {
public:
Protector(Thread* t) : t(t), next(t->protector)
{
t->protector = this;
}
~Protector()
{
t->protector = next;
}
virtual void visit(Heap::Visitor* v) = 0;
Thread* t;
Protector* next;
};
class SingleProtector : public Protector {
public:
SingleProtector(Thread* t, void* p) : Protector(t), p(p)
{
}
virtual void visit(Heap::Visitor* v)
{
v->visit(p);
}
void* p;
};
class Resource {
public:
Resource(Thread* t, Resource* next) : t(t), next(next)
{
t->resource = this;
}
virtual void release() = 0;
Thread* t;
Resource* next;
};
class AutoResource : public Resource {
public:
AutoResource(Thread* t) : Resource(t, t->resource)
{
}
~AutoResource()
{
t->resource = next;
}
virtual void release() = 0;
};
class ClassInitStack : public AutoResource {
public:
ClassInitStack(Thread* t, GcClass* class_)
: AutoResource(t),
next(t->classInitStack),
class_(class_),
protector(t, &(this->class_))
{
t->classInitStack = this;
}
~ClassInitStack()
{
t->classInitStack = next;
}
virtual void release()
{
this->ClassInitStack::~ClassInitStack();
}
ClassInitStack* next;
GcClass* class_;
SingleProtector protector;
};
class LibraryLoadStack : public AutoResource {
public:
LibraryLoadStack(Thread* t, GcClassLoader* classLoader)
: AutoResource(t),
next(t->libraryLoadStack),
classLoader(classLoader),
protector(t, &(this->classLoader))
{
t->libraryLoadStack = this;
}
~LibraryLoadStack()
{
t->libraryLoadStack = next;
}
virtual void release()
{
this->LibraryLoadStack::~LibraryLoadStack();
}
LibraryLoadStack* next;
GcClassLoader* classLoader;
SingleProtector protector;
};
class Checkpoint {
public:
Checkpoint(Thread* t)
: t(t),
next(t->checkpoint),
resource(t->resource),
protector(t->protector),
noThrow(false)
{
t->checkpoint = this;
}
~Checkpoint()
{
t->checkpoint = next;
}
virtual void NO_RETURN unwind() = 0;
Thread* t;
Checkpoint* next;
Resource* resource;
Protector* protector;
bool noThrow;
};
class RunCheckpoint : public Checkpoint {
public:
RunCheckpoint(Thread* t) : Checkpoint(t), stack(0)
{
}
virtual void unwind()
{
void* stack = this->stack;
this->stack = 0;
expect(t->m->system, stack);
vmJump(voidPointer(vmRun_returnAddress), 0, stack, t, 0, 0);
}
void* stack;
};
class Runnable : public System::Runnable {
public:
Runnable(Thread* t) : t(t)
{
}
virtual void attach(System::Thread* st)
{
t->systemThread = st;
}
virtual void run();
virtual bool interrupted();
virtual void setInterrupted(bool v);
Thread* t;
};
Thread(Machine* m, GcThread* javaThread, Thread* parent);
void init();
void exit();
void dispose();
void setFlag(Flag flag) {
atomicOr(&flags, flag);
}
void clearFlag(Flag flag) {
atomicAnd(&flags, ~flag);
}
unsigned getFlags() {
return flags;
}
JNIEnvVTable* vtable;
Machine* m;
Thread* parent;
Thread* peer;
Thread* child;
Thread* waitNext;
State state;
unsigned criticalLevel;
System::Thread* systemThread;
System::Monitor* lock;
GcThread* javaThread;
GcThrowable* exception;
unsigned heapIndex;
unsigned heapOffset;
Protector* protector;
ClassInitStack* classInitStack;
LibraryLoadStack* libraryLoadStack;
Resource* resource;
Checkpoint* checkpoint;
Runnable runnable;
uintptr_t* defaultHeap;
uintptr_t* heap;
uintptr_t backupHeap[ThreadBackupHeapSizeInWords];
unsigned backupHeapIndex;
private:
unsigned flags;
};
class GcJfield;
class Classpath {
public:
virtual GcJclass* makeJclass(Thread* t, GcClass* class_) = 0;
virtual GcString* makeString(Thread* t,
object array,
int32_t offset,
int32_t length) = 0;
virtual GcThread* makeThread(Thread* t, Thread* parent) = 0;
virtual object makeJMethod(Thread* t, GcMethod* vmMethod) = 0;
virtual GcMethod* getVMMethod(Thread* t, object jmethod) = 0;
virtual object makeJField(Thread* t, GcField* vmField) = 0;
virtual GcField* getVMField(Thread* t, GcJfield* jfield) = 0;
virtual void clearInterrupted(Thread* t) = 0;
virtual void runThread(Thread* t) = 0;
virtual void resolveNative(Thread* t, GcMethod* method) = 0;
virtual void interceptMethods(Thread* t) = 0;
virtual void preBoot(Thread* t) = 0;
virtual bool mayInitClasses() = 0;
virtual void boot(Thread* t) = 0;
virtual const char* bootClasspath() = 0;
virtual object makeDirectByteBuffer(Thread* t, void* p, jlong capacity) = 0;
virtual void* getDirectBufferAddress(Thread* t, object buffer) = 0;
virtual int64_t getDirectBufferCapacity(Thread* t, object buffer) = 0;
virtual bool canTailCall(Thread* t,
GcMethod* caller,
GcByteArray* calleeClassName,
GcByteArray* calleeMethodName,
GcByteArray* calleeMethodSpec) = 0;
virtual GcClassLoader* libraryClassLoader(Thread* t, GcMethod* caller) = 0;
virtual void shutDown(Thread* t) = 0;
virtual void dispose() = 0;
};
#ifdef _MSC_VER
template <class T>
class ThreadRuntimeArray : public Thread::AutoResource {
public:
ThreadRuntimeArray(Thread* t, unsigned size)
: AutoResource(t),
body(static_cast<T*>(t->m->heap->allocate(size * sizeof(T)))),
size(size)
{
}
~ThreadRuntimeArray()
{
t->m->heap->free(body, size * sizeof(T));
}
virtual void release()
{
ThreadRuntimeArray::~ThreadRuntimeArray();
}
T* body;
unsigned size;
};
#define THREAD_RUNTIME_ARRAY(thread, type, name, size) \
ThreadRuntimeArray<type> name(thread, size);
#else // not _MSC_VER
#define THREAD_RUNTIME_ARRAY(thread, type, name, size) type name##_body[size];
#endif // not _MSC_VER
Classpath* makeClasspath(System* system,
Allocator* allocator,
const char* javaHome,
const char* embedPrefix);
typedef uint64_t(JNICALL* FastNativeFunction)(Thread*, GcMethod*, uintptr_t*);
typedef void(JNICALL* FastVoidNativeFunction)(Thread*, GcMethod*, uintptr_t*);
inline GcClass* objectClass(Thread*, object o)
{
return reinterpret_cast<GcClass*>(
maskAlignedPointer(fieldAtOffset<object>(o, 0)));
}
inline unsigned stackSizeInWords(Thread* t)
{
return t->m->stackSizeInBytes / BytesPerWord;
}
void enter(Thread* t, Thread::State state);
inline void enterActiveState(Thread* t)
{
enter(t, Thread::ActiveState);
}
class StateResource : public Thread::AutoResource {
public:
StateResource(Thread* t, Thread::State state)
: AutoResource(t), oldState(t->state)
{
enter(t, state);
}
~StateResource()
{
enter(t, oldState);
}
virtual void release()
{
this->StateResource::~StateResource();
}
private:
Thread::State oldState;
};
inline void dispose(Thread* t, Reference* r)
{
*(r->handle) = r->next;
if (r->next) {
r->next->handle = r->handle;
}
t->m->heap->free(r, sizeof(*r));
}
inline void acquire(Thread*, Reference* r)
{
++r->count;
}
inline void release(Thread* t, Reference* r)
{
if ((--r->count) == 0) {
dispose(t, r);
}
}
void collect(Thread* t, Heap::CollectionType type, int pendingAllocation = 0);
void shutDown(Thread* t);
#ifdef VM_STRESS
inline void stress(Thread* t)
{
if ((not t->m->unsafe)
and (t->getFlags() & (Thread::StressFlag | Thread::TracingFlag)) == 0
and t->state != Thread::NoState and t->state != Thread::IdleState) {
t->setFlag(Thread::StressFlag);
#ifdef VM_STRESS_MAJOR
collect(t, Heap::MajorCollection);
#else // not VM_STRESS_MAJOR
collect(t, Heap::MinorCollection);
#endif // not VM_STRESS_MAJOR
t->clearFlag(Thread::StressFlag);
}
}
#endif // not VM_STRESS
inline void acquire(Thread* t, System::Monitor* m)
{
if (not m->tryAcquire(t->systemThread)) {
ENTER(t, Thread::IdleState);
m->acquire(t->systemThread);
}
stress(t);
}
inline void release(Thread* t, System::Monitor* m)
{
m->release(t->systemThread);
}
class MonitorResource : public Thread::AutoResource {
public:
MonitorResource(Thread* t, System::Monitor* m) : AutoResource(t), m(m)
{
acquire(t, m);
}
~MonitorResource()
{
vm::release(t, m);
}
virtual void release()
{
this->MonitorResource::~MonitorResource();
}
private:
System::Monitor* m;
};
class RawMonitorResource : public Thread::Resource {
public:
RawMonitorResource(Thread* t, System::Monitor* m)
: Resource(t, t->resource), m(m)
{
m->acquire(t->systemThread);
}
~RawMonitorResource()
{
t->resource = next;
vm::release(t, m);
}
virtual void release()
{
this->RawMonitorResource::~RawMonitorResource();
}
private:
System::Monitor* m;
};
inline Aborter* getAborter(Thread* t)
{
return t->m->system;
}
inline bool ensure(Thread* t, unsigned sizeInBytes)
{
if (t->heapIndex + ceilingDivide(sizeInBytes, BytesPerWord)
> ThreadHeapSizeInWords) {
if (sizeInBytes <= ThreadBackupHeapSizeInBytes) {
expect(t, (t->getFlags() & Thread::UseBackupHeapFlag) == 0);
t->setFlag(Thread::UseBackupHeapFlag);
return true;
} else {
return false;
}
} else {
return true;
}
}
object allocate2(Thread* t, unsigned sizeInBytes, bool objectMask);
object allocate3(Thread* t,
Alloc* allocator,
Machine::AllocationType type,
unsigned sizeInBytes,
bool objectMask);
inline object allocateSmall(Thread* t, unsigned sizeInBytes)
{
assertT(t,
t->heapIndex + ceilingDivide(sizeInBytes, BytesPerWord)
<= ThreadHeapSizeInWords);
object o = reinterpret_cast<object>(t->heap + t->heapIndex);
t->heapIndex += ceilingDivide(sizeInBytes, BytesPerWord);
return o;
}
inline object allocate(Thread* t, unsigned sizeInBytes, bool objectMask)
{
stress(t);
if (UNLIKELY(t->heapIndex + ceilingDivide(sizeInBytes, BytesPerWord)
> ThreadHeapSizeInWords or t->m->exclusive)) {
return allocate2(t, sizeInBytes, objectMask);
} else {
assertT(t, t->criticalLevel == 0);
return allocateSmall(t, sizeInBytes);
}
}
inline void mark(Thread* t, object o, unsigned offset, unsigned count)
{
t->m->heap->mark(o, offset / BytesPerWord, count);
}
inline void mark(Thread* t, object o, unsigned offset)
{
t->m->heap->mark(o, offset / BytesPerWord, 1);
}
inline void setField(Thread* t, object target, unsigned offset, object value)
{
fieldAtOffset<object>(target, offset) = value;
mark(t, target, offset);
}
inline void setObject(Thread* t,
GcObject* target,
unsigned offset,
GcObject* value)
{
setField(t, target, offset, value);
}
inline void setObjectClass(Thread*, object o, GcClass* c)
{
fieldAtOffset<object>(o, 0) = reinterpret_cast<object>(
reinterpret_cast<intptr_alias_t>(c)
| (reinterpret_cast<intptr_alias_t>(fieldAtOffset<object>(o, 0))
& (~PointerMask)));
}
inline const char* findProperty(Machine* m, const char* name)
{
for (unsigned i = 0; i < m->propertyCount; ++i) {
const char* p = m->properties[i];
const char* n = name;
while (*p and *p != '=' and *n and *p == *n) {
++p;
++n;
}
if (*p == '=' and *n == 0) {
return p + 1;
}
}
return 0;
}
inline const char* findProperty(Thread* t, const char* name)
{
return findProperty(t->m, name);
}
object arrayBodyUnsafe(Thread*, GcArray*, unsigned);
bool instanceOf(Thread* t, GcClass* class_, object o);
template <class T>
T* GcObject::as(Thread* t UNUSED)
{
assertT(t,
t->m->unsafe || instanceOf(t,
reinterpret_cast<GcClass*>(arrayBodyUnsafe(
t, t->m->types, T::Type)),
this));
return static_cast<T*>(this);
}
template <class T>
bool GcObject::isa(Thread* t)
{
return instanceOf(
t,
reinterpret_cast<GcClass*>(arrayBodyUnsafe(t, t->m->types, T::Type)),
this);
}
template <class T>
T* cast(Thread* t UNUSED, object o)
{
if (o == 0) {
return 0;
}
assertT(t,
t->m->unsafe || instanceOf(t,
reinterpret_cast<GcClass*>(arrayBodyUnsafe(
t, t->m->types, T::Type)),
o));
return reinterpret_cast<T*>(o);
}
#include "type-declarations.cpp"
inline object arrayBodyUnsafe(Thread*, GcArray* a, unsigned index)
{
return a->body()[index];
}
inline void Thread::Runnable::run()
{
enterActiveState(t);
vm::run(t, runThread, 0);
if (t->exception and t->exception != roots(t)->shutdownInProgress()) {
printTrace(t, t->exception);
}
t->exit();
}
inline bool Thread::Runnable::interrupted()
{
return t->javaThread and t->javaThread->interrupted();
}
inline void Thread::Runnable::setInterrupted(bool v)
{
t->javaThread->interrupted() = v;
}
inline uint64_t runRaw(Thread* t,
uint64_t (*function)(Thread*, uintptr_t*),
uintptr_t* arguments)
{
Thread::RunCheckpoint checkpoint(t);
return vmRun(function, arguments, &checkpoint);
}
inline uint64_t run(Thread* t,
uint64_t (*function)(Thread*, uintptr_t*),
uintptr_t* arguments)
{
ENTER(t, Thread::ActiveState);
return runRaw(t, function, arguments);
}
inline void runJavaThread(Thread* t)
{
t->m->classpath->runThread(t);
}
void runFinalizeThread(Thread* t);
inline uint64_t runThread(Thread* t, uintptr_t*)
{
t->m->localThread->set(t);
checkDaemon(t);
if (t == t->m->finalizeThread) {
runFinalizeThread(t);
} else if (t->javaThread) {
runJavaThread(t);
}
return 1;
}
inline bool startThread(Thread* t, Thread* p)
{
p->setFlag(Thread::JoinFlag);
return t->m->system->success(t->m->system->start(&(p->runnable)));
}
inline void addThread(Thread* t, Thread* p)
{
ACQUIRE_RAW(t, t->m->stateLock);
assertT(t, p->state == Thread::NoState);
expect(t,
t->state == Thread::ActiveState || t->state == Thread::ExclusiveState
|| t->state == Thread::NoState);
p->state = Thread::IdleState;
++t->m->threadCount;
++t->m->liveCount;
p->peer = p->parent->child;
p->parent->child = p;
if (p->javaThread) {
p->javaThread->peer() = reinterpret_cast<jlong>(p);
}
}
inline void removeThread(Thread* t, Thread* p)
{
ACQUIRE_RAW(t, t->m->stateLock);
assertT(t, p->state == Thread::IdleState);
--t->m->liveCount;
--t->m->threadCount;
t->m->stateLock->notifyAll(t->systemThread);
p->parent->child = p->peer;
if (p->javaThread) {
p->javaThread->peer() = 0;
}
p->dispose();
}
inline Thread* startThread(Thread* t, GcThread* javaThread)
{
{
PROTECT(t, javaThread);
stress(t);
ACQUIRE_RAW(t, t->m->stateLock);
if (t->m->threadCount > t->m->liveCount + ZombieCollectionThreshold) {
collect(t, Heap::MinorCollection);
}
}
Thread* p = t->m->processor->makeThread(t->m, javaThread, t);
addThread(t, p);
if (startThread(t, p)) {
return p;
} else {
removeThread(t, p);
return 0;
}
}
inline void registerDaemon(Thread* t)
{
ACQUIRE_RAW(t, t->m->stateLock);
t->setFlag(Thread::DaemonFlag);
++t->m->daemonCount;
t->m->stateLock->notifyAll(t->systemThread);
}
inline void checkDaemon(Thread* t)
{
if (t->javaThread->daemon()) {
registerDaemon(t);
}
}
inline uint64_t initAttachedThread(Thread* t, uintptr_t* arguments)
{
bool daemon = arguments[0];
t->javaThread = t->m->classpath->makeThread(t, t->m->rootThread);
t->javaThread->peer() = reinterpret_cast<jlong>(t);
if (daemon) {
t->javaThread->daemon() = true;
registerDaemon(t);
}
t->m->localThread->set(t);
return 1;
}
inline Thread* attachThread(Machine* m, bool daemon)
{
Thread* t = m->processor->makeThread(m, 0, m->rootThread);
m->system->attach(&(t->runnable));
addThread(t, t);
enter(t, Thread::ActiveState);
uintptr_t arguments[] = {daemon};
if (run(t, initAttachedThread, arguments)) {
enter(t, Thread::IdleState);
return t;
} else {
t->exit();
return 0;
}
}
inline GcRoots* roots(Thread* t)
{
return t->m->roots;
}
inline GcClass* type(Thread* t, Gc::Type type)
{
return cast<GcClass>(t, t->m->types->body()[type]);
}
inline void setType(Thread* t, Gc::Type type, GcClass* value)
{
t->m->types->setBodyElement(t, type, value);
}
inline bool objectFixed(Thread*, object o)
{
return (alias(o, 0) & (~PointerMask)) == FixedMark;
}
inline bool objectExtended(Thread*, object o)
{
return (alias(o, 0) & (~PointerMask)) == ExtendedMark;
}
inline bool hashTaken(Thread*, object o)
{
return (alias(o, 0) & (~PointerMask)) == HashTakenMark;
}
inline unsigned baseSize(Thread* t UNUSED, object o, GcClass* class_)
{
assertT(t, class_->fixedSize() >= BytesPerWord);
unsigned size = ceilingDivide(class_->fixedSize(), BytesPerWord);
if (class_->arrayElementSize() > 0) {
size += ceilingDivide(class_->arrayElementSize()
* fieldAtOffset<uintptr_t>(
o, class_->fixedSize() - BytesPerWord),
BytesPerWord);
}
return size;
}
object makeTrace(Thread* t, Processor::StackWalker* walker);
object makeTrace(Thread* t, Thread* target);
inline object makeTrace(Thread* t)
{
return makeTrace(t, t);
}
inline object makeNew(Thread* t, GcClass* class_)
{
assertT(t, t->state == Thread::NoState or t->state == Thread::ActiveState);
PROTECT(t, class_);
unsigned sizeInBytes = pad(class_->fixedSize());
assertT(t, sizeInBytes);
object instance = allocate(t, sizeInBytes, class_->objectMask());
setObjectClass(t, instance, class_);
return instance;
}
object makeNewGeneral(Thread* t, GcClass* class_);
inline object make(Thread* t, GcClass* class_)
{
if (UNLIKELY(class_->vmFlags() & (WeakReferenceFlag | HasFinalizerFlag))) {
return makeNewGeneral(t, class_);
} else {
return makeNew(t, class_);
}
}
GcByteArray* makeByteArrayV(Thread* t, const char* format, va_list a, int size);
GcByteArray* makeByteArray(Thread* t, const char* format, ...);
GcString* makeString(Thread* t, const char* format, ...);
#ifndef HAVE_StringOffset
inline uint32_t GcString::length(Thread* t)
{
return cast<GcCharArray>(t, this->data())->length();
}
inline uint32_t GcString::offset(Thread*)
{
return 0;
}
#ifndef HAVE_StringHash32
inline GcString* makeString(Thread* t, object data, int32_t hash, int32_t)
{
return makeString(t, data, hash);
}
#endif // not HAVE_StringHash32
inline GcString* makeString(Thread* t,
object odata,
unsigned offset,
unsigned length,
unsigned)
{
GcCharArray* data = cast<GcCharArray>(t, odata);
if (offset == 0 and length == data->length()) {
return makeString(t, reinterpret_cast<object>(data), 0, 0);
} else {
PROTECT(t, data);
GcCharArray* array = makeCharArray(t, length);
memcpy(array->body().begin(), &data->body()[offset], length * 2);
return makeString(t, reinterpret_cast<object>(array), 0, 0);
}
}
#endif // not HAVE_StringOffset
int stringUTFLength(Thread* t,
GcString* string,
unsigned start,
unsigned length);
inline int stringUTFLength(Thread* t, GcString* string)
{
return stringUTFLength(t, string, 0, string->length(t));
}
void stringChars(Thread* t,
GcString* string,
unsigned start,
unsigned length,
char* chars);
inline void stringChars(Thread* t, GcString* string, char* chars)
{
stringChars(t, string, 0, string->length(t), chars);
}
void stringChars(Thread* t,
GcString* string,
unsigned start,
unsigned length,
uint16_t* chars);
inline void stringChars(Thread* t, GcString* string, uint16_t* chars)
{
stringChars(t, string, 0, string->length(t), chars);
}
void stringUTFChars(Thread* t,
GcString* string,
unsigned start,
unsigned length,
char* chars,
unsigned charsLength);
inline void stringUTFChars(Thread* t,
GcString* string,
char* chars,
unsigned charsLength)
{
stringUTFChars(t, string, 0, string->length(t), chars, charsLength);
}
bool isAssignableFrom(Thread* t, GcClass* a, GcClass* b);
GcMethod* classInitializer(Thread* t, GcClass* class_);
object frameMethod(Thread* t, int frame);
inline uintptr_t& extendedWord(Thread* t UNUSED, object o, unsigned baseSize)
{
assertT(t, objectExtended(t, o));
return fieldAtOffset<uintptr_t>(o, baseSize * BytesPerWord);
}
inline unsigned extendedSize(Thread* t, object o, unsigned baseSize)
{
return baseSize + objectExtended(t, o);
}
inline void markHashTaken(Thread* t, object o)
{
assertT(t, not objectExtended(t, o));
assertT(t, not objectFixed(t, o));
ACQUIRE_RAW(t, t->m->heapLock);
alias(o, 0) |= HashTakenMark;
t->m->heap->pad(o);
}
inline uint32_t takeHash(Thread*, object o)
{
// some broken code implicitly relies on System.identityHashCode
// always returning a non-negative number (e.g. old versions of
// com/sun/xml/bind/v2/util/CollisionCheckStack.hash), hence the "&
// 0x7FFFFFFF":
return (reinterpret_cast<uintptr_t>(o) / BytesPerWord) & 0x7FFFFFFF;
}
inline uint32_t objectHash(Thread* t, object o)
{
if (objectExtended(t, o)) {
return extendedWord(t, o, baseSize(t, o, objectClass(t, o)));
} else {
if (not objectFixed(t, o)) {
markHashTaken(t, o);
}
return takeHash(t, o);
}
}
inline bool objectEqual(Thread*, object a, object b)
{
return a == b;
}
inline uint32_t byteArrayHash(Thread* t UNUSED, object ao)
{
GcByteArray* a = cast<GcByteArray>(t, ao);
return hash(a->body());
}
inline uint32_t charArrayHash(Thread* t UNUSED, object ao)
{
GcByteArray* a = cast<GcByteArray>(t, ao);
return hash(a->body());
}
inline bool byteArrayEqual(Thread* t UNUSED, object ao, object bo)
{
GcByteArray* a = cast<GcByteArray>(t, ao);
GcByteArray* b = cast<GcByteArray>(t, bo);
return a == b
or ((a->length() == b->length())
and memcmp(a->body().begin(), b->body().begin(), a->length())
== 0);
}
inline uint32_t stringHash(Thread* t, object so)
{
GcString* s = cast<GcString>(t, so);
if (s->hashCode() == 0 and s->length(t)) {
if (objectClass(t, s->data()) == type(t, GcByteArray::Type)) {
s->hashCode() = hash(cast<GcByteArray>(t, s->data())->body().subslice(
s->offset(t), s->length(t)));
} else {
s->hashCode() = hash(cast<GcCharArray>(t, s->data())->body().subslice(
s->offset(t), s->length(t)));
}
}
return s->hashCode();
}
inline uint16_t stringCharAt(Thread* t, GcString* s, int i)
{
if (objectClass(t, s->data()) == type(t, GcByteArray::Type)) {
return cast<GcByteArray>(t, s->data())->body()[s->offset(t) + i];
} else {
return cast<GcCharArray>(t, s->data())->body()[s->offset(t) + i];
}
}
inline bool stringEqual(Thread* t, object ao, object bo)
{
GcString* a = cast<GcString>(t, ao);
GcString* b = cast<GcString>(t, bo);
if (a == b) {
return true;
} else if (a->length(t) == b->length(t)) {
for (unsigned i = 0; i < a->length(t); ++i) {
if (stringCharAt(t, a, i) != stringCharAt(t, b, i)) {
return false;
}
}
return true;
} else {
return false;
}
}
inline uint32_t methodHash(Thread* t, object mo)
{
GcMethod* method = cast<GcMethod>(t, mo);
return byteArrayHash(t, method->name()) ^ byteArrayHash(t, method->spec());
}
inline bool methodEqual(Thread* t, object ao, object bo)
{
GcMethod* a = cast<GcMethod>(t, ao);
GcMethod* b = cast<GcMethod>(t, bo);
return a == b or (byteArrayEqual(t, a->name(), b->name())
and byteArrayEqual(t, a->spec(), b->spec()));
}
class MethodSpecIterator {
public:
MethodSpecIterator(Thread* t, const char* s) : t(t), s(s + 1)
{
}
const char* next()
{
assertT(t, *s != ')');
const char* p = s;
switch (*s) {
case 'L':
while (*s and *s != ';')
++s;
++s;
break;
case '[':
while (*s == '[')
++s;
switch (*s) {
case 'L':
while (*s and *s != ';')
++s;
++s;
break;
default:
++s;
break;
}
break;
default:
++s;
break;
}
return p;
}
bool hasNext()
{
return *s != ')';
}
const char* returnSpec()
{
assertT(t, *s == ')');
return s + 1;
}
Thread* t;
const char* s;
};
unsigned fieldCode(Thread* t, unsigned javaCode);
unsigned fieldType(Thread* t, unsigned code);
unsigned primitiveSize(Thread* t, unsigned code);
inline unsigned fieldSize(Thread* t, unsigned code)
{
if (code == ObjectField) {
return BytesPerWord;
} else {
return primitiveSize(t, code);
}
}
inline unsigned fieldSize(Thread* t, GcField* field)
{
return fieldSize(t, field->code());
}
inline void scanMethodSpec(Thread* t,
const char* s,
bool static_,
unsigned* parameterCount,
unsigned* parameterFootprint,
unsigned* returnCode)
{
unsigned count = 0;
unsigned footprint = 0;
MethodSpecIterator it(t, s);
while (it.hasNext()) {
++count;
switch (*it.next()) {
case 'J':
case 'D':
footprint += 2;
break;
default:
++footprint;
break;
}
}
if (not static_) {
++footprint;
}
*parameterCount = count;
*parameterFootprint = footprint;
*returnCode = fieldCode(t, *it.returnSpec());
}
GcClass* findLoadedClass(Thread* t, GcClassLoader* loader, GcByteArray* spec);
GcJclass* getDeclaringClass(Thread* t, GcClass* c);
GcCallSite* resolveDynamic(Thread* t, GcInvocation* invocation);
inline bool emptyMethod(Thread* t UNUSED, GcMethod* method)
{
return ((method->flags() & ACC_NATIVE) == 0)
and (method->code()->length() == 1)
and (method->code()->body()[0] == return_);
}
object parseUtf8(Thread* t, const char* data, unsigned length);
object parseUtf8(Thread* t, GcByteArray* array);
GcClass* parseClass(Thread* t,
GcClassLoader* loader,
const uint8_t* data,
unsigned length,
Gc::Type throwType = GcNoClassDefFoundError::Type);
GcClass* resolveClass(Thread* t,
GcClassLoader* loader,
GcByteArray* name,
bool throw_ = true,
Gc::Type throwType = GcNoClassDefFoundError::Type);
inline GcClass* resolveClass(Thread* t,
GcClassLoader* loader,
const char* name,
bool throw_ = true,
Gc::Type throwType = GcNoClassDefFoundError::Type)
{
PROTECT(t, loader);
GcByteArray* n = makeByteArray(t, "%s", name);
return resolveClass(t, loader, n, throw_, throwType);
}
GcClass* resolveSystemClass(Thread* t,
GcClassLoader* loader,
GcByteArray* name,
bool throw_ = true,
Gc::Type throwType = GcNoClassDefFoundError::Type);
inline GcClass* resolveSystemClass(Thread* t,
GcClassLoader* loader,
const char* name)
{
return resolveSystemClass(t, loader, makeByteArray(t, "%s", name));
}
void linkClass(Thread* t, GcClassLoader* loader, GcClass* class_);
GcMethod* resolveMethod(Thread* t,
GcClass* class_,
const char* methodName,
const char* methodSpec);
inline GcMethod* resolveMethod(Thread* t,
GcClassLoader* loader,
const char* className,
const char* methodName,
const char* methodSpec)
{
return resolveMethod(
t, resolveClass(t, loader, className), methodName, methodSpec);
}
GcField* resolveField(Thread* t,
GcClass* class_,
const char* fieldName,
const char* fieldSpec);
inline GcField* resolveField(Thread* t,
GcClassLoader* loader,
const char* className,
const char* fieldName,
const char* fieldSpec)
{
return resolveField(
t, resolveClass(t, loader, className), fieldName, fieldSpec);
}
bool classNeedsInit(Thread* t, GcClass* c);
bool preInitClass(Thread* t, GcClass* c);
void postInitClass(Thread* t, GcClass* c);
void initClass(Thread* t, GcClass* c);
GcClass* resolveObjectArrayClass(Thread* t,
GcClassLoader* loader,
GcClass* elementClass);
object makeObjectArray(Thread* t, GcClass* elementClass, unsigned count);
inline object makeObjectArray(Thread* t, unsigned count)
{
return makeObjectArray(t, type(t, GcJobject::Type), count);
}
object findFieldInClass(Thread* t,
GcClass* class_,
GcByteArray* name,
GcByteArray* spec);
inline GcField* findFieldInClass2(Thread* t,
GcClass* class_,
const char* name,
const char* spec)
{
PROTECT(t, class_);
GcByteArray* n = makeByteArray(t, "%s", name);
PROTECT(t, n);
GcByteArray* s = makeByteArray(t, "%s", spec);
return cast<GcField>(t, findFieldInClass(t, class_, n, s));
}
object findMethodInClass(Thread* t,
GcClass* class_,
GcByteArray* name,
GcByteArray* spec);
inline GcThrowable* makeThrowable(Thread* t,
Gc::Type type,
GcString* message = 0,
object trace = 0,
GcThrowable* cause = 0)
{
PROTECT(t, message);
PROTECT(t, trace);
PROTECT(t, cause);
if (trace == 0) {
trace = makeTrace(t);
}
GcThrowable* result = cast<GcThrowable>(t, make(t, vm::type(t, type)));
result->setMessage(t, message);
result->setTrace(t, trace);
result->setCause(t, cause);
return result;
}
inline GcThrowable* makeThrowableV(Thread* t,
Gc::Type type,
const char* format,
va_list a,
int size)
{
GcByteArray* s = makeByteArrayV(t, format, a, size);
if (s) {
GcString* message = t->m->classpath->makeString(t, s, 0, s->length() - 1);
return makeThrowable(t, type, message);
} else {
return 0;
}
}
inline GcThrowable* makeThrowable(Thread* t,
Gc::Type type,
const char* format,
...)
{
int size = 256;
while (true) {
va_list a;
va_start(a, format);
GcThrowable* r = makeThrowableV(t, type, format, a, size);
va_end(a);
if (r) {
return r;
} else {
size *= 2;
}
}
}
void popResources(Thread* t);
} // namespace vm
AVIAN_EXPORT void vmPrintTrace(vm::Thread* t);
AVIAN_EXPORT void vmfPrintTrace(vm::Thread* t, FILE* out);
namespace vm {
void dumpHeap(Thread* t, FILE* out);
inline void NO_RETURN throw_(Thread* t, GcThrowable* e)
{
assertT(t, t->exception == 0);
assertT(t, e);
expect(t, not t->checkpoint->noThrow);
t->exception = e;
if (objectClass(t, e) == type(t, GcOutOfMemoryError::Type)) {
if (not t->m->dumpedHeapOnOOM) {
t->m->dumpedHeapOnOOM = true;
const char* path = findProperty(t, "avian.heap.dump");
if (path) {
FILE* out = vm::fopen(path, "wb");
if (out) {
dumpHeap(t, out);
fclose(out);
}
}
}
if (AbortOnOutOfMemoryError) {
fprintf(stderr, "OutOfMemoryError\n");
vmPrintTrace(t);
abort();
}
}
// printTrace(t, e);
popResources(t);
t->checkpoint->unwind();
abort(t);
}
inline void NO_RETURN throwNew(Thread* t,
Gc::Type type,
GcString* message = 0,
object trace = 0,
GcThrowable* cause = 0)
{
throw_(t, makeThrowable(t, type, message, trace, cause));
}
inline void NO_RETURN
throwNew(Thread* t, Gc::Type type, const char* format, ...)
{
int size = 256;
while (true) {
va_list a;
va_start(a, format);
GcThrowable* r = makeThrowableV(t, type, format, a, size);
va_end(a);
if (r) {
throw_(t, r);
} else {
size *= 2;
}
}
}
object findInHierarchyOrNull(
Thread* t,
GcClass* class_,
GcByteArray* name,
GcByteArray* spec,
object (*find)(Thread*, GcClass*, GcByteArray*, GcByteArray*));
inline object findInHierarchy(
Thread* t,
GcClass* class_,
GcByteArray* name,
GcByteArray* spec,
object (*find)(Thread*, GcClass*, GcByteArray*, GcByteArray*),
Gc::Type errorType,
bool throw_ = true)
{
object o = findInHierarchyOrNull(t, class_, name, spec, find);
if (throw_ and o == 0) {
throwNew(t,
errorType,
"%s %s not found in %s",
name->body().begin(),
spec->body().begin(),
class_->name()->body().begin());
}
return o;
}
inline GcMethod* findMethod(Thread* t,
GcClass* class_,
GcByteArray* name,
GcByteArray* spec)
{
return cast<GcMethod>(
t,
findInHierarchy(
t, class_, name, spec, findMethodInClass, GcNoSuchMethodError::Type));
}
inline GcMethod* findMethodOrNull(Thread* t,
GcClass* class_,
const char* name,
const char* spec)
{
PROTECT(t, class_);
GcByteArray* n = makeByteArray(t, "%s", name);
PROTECT(t, n);
GcByteArray* s = makeByteArray(t, "%s", spec);
return cast<GcMethod>(
t, findInHierarchyOrNull(t, class_, n, s, findMethodInClass));
}
inline GcMethod* findVirtualMethod(Thread* t, GcMethod* method, GcClass* class_)
{
return cast<GcMethod>(
t, cast<GcArray>(t, class_->virtualTable())->body()[method->offset()]);
}
inline GcMethod* findInterfaceMethod(Thread* t,
GcMethod* method,
GcClass* class_)
{
if (UNLIKELY(class_->vmFlags() & BootstrapFlag)) {
PROTECT(t, method);
PROTECT(t, class_);
resolveSystemClass(t, roots(t)->bootLoader(), class_->name());
}
GcClass* interface = method->class_();
GcArray* itable = cast<GcArray>(t, class_->interfaceTable());
for (unsigned i = 0; i < itable->length(); i += 2) {
if (itable->body()[i] == interface) {
return cast<GcMethod>(
t, cast<GcArray>(t, itable->body()[i + 1])->body()[method->offset()]);
}
}
abort(t);
}
inline unsigned objectArrayLength(Thread* t UNUSED, object array)
{
assertT(t, objectClass(t, array)->fixedSize() == BytesPerWord * 2);
assertT(t, objectClass(t, array)->arrayElementSize() == BytesPerWord);
return fieldAtOffset<uintptr_t>(array, BytesPerWord);
}
inline object& objectArrayBody(Thread* t UNUSED, object array, unsigned index)
{
assertT(t, objectClass(t, array)->fixedSize() == BytesPerWord * 2);
assertT(t, objectClass(t, array)->arrayElementSize() == BytesPerWord);
assertT(
t,
objectClass(t, array)->objectMask()
== cast<GcClass>(t, t->m->types->body()[GcArray::Type])->objectMask());
return fieldAtOffset<object>(array, ArrayBody + (index * BytesPerWord));
}
unsigned parameterFootprint(Thread* t, const char* s, bool static_);
void addFinalizer(Thread* t, object target, void (*finalize)(Thread*, object));
inline bool acquireSystem(Thread* t, Thread* target)
{
ACQUIRE_RAW(t, t->m->stateLock);
if (t->state != Thread::JoinedState) {
target->setFlag(Thread::SystemFlag);
return true;
} else {
return false;
}
}
inline void releaseSystem(Thread* t, Thread* target)
{
ACQUIRE_RAW(t, t->m->stateLock);
assertT(t, t->state != Thread::JoinedState);
target->clearFlag(Thread::SystemFlag);
}
inline bool atomicCompareAndSwapObject(Thread* t,
object target,
unsigned offset,
object old,
object new_)
{
if (atomicCompareAndSwap(&fieldAtOffset<uintptr_t>(target, offset),
reinterpret_cast<uintptr_t>(old),
reinterpret_cast<uintptr_t>(new_))) {
mark(t, target, offset);
return true;
} else {
return false;
}
}
// The following two methods (monitorAtomicAppendAcquire and
// monitorAtomicPollAcquire) use the Michael and Scott Non-Blocking
// Queue Algorithm: http://www.cs.rochester.edu/u/michael/PODC96.html
inline void monitorAtomicAppendAcquire(Thread* t,
GcMonitor* monitor,
GcMonitorNode* node)
{
if (node == 0) {
PROTECT(t, monitor);
node = makeMonitorNode(t, t, 0);
}
while (true) {
GcMonitorNode* tail = cast<GcMonitorNode>(t, monitor->acquireTail());
loadMemoryBarrier();
object next = tail->next();
loadMemoryBarrier();
if (tail == cast<GcMonitorNode>(t, monitor->acquireTail())) {
if (next) {
atomicCompareAndSwapObject(t, monitor, MonitorAcquireTail, tail, next);
} else if (atomicCompareAndSwapObject(
t, tail, MonitorNodeNext, 0, node)) {
atomicCompareAndSwapObject(t, monitor, MonitorAcquireTail, tail, node);
return;
}
}
}
}
inline Thread* monitorAtomicPollAcquire(Thread* t,
GcMonitor* monitor,
bool remove)
{
while (true) {
GcMonitorNode* head = cast<GcMonitorNode>(t, monitor->acquireHead());
loadMemoryBarrier();
GcMonitorNode* tail = cast<GcMonitorNode>(t, monitor->acquireTail());
loadMemoryBarrier();
GcMonitorNode* next = cast<GcMonitorNode>(t, head->next());
loadMemoryBarrier();
if (head == cast<GcMonitorNode>(t, monitor->acquireHead())) {
if (head == tail) {
if (next) {
atomicCompareAndSwapObject(
t, monitor, MonitorAcquireTail, tail, next);
} else {
return 0;
}
} else {
Thread* value = static_cast<Thread*>(next->value());
if ((not remove) or atomicCompareAndSwapObject(
t, monitor, MonitorAcquireHead, head, next)) {
return value;
}
}
}
}
}
inline bool monitorTryAcquire(Thread* t, GcMonitor* monitor)
{
if (monitor->owner() == t
or (monitorAtomicPollAcquire(t, monitor, false) == 0
and atomicCompareAndSwap(
reinterpret_cast<uintptr_t*>(&monitor->owner()),
0,
reinterpret_cast<uintptr_t>(t)))) {
++monitor->depth();
return true;
} else {
return false;
}
}
inline void monitorAcquire(Thread* t,
GcMonitor* monitor,
GcMonitorNode* node = 0)
{
if (not monitorTryAcquire(t, monitor)) {
PROTECT(t, monitor);
PROTECT(t, node);
ACQUIRE(t, t->lock);
monitorAtomicAppendAcquire(t, monitor, node);
// note that we don't try to acquire the lock until we're first in
// line, both because it's fair and because we don't support
// removing elements from arbitrary positions in the queue
while (not(t == monitorAtomicPollAcquire(t, monitor, false)
and atomicCompareAndSwap(
reinterpret_cast<uintptr_t*>(&monitor->owner()),
0,
reinterpret_cast<uintptr_t>(t)))) {
ENTER(t, Thread::IdleState);
t->lock->wait(t->systemThread, 0);
}
expect(t, t == monitorAtomicPollAcquire(t, monitor, true));
++monitor->depth();
}
assertT(t, monitor->owner() == t);
}
inline void monitorRelease(Thread* t, GcMonitor* monitor)
{
expect(t, monitor->owner() == t);
if (--monitor->depth() == 0) {
monitor->owner() = 0;
storeLoadMemoryBarrier();
Thread* next = monitorAtomicPollAcquire(t, monitor, false);
if (next and acquireSystem(t, next)) {
ACQUIRE(t, next->lock);
next->lock->notify(t->systemThread);
releaseSystem(t, next);
}
}
}
inline void monitorAppendWait(Thread* t, GcMonitor* monitor)
{
assertT(t, monitor->owner() == t);
expect(t, (t->getFlags() & Thread::WaitingFlag) == 0);
expect(t, t->waitNext == 0);
t->setFlag(Thread::WaitingFlag);
if (monitor->waitTail()) {
static_cast<Thread*>(monitor->waitTail())->waitNext = t;
} else {
monitor->waitHead() = t;
}
monitor->waitTail() = t;
}
inline void monitorRemoveWait(Thread* t, GcMonitor* monitor)
{
assertT(t, monitor->owner() == t);
Thread* previous = 0;
for (Thread* current = static_cast<Thread*>(monitor->waitHead()); current;
current = current->waitNext) {
if (t == current) {
if (t == monitor->waitHead()) {
monitor->waitHead() = t->waitNext;
} else {
previous->waitNext = t->waitNext;
}
if (t == monitor->waitTail()) {
assertT(t, t->waitNext == 0);
monitor->waitTail() = previous;
}
t->waitNext = 0;
t->clearFlag(Thread::WaitingFlag);
return;
} else {
previous = current;
}
}
abort(t);
}
inline bool monitorFindWait(Thread* t, GcMonitor* monitor)
{
assertT(t, monitor->owner() == t);
for (Thread* current = static_cast<Thread*>(monitor->waitHead()); current;
current = current->waitNext) {
if (t == current) {
return true;
}
}
return false;
}
inline bool monitorWait(Thread* t, GcMonitor* monitor, int64_t time)
{
expect(t, monitor->owner() == t);
bool interrupted;
unsigned depth;
PROTECT(t, monitor);
// pre-allocate monitor node so we don't get an OutOfMemoryError
// when we try to re-acquire the monitor below
GcMonitorNode* monitorNode = makeMonitorNode(t, t, 0);
PROTECT(t, monitorNode);
{
ACQUIRE(t, t->lock);
monitorAppendWait(t, monitor);
depth = monitor->depth();
monitor->depth() = 1;
monitorRelease(t, monitor);
ENTER(t, Thread::IdleState);
interrupted = t->lock->waitAndClearInterrupted(t->systemThread, time);
}
monitorAcquire(t, monitor, monitorNode);
monitor->depth() = depth;
if (t->getFlags() & Thread::WaitingFlag) {
monitorRemoveWait(t, monitor);
} else {
expect(t, not monitorFindWait(t, monitor));
}
assertT(t, monitor->owner() == t);
return interrupted;
}
inline Thread* monitorPollWait(Thread* t UNUSED, GcMonitor* monitor)
{
assertT(t, monitor->owner() == t);
Thread* next = static_cast<Thread*>(monitor->waitHead());
if (next) {
monitor->waitHead() = next->waitNext;
next->clearFlag(Thread::WaitingFlag);
next->waitNext = 0;
if (next == monitor->waitTail()) {
monitor->waitTail() = 0;
}
} else {
assertT(t, monitor->waitTail() == 0);
}
return next;
}
inline bool monitorNotify(Thread* t, GcMonitor* monitor)
{
expect(t, monitor->owner() == t);
Thread* next = monitorPollWait(t, monitor);
if (next) {
ACQUIRE(t, next->lock);
next->lock->notify(t->systemThread);
return true;
} else {
return false;
}
}
inline void monitorNotifyAll(Thread* t, GcMonitor* monitor)
{
PROTECT(t, monitor);
while (monitorNotify(t, monitor)) {
}
}
class ObjectMonitorResource {
public:
ObjectMonitorResource(Thread* t, GcMonitor* o)
: o(o), protector(t, &(this->o))
{
monitorAcquire(protector.t, o);
}
~ObjectMonitorResource()
{
monitorRelease(protector.t, o);
}
private:
GcMonitor* o;
Thread::SingleProtector protector;
};
GcMonitor* objectMonitor(Thread* t, object o, bool createNew);
inline void acquire(Thread* t, object o)
{
unsigned hash;
if (DebugMonitors) {
hash = objectHash(t, o);
}
GcMonitor* m = objectMonitor(t, o, true);
if (DebugMonitors) {
fprintf(stderr, "thread %p acquires %p for %x\n", t, m, hash);
}
monitorAcquire(t, m);
}
inline void release(Thread* t, object o)
{
unsigned hash;
if (DebugMonitors) {
hash = objectHash(t, o);
}
GcMonitor* m = objectMonitor(t, o, false);
if (DebugMonitors) {
fprintf(stderr, "thread %p releases %p for %x\n", t, m, hash);
}
monitorRelease(t, m);
}
inline void wait(Thread* t, object o, int64_t milliseconds)
{
unsigned hash;
if (DebugMonitors) {
hash = objectHash(t, o);
}
GcMonitor* m = objectMonitor(t, o, false);
if (DebugMonitors) {
fprintf(stderr,
"thread %p waits %d millis on %p for %x\n",
t,
static_cast<int>(milliseconds),
m,
hash);
}
if (m and m->owner() == t) {
PROTECT(t, m);
bool interrupted = monitorWait(t, m, milliseconds);
if (interrupted) {
if (t->m->alive or (t->getFlags() & Thread::DaemonFlag) == 0) {
t->m->classpath->clearInterrupted(t);
throwNew(t, GcInterruptedException::Type);
} else {
throw_(t, roots(t)->shutdownInProgress());
}
}
} else {
throwNew(t, GcIllegalMonitorStateException::Type);
}
if (DebugMonitors) {
fprintf(stderr, "thread %p wakes up on %p for %x\n", t, m, hash);
}
stress(t);
}
inline void notify(Thread* t, object o)
{
unsigned hash;
if (DebugMonitors) {
hash = objectHash(t, o);
}
GcMonitor* m = objectMonitor(t, o, false);
if (DebugMonitors) {
fprintf(stderr, "thread %p notifies on %p for %x\n", t, m, hash);
}
if (m and m->owner() == t) {
monitorNotify(t, m);
} else {
throwNew(t, GcIllegalMonitorStateException::Type);
}
}
inline void notifyAll(Thread* t, object o)
{
GcMonitor* m = objectMonitor(t, o, false);
if (DebugMonitors) {
fprintf(stderr,
"thread %p notifies all on %p for %x\n",
t,
m,
objectHash(t, o));
}
if (m and m->owner() == t) {
monitorNotifyAll(t, m);
} else {
throwNew(t, GcIllegalMonitorStateException::Type);
}
}
inline void interrupt(Thread* t, Thread* target)
{
if (acquireSystem(t, target)) {
target->systemThread->interrupt();
releaseSystem(t, target);
}
}
inline bool getAndClearInterrupted(Thread* t, Thread* target)
{
if (acquireSystem(t, target)) {
bool result = target->systemThread->getAndClearInterrupted();
releaseSystem(t, target);
return result;
} else {
return false;
}
}
inline bool exceptionMatch(Thread* t, GcClass* type, GcThrowable* exception)
{
return type == 0 or (exception != roots(t)->shutdownInProgress()
and instanceOf(t, type, t->exception));
}
object intern(Thread* t, object s);
object clone(Thread* t, object o);
void walk(Thread* t, Heap::Walker* w, object o, unsigned start);
int walkNext(Thread* t, object o, int previous);
void visitRoots(Machine* m, Heap::Visitor* v);
inline jobject makeLocalReference(Thread* t, object o)
{
return t->m->processor->makeLocalReference(t, o);
}
inline void disposeLocalReference(Thread* t, jobject r)
{
t->m->processor->disposeLocalReference(t, r);
}
inline bool methodVirtual(Thread* t UNUSED, GcMethod* method)
{
return (method->flags() & (ACC_STATIC | ACC_PRIVATE)) == 0
and method->name()->body()[0] != '<';
}
inline unsigned singletonMaskSize(unsigned count, unsigned bitsPerWord)
{
if (count) {
return ceilingDivide(count + 2, bitsPerWord);
}
return 0;
}
inline unsigned singletonMaskSize(unsigned count)
{
return singletonMaskSize(count, BitsPerWord);
}
inline unsigned singletonMaskSize(Thread* t UNUSED, GcSingleton* singleton)
{
unsigned length = singleton->length();
if (length) {
return ceilingDivide(length + 2, BitsPerWord + 1);
}
return 0;
}
inline unsigned singletonCount(Thread* t, GcSingleton* singleton)
{
return singleton->length() - singletonMaskSize(t, singleton);
}
inline uint32_t* singletonMask(Thread* t UNUSED, GcSingleton* singleton)
{
assertT(t, singleton->length());
return reinterpret_cast<uint32_t*>(
&singleton->body()[singletonCount(t, singleton)]);
}
inline void singletonMarkObject(uint32_t* mask, unsigned index)
{
mask[(index + 2) / 32] |= (static_cast<uint32_t>(1) << ((index + 2) % 32));
}
inline void singletonMarkObject(Thread* t,
GcSingleton* singleton,
unsigned index)
{
singletonMarkObject(singletonMask(t, singleton), index);
}
inline bool singletonIsObject(Thread* t, GcSingleton* singleton, unsigned index)
{
assertT(t, index < singletonCount(t, singleton));
return (singletonMask(t, singleton)[(index + 2) / 32]
& (static_cast<uint32_t>(1) << ((index + 2) % 32))) != 0;
}
inline object& singletonObject(Thread* t UNUSED,
GcSingleton* singleton,
unsigned index)
{
assertT(t, singletonIsObject(t, singleton, index));
return reinterpret_cast<object&>(singleton->body()[index]);
}
inline uintptr_t& singletonValue(Thread* t UNUSED,
GcSingleton* singleton,
unsigned index)
{
assertT(t, not singletonIsObject(t, singleton, index));
return singleton->body()[index];
}
inline GcSingleton* makeSingletonOfSize(Thread* t, unsigned count)
{
GcSingleton* o = makeSingleton(t, count + singletonMaskSize(count));
assertT(t, o->length() == count + singletonMaskSize(t, o));
if (count) {
singletonMask(t, o)[0] = 1;
}
return o;
}
inline void singletonSetBit(Thread* t,
GcSingleton* singleton,
unsigned start,
unsigned index)
{
singletonValue(t, singleton, start + (index / BitsPerWord))
|= static_cast<uintptr_t>(1) << (index % BitsPerWord);
}
inline bool singletonBit(Thread* t,
GcSingleton* singleton,
unsigned start,
unsigned index)
{
return (singletonValue(t, singleton, start + (index / BitsPerWord))
& (static_cast<uintptr_t>(1) << (index % BitsPerWord))) != 0;
}
inline unsigned poolMaskSize(unsigned count, unsigned bitsPerWord)
{
return ceilingDivide(count, bitsPerWord);
}
inline unsigned poolMaskSize(unsigned count)
{
return poolMaskSize(count, BitsPerWord);
}
inline unsigned poolMaskSize(Thread* t, GcSingleton* pool)
{
return ceilingDivide(singletonCount(t, pool), BitsPerWord + 1);
}
inline unsigned poolSize(Thread* t, GcSingleton* pool)
{
return singletonCount(t, pool) - poolMaskSize(t, pool);
}
inline GcClass* resolveClassInObject(Thread* t,
GcClassLoader* loader,
object container,
unsigned classOffset,
bool throw_ = true)
{
object o = fieldAtOffset<object>(container, classOffset);
loadMemoryBarrier();
if (objectClass(t, o) == type(t, GcByteArray::Type)) {
GcByteArray* name = cast<GcByteArray>(t, o);
PROTECT(t, container);
GcClass* c = resolveClass(t, loader, name, throw_);
if (c) {
storeStoreMemoryBarrier();
setField(t, container, classOffset, c);
}
return c;
}
return cast<GcClass>(t, o);
}
inline GcClass* resolveClassInPool(Thread* t,
GcClassLoader* loader,
GcMethod* method,
unsigned index,
bool throw_ = true)
{
object o = singletonObject(t, method->code()->pool(), index);
loadMemoryBarrier();
if (objectClass(t, o) == type(t, GcReference::Type)) {
PROTECT(t, method);
GcClass* c
= resolveClass(t, loader, cast<GcReference>(t, o)->name(), throw_);
if (c) {
storeStoreMemoryBarrier();
method->code()->pool()->setBodyElement(
t, index, reinterpret_cast<uintptr_t>(c));
}
return c;
}
return cast<GcClass>(t, o);
}
inline GcClass* resolveClassInPool(Thread* t,
GcMethod* method,
unsigned index,
bool throw_ = true)
{
return resolveClassInPool(
t, method->class_()->loader(), method, index, throw_);
}
inline object resolve(
Thread* t,
GcClassLoader* loader,
GcSingleton* pool,
unsigned index,
object (*find)(vm::Thread*, GcClass*, GcByteArray*, GcByteArray*),
Gc::Type errorType,
bool throw_ = true)
{
object o = singletonObject(t, pool, index);
loadMemoryBarrier();
if (objectClass(t, o) == type(t, GcReference::Type)) {
PROTECT(t, pool);
GcReference* reference = cast<GcReference>(t, o);
PROTECT(t, reference);
GcClass* class_
= resolveClassInObject(t, loader, o, ReferenceClass, throw_);
if (class_) {
o = findInHierarchy(t,
class_,
reference->name(),
reference->spec(),
find,
errorType,
throw_);
if (o) {
storeStoreMemoryBarrier();
pool->setBodyElement(t, index, reinterpret_cast<uintptr_t>(o));
}
} else {
o = 0;
}
}
return o;
}
inline GcField* resolveField(Thread* t,
GcClassLoader* loader,
GcMethod* method,
unsigned index,
bool throw_ = true)
{
return cast<GcField>(t,
resolve(t,
loader,
method->code()->pool(),
index,
findFieldInClass,
GcNoSuchFieldError::Type,
throw_));
}
inline GcField* resolveField(Thread* t,
GcMethod* method,
unsigned index,
bool throw_ = true)
{
return resolveField(t, method->class_()->loader(), method, index, throw_);
}
inline void acquireFieldForRead(Thread* t, GcField* field)
{
if (UNLIKELY(
(field->flags() & ACC_VOLATILE) and BytesPerWord == 4
and (field->code() == DoubleField or field->code() == LongField))) {
acquire(t, field);
}
}
inline void releaseFieldForRead(Thread* t, GcField* field)
{
if (UNLIKELY(field->flags() & ACC_VOLATILE)) {
if (BytesPerWord == 4
and (field->code() == DoubleField or field->code() == LongField)) {
release(t, field);
} else {
loadMemoryBarrier();
}
}
}
class FieldReadResource {
public:
FieldReadResource(Thread* t, GcField* o) : o(o), protector(t, &(this->o))
{
acquireFieldForRead(protector.t, o);
}
~FieldReadResource()
{
releaseFieldForRead(protector.t, o);
}
private:
GcField* o;
Thread::SingleProtector protector;
};
inline void acquireFieldForWrite(Thread* t, GcField* field)
{
if (UNLIKELY(field->flags() & ACC_VOLATILE)) {
if (BytesPerWord == 4
and (field->code() == DoubleField or field->code() == LongField)) {
acquire(t, field);
} else {
storeStoreMemoryBarrier();
}
}
}
inline void releaseFieldForWrite(Thread* t, GcField* field)
{
if (UNLIKELY(field->flags() & ACC_VOLATILE)) {
if (BytesPerWord == 4
and (field->code() == DoubleField or field->code() == LongField)) {
release(t, field);
} else {
storeLoadMemoryBarrier();
}
}
}
class FieldWriteResource {
public:
FieldWriteResource(Thread* t, GcField* o) : o(o), protector(t, &(this->o))
{
acquireFieldForWrite(protector.t, o);
}
~FieldWriteResource()
{
releaseFieldForWrite(protector.t, o);
}
private:
GcField* o;
Thread::SingleProtector protector;
};
inline GcMethod* resolveMethod(Thread* t,
GcClassLoader* loader,
GcMethod* method,
unsigned index,
bool throw_ = true)
{
return cast<GcMethod>(t,
resolve(t,
loader,
method->code()->pool(),
index,
findMethodInClass,
GcNoSuchMethodError::Type,
throw_));
}
inline GcMethod* resolveMethod(Thread* t,
GcMethod* method,
unsigned index,
bool throw_ = true)
{
return resolveMethod(t, method->class_()->loader(), method, index, throw_);
}
GcVector* vectorAppend(Thread*, GcVector*, object);
inline GcClassRuntimeData* getClassRuntimeDataIfExists(Thread* t, GcClass* c)
{
if (c->runtimeDataIndex()) {
return cast<GcClassRuntimeData>(
t,
roots(t)->classRuntimeDataTable()->body()[c->runtimeDataIndex() - 1]);
} else {
return 0;
}
}
inline GcClassRuntimeData* getClassRuntimeData(Thread* t, GcClass* c)
{
if (c->runtimeDataIndex() == 0) {
PROTECT(t, c);
ACQUIRE(t, t->m->classLock);
if (c->runtimeDataIndex() == 0) {
GcClassRuntimeData* runtimeData = makeClassRuntimeData(t, 0, 0, 0, 0);
{
GcVector* v
= vectorAppend(t, roots(t)->classRuntimeDataTable(), runtimeData);
// sequence point, for gc (don't recombine statements)
roots(t)->setClassRuntimeDataTable(t, v);
}
c->runtimeDataIndex() = roots(t)->classRuntimeDataTable()->size();
}
}
return cast<GcClassRuntimeData>(
t, roots(t)->classRuntimeDataTable()->body()[c->runtimeDataIndex() - 1]);
}
inline GcMethodRuntimeData* getMethodRuntimeData(Thread* t, GcMethod* method)
{
int index = method->runtimeDataIndex();
loadMemoryBarrier();
if (index == 0) {
PROTECT(t, method);
ACQUIRE(t, t->m->classLock);
if (method->runtimeDataIndex() == 0) {
GcMethodRuntimeData* runtimeData = makeMethodRuntimeData(t, 0);
{
GcVector* v
= vectorAppend(t, roots(t)->methodRuntimeDataTable(), runtimeData);
// sequence point, for gc (don't recombine statements)
roots(t)->setMethodRuntimeDataTable(t, v);
}
storeStoreMemoryBarrier();
method->runtimeDataIndex() = roots(t)->methodRuntimeDataTable()->size();
}
}
return cast<GcMethodRuntimeData>(t,
roots(t)->methodRuntimeDataTable()->body()
[method->runtimeDataIndex() - 1]);
}
inline GcJclass* getJClass(Thread* t, GcClass* c)
{
PROTECT(t, c);
GcJclass* jclass = cast<GcJclass>(t, getClassRuntimeData(t, c)->jclass());
loadMemoryBarrier();
if (jclass == 0) {
ACQUIRE(t, t->m->classLock);
jclass = cast<GcJclass>(t, getClassRuntimeData(t, c)->jclass());
if (jclass == 0) {
jclass = t->m->classpath->makeJclass(t, c);
storeStoreMemoryBarrier();
getClassRuntimeData(t, c)->setJclass(t, jclass);
}
}
return jclass;
}
inline GcClass* primitiveClass(Thread* t, char name)
{
switch (name) {
case 'B':
return type(t, GcJbyte::Type);
case 'C':
return type(t, GcJchar::Type);
case 'D':
return type(t, GcJdouble::Type);
case 'F':
return type(t, GcJfloat::Type);
case 'I':
return type(t, GcJint::Type);
case 'J':
return type(t, GcJlong::Type);
case 'S':
return type(t, GcJshort::Type);
case 'V':
return type(t, GcJvoid::Type);
case 'Z':
return type(t, GcJboolean::Type);
default:
throwNew(t, GcIllegalArgumentException::Type);
}
}
inline void registerNative(Thread* t, GcMethod* method, void* function)
{
PROTECT(t, method);
expect(t, method->flags() & ACC_NATIVE);
GcNative* native = makeNative(t, function, false);
PROTECT(t, native);
GcMethodRuntimeData* runtimeData = getMethodRuntimeData(t, method);
// ensure other threads only see the methodRuntimeDataNative field
// populated once the object it points to has been populated:
storeStoreMemoryBarrier();
runtimeData->setNative(t, native);
}
inline void unregisterNatives(Thread* t, GcClass* c)
{
GcArray* table = cast<GcArray>(t, c->methodTable());
if (table) {
for (unsigned i = 0; i < table->length(); ++i) {
GcMethod* method = cast<GcMethod>(t, table->body()[i]);
if (method->flags() & ACC_NATIVE) {
getMethodRuntimeData(t, method)->setNative(t, 0);
}
}
}
}
void populateMultiArray(Thread* t,
object array,
int32_t* counts,
unsigned index,
unsigned dimensions);
GcMethod* getCaller(Thread* t, unsigned target, bool skipMethodInvoke = false);
GcClass* defineClass(Thread* t,
GcClassLoader* loader,
const uint8_t* buffer,
unsigned length);
inline GcMethod* methodClone(Thread* t, GcMethod* method)
{
return makeMethod(t,
method->vmFlags(),
method->returnCode(),
method->parameterCount(),
method->parameterFootprint(),
method->flags(),
method->offset(),
method->nativeID(),
method->runtimeDataIndex(),
method->name(),
method->spec(),
method->addendum(),
method->class_(),
method->code());
}
inline uint64_t exceptionHandler(uint64_t start,
uint64_t end,
uint64_t ip,
uint64_t catchType)
{
return (start << 48) | (end << 32) | (ip << 16) | catchType;
}
inline unsigned exceptionHandlerStart(uint64_t eh)
{
return eh >> 48;
}
inline unsigned exceptionHandlerEnd(uint64_t eh)
{
return (eh >> 32) & 0xFFFF;
}
inline unsigned exceptionHandlerIp(uint64_t eh)
{
return (eh >> 16) & 0xFFFF;
}
inline unsigned exceptionHandlerCatchType(uint64_t eh)
{
return eh & 0xFFFF;
}
inline uint64_t lineNumber(uint64_t ip, uint64_t line)
{
return (ip << 32) | line;
}
inline unsigned lineNumberIp(uint64_t ln)
{
return ln >> 32;
}
inline unsigned lineNumberLine(uint64_t ln)
{
return ln & 0xFFFFFFFF;
}
object interruptLock(Thread* t, GcThread* thread);
void clearInterrupted(Thread* t);
void threadInterrupt(Thread* t, GcThread* thread);
bool threadIsInterrupted(Thread* t, GcThread* thread, bool clear);
inline FILE* errorLog(Thread* t)
{
if (t->m->errorLog == 0) {
const char* path = findProperty(t, "avian.error.log");
if (path) {
t->m->errorLog = vm::fopen(path, "wb");
} else {
t->m->errorLog = stderr;
}
}
return t->m->errorLog;
}
} // namespace vm
AVIAN_EXPORT void* vmAddressFromLine(vm::Thread* t,
vm::object m,
unsigned line);
#endif // MACHINE_H