2009-03-15 18:02:36 +00:00
|
|
|
/* Copyright (c) 2008-2009, Avian Contributors
|
2008-02-19 18:06:52 +00:00
|
|
|
|
|
|
|
Permission to use, copy, modify, and/or distribute this software
|
|
|
|
for any purpose with or without fee is hereby granted, provided
|
|
|
|
that the above copyright notice and this permission notice appear
|
|
|
|
in all copies.
|
|
|
|
|
|
|
|
There is NO WARRANTY for this software. See license.txt for
|
|
|
|
details. */
|
|
|
|
|
2007-07-06 23:50:26 +00:00
|
|
|
#ifndef MACHINE_H
|
|
|
|
#define MACHINE_H
|
2007-06-25 01:34:07 +00:00
|
|
|
|
2007-07-06 15:24:06 +00:00
|
|
|
#include "common.h"
|
|
|
|
#include "system.h"
|
|
|
|
#include "heap.h"
|
2007-08-10 23:45:47 +00:00
|
|
|
#include "finder.h"
|
2007-09-24 01:39:03 +00:00
|
|
|
#include "processor.h"
|
2007-11-05 14:28:46 +00:00
|
|
|
#include "constants.h"
|
2010-02-05 00:56:21 +00:00
|
|
|
#include "arch.h"
|
2007-06-25 01:34:07 +00:00
|
|
|
|
2009-08-27 00:26:44 +00:00
|
|
|
#ifdef PLATFORM_WINDOWS
|
2007-10-24 17:24:19 +00:00
|
|
|
# define JNICALL __stdcall
|
|
|
|
#else
|
|
|
|
# define JNICALL
|
|
|
|
#endif
|
2007-06-25 01:34:07 +00:00
|
|
|
|
2007-07-06 15:24:06 +00:00
|
|
|
#define PROTECT(thread, name) \
|
2007-10-12 17:56:43 +00:00
|
|
|
Thread::SingleProtector MAKE_NAME(protector_) (thread, &name);
|
2007-07-06 15:24:06 +00:00
|
|
|
|
|
|
|
#define ACQUIRE(t, x) MonitorResource MAKE_NAME(monitorResource_) (t, x)
|
|
|
|
|
2010-09-17 01:43:27 +00:00
|
|
|
#define ACQUIRE_OBJECT(t, x) \
|
|
|
|
ObjectMonitorResource MAKE_NAME(monitorResource_) (t, x)
|
|
|
|
|
2007-07-06 15:24:06 +00:00
|
|
|
#define ACQUIRE_RAW(t, x) RawMonitorResource MAKE_NAME(monitorResource_) (t, x)
|
|
|
|
|
|
|
|
#define ENTER(t, state) StateResource MAKE_NAME(stateResource_) (t, state)
|
|
|
|
|
2007-06-25 02:02:24 +00:00
|
|
|
namespace vm {
|
|
|
|
|
2007-07-06 15:24:06 +00:00
|
|
|
const bool Verbose = false;
|
2007-07-27 23:56:19 +00:00
|
|
|
const bool DebugRun = false;
|
2007-07-10 03:04:49 +00:00
|
|
|
const bool DebugStack = false;
|
2007-11-02 21:42:19 +00:00
|
|
|
const bool DebugMonitors = false;
|
2007-11-05 15:39:48 +00:00
|
|
|
const bool DebugReferences = false;
|
2007-07-06 15:24:06 +00:00
|
|
|
|
|
|
|
const uintptr_t HashTakenMark = 1;
|
|
|
|
const uintptr_t ExtendedMark = 2;
|
2007-10-22 17:22:30 +00:00
|
|
|
const uintptr_t FixedMark = 3;
|
2007-07-06 15:24:06 +00:00
|
|
|
|
2008-11-22 21:47:18 +00:00
|
|
|
const unsigned ThreadHeapSizeInBytes = 64 * 1024;
|
|
|
|
const unsigned ThreadHeapSizeInWords = ThreadHeapSizeInBytes / BytesPerWord;
|
|
|
|
|
2010-06-19 22:40:21 +00:00
|
|
|
const unsigned ThreadBackupHeapSizeInBytes = 2 * 1024;
|
|
|
|
const unsigned ThreadBackupHeapSizeInWords
|
|
|
|
= ThreadBackupHeapSizeInBytes / BytesPerWord;
|
|
|
|
|
2008-11-22 21:48:10 +00:00
|
|
|
const unsigned ThreadHeapPoolSize = 64;
|
2008-11-22 21:47:18 +00:00
|
|
|
|
|
|
|
const unsigned FixedFootprintThresholdInBytes
|
|
|
|
= ThreadHeapPoolSize * ThreadHeapSizeInBytes;
|
|
|
|
|
2007-07-06 15:24:06 +00:00
|
|
|
enum FieldCode {
|
|
|
|
VoidField,
|
|
|
|
ByteField,
|
|
|
|
CharField,
|
|
|
|
DoubleField,
|
|
|
|
FloatField,
|
|
|
|
IntField,
|
|
|
|
LongField,
|
|
|
|
ShortField,
|
|
|
|
BooleanField,
|
|
|
|
ObjectField
|
|
|
|
};
|
|
|
|
|
|
|
|
enum StackTag {
|
|
|
|
IntTag, // must be zero
|
|
|
|
ObjectTag
|
|
|
|
};
|
|
|
|
|
|
|
|
const int NativeLine = -1;
|
|
|
|
const int UnknownLine = -2;
|
|
|
|
|
2009-05-03 20:57:11 +00:00
|
|
|
// class vmFlags:
|
2007-07-20 03:18:25 +00:00
|
|
|
const unsigned ReferenceFlag = 1 << 0;
|
|
|
|
const unsigned WeakReferenceFlag = 1 << 1;
|
2007-07-28 16:10:13 +00:00
|
|
|
const unsigned NeedInitFlag = 1 << 2;
|
2007-08-01 23:48:36 +00:00
|
|
|
const unsigned InitFlag = 1 << 3;
|
2009-07-20 20:12:38 +00:00
|
|
|
const unsigned InitErrorFlag = 1 << 4;
|
|
|
|
const unsigned PrimitiveFlag = 1 << 5;
|
|
|
|
const unsigned BootstrapFlag = 1 << 6;
|
2009-07-22 00:57:55 +00:00
|
|
|
const unsigned HasFinalizerFlag = 1 << 7;
|
2009-08-18 20:26:28 +00:00
|
|
|
const unsigned LinkFlag = 1 << 8;
|
|
|
|
const unsigned HasFinalMemberFlag = 1 << 9;
|
|
|
|
const unsigned SingletonFlag = 1 << 10;
|
|
|
|
const unsigned ContinuationFlag = 1 << 11;
|
2007-08-01 23:48:36 +00:00
|
|
|
|
2009-05-03 20:57:11 +00:00
|
|
|
// method vmFlags:
|
2007-08-01 23:48:36 +00:00
|
|
|
const unsigned ClassInitFlag = 1 << 0;
|
2007-09-25 23:53:11 +00:00
|
|
|
const unsigned CompiledFlag = 1 << 1;
|
2009-03-03 03:18:15 +00:00
|
|
|
const unsigned ConstructorFlag = 1 << 2;
|
2007-07-20 03:18:25 +00:00
|
|
|
|
2009-06-22 22:25:13 +00:00
|
|
|
#ifndef JNI_VERSION_1_6
|
|
|
|
#define JNI_VERSION_1_6 0x00010006
|
|
|
|
#endif
|
|
|
|
|
2007-09-10 23:33:58 +00:00
|
|
|
typedef Machine JavaVM;
|
2007-07-06 15:24:06 +00:00
|
|
|
typedef Thread JNIEnv;
|
|
|
|
|
2007-06-25 01:34:07 +00:00
|
|
|
typedef uint8_t jboolean;
|
|
|
|
typedef int8_t jbyte;
|
|
|
|
typedef uint16_t jchar;
|
|
|
|
typedef int16_t jshort;
|
|
|
|
typedef int32_t jint;
|
|
|
|
typedef int64_t jlong;
|
|
|
|
typedef float jfloat;
|
|
|
|
typedef double jdouble;
|
|
|
|
|
|
|
|
typedef jint jsize;
|
|
|
|
|
2007-09-07 00:21:52 +00:00
|
|
|
typedef object* jobject;
|
2007-06-25 01:34:07 +00:00
|
|
|
|
|
|
|
typedef jobject jclass;
|
|
|
|
typedef jobject jthrowable;
|
|
|
|
typedef jobject jstring;
|
|
|
|
typedef jobject jweak;
|
|
|
|
|
|
|
|
typedef jobject jarray;
|
|
|
|
typedef jarray jbooleanArray;
|
|
|
|
typedef jarray jbyteArray;
|
|
|
|
typedef jarray jcharArray;
|
|
|
|
typedef jarray jshortArray;
|
|
|
|
typedef jarray jintArray;
|
|
|
|
typedef jarray jlongArray;
|
|
|
|
typedef jarray jfloatArray;
|
|
|
|
typedef jarray jdoubleArray;
|
|
|
|
typedef jarray jobjectArray;
|
|
|
|
|
2007-09-07 00:21:52 +00:00
|
|
|
typedef uintptr_t jfieldID;
|
|
|
|
typedef uintptr_t jmethodID;
|
2007-06-25 01:34:07 +00:00
|
|
|
|
|
|
|
union jvalue {
|
|
|
|
jboolean z;
|
|
|
|
jbyte b;
|
|
|
|
jchar c;
|
|
|
|
jshort s;
|
|
|
|
jint i;
|
|
|
|
jlong j;
|
|
|
|
jfloat f;
|
|
|
|
jdouble d;
|
|
|
|
jobject l;
|
|
|
|
};
|
|
|
|
|
|
|
|
struct JNINativeMethod {
|
|
|
|
char* name;
|
|
|
|
char* signature;
|
|
|
|
void* function;
|
|
|
|
};
|
|
|
|
|
2007-06-25 02:02:24 +00:00
|
|
|
struct JavaVMVTable {
|
2007-06-25 01:34:07 +00:00
|
|
|
void* reserved0;
|
|
|
|
void* reserved1;
|
|
|
|
void* reserved2;
|
|
|
|
|
2008-06-15 20:17:52 +00:00
|
|
|
#if (! TARGET_RT_MAC_CFM) && defined(__ppc__)
|
|
|
|
void* cfm_vectors[4];
|
|
|
|
#endif
|
|
|
|
|
2007-06-25 01:34:07 +00:00
|
|
|
jint
|
|
|
|
(JNICALL *DestroyJavaVM)
|
|
|
|
(JavaVM*);
|
|
|
|
|
|
|
|
jint
|
|
|
|
(JNICALL *AttachCurrentThread)
|
2007-09-10 23:33:58 +00:00
|
|
|
(JavaVM*, JNIEnv**, void*);
|
2007-06-25 01:34:07 +00:00
|
|
|
|
|
|
|
jint
|
|
|
|
(JNICALL *DetachCurrentThread)
|
|
|
|
(JavaVM*);
|
|
|
|
|
|
|
|
jint
|
|
|
|
(JNICALL *GetEnv)
|
2007-09-10 23:33:58 +00:00
|
|
|
(JavaVM*, JNIEnv**, jint);
|
2007-06-25 01:34:07 +00:00
|
|
|
|
|
|
|
jint
|
|
|
|
(JNICALL *AttachCurrentThreadAsDaemon)
|
2007-09-10 23:33:58 +00:00
|
|
|
(JavaVM*, JNIEnv**, void*);
|
2008-06-15 20:17:52 +00:00
|
|
|
|
|
|
|
#if TARGET_RT_MAC_CFM && defined(__ppc__)
|
|
|
|
void* real_functions[5];
|
|
|
|
#endif
|
2007-06-25 01:34:07 +00:00
|
|
|
};
|
|
|
|
|
2007-06-25 02:02:24 +00:00
|
|
|
struct JNIEnvVTable {
|
2007-06-25 01:34:07 +00:00
|
|
|
void* reserved0;
|
|
|
|
void* reserved1;
|
|
|
|
void* reserved2;
|
|
|
|
void* reserved3;
|
|
|
|
|
2008-06-15 18:51:04 +00:00
|
|
|
#if (! TARGET_RT_MAC_CFM) && defined(__ppc__)
|
|
|
|
void* cfm_vectors[225];
|
|
|
|
#endif
|
|
|
|
|
2007-06-25 01:34:07 +00:00
|
|
|
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);
|
2008-06-15 20:17:52 +00:00
|
|
|
|
|
|
|
#if TARGET_RT_MAC_CFM && defined(__ppc__)
|
|
|
|
void* real_functions[228];
|
|
|
|
#endif
|
2007-06-25 01:34:07 +00:00
|
|
|
};
|
|
|
|
|
2010-09-14 16:49:41 +00:00
|
|
|
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)
|
|
|
|
{ }
|
|
|
|
}
|
|
|
|
|
2007-07-06 15:24:06 +00:00
|
|
|
inline int
|
|
|
|
strcmp(const int8_t* a, const int8_t* b)
|
|
|
|
{
|
|
|
|
return ::strcmp(reinterpret_cast<const char*>(a),
|
|
|
|
reinterpret_cast<const char*>(b));
|
|
|
|
}
|
|
|
|
|
2007-07-10 01:43:43 +00:00
|
|
|
void
|
|
|
|
noop();
|
|
|
|
|
2007-09-07 23:20:21 +00:00
|
|
|
class Reference {
|
|
|
|
public:
|
2007-09-30 03:33:38 +00:00
|
|
|
Reference(object target, Reference** handle):
|
2007-09-07 23:20:21 +00:00
|
|
|
target(target),
|
2007-09-30 03:33:38 +00:00
|
|
|
next(*handle),
|
2009-12-17 02:16:51 +00:00
|
|
|
handle(handle),
|
|
|
|
count(0)
|
2007-09-30 03:33:38 +00:00
|
|
|
{
|
|
|
|
if (next) {
|
|
|
|
next->handle = &next;
|
|
|
|
}
|
|
|
|
*handle = this;
|
|
|
|
}
|
2007-09-07 23:20:21 +00:00
|
|
|
|
|
|
|
object target;
|
|
|
|
Reference* next;
|
2007-09-30 03:33:38 +00:00
|
|
|
Reference** handle;
|
2009-12-17 02:16:51 +00:00
|
|
|
unsigned count;
|
2007-09-07 23:20:21 +00:00
|
|
|
};
|
|
|
|
|
2010-09-10 21:05:29 +00:00
|
|
|
class Classpath;
|
|
|
|
|
2007-07-06 15:24:06 +00:00
|
|
|
class Machine {
|
|
|
|
public:
|
2007-09-13 03:15:16 +00:00
|
|
|
enum Type {
|
2007-07-06 15:24:06 +00:00
|
|
|
#include "type-enums.cpp"
|
2007-09-13 03:15:16 +00:00
|
|
|
};
|
2007-07-06 15:24:06 +00:00
|
|
|
|
2008-01-10 01:20:36 +00:00
|
|
|
enum AllocationType {
|
|
|
|
MovableAllocation,
|
|
|
|
FixedAllocation,
|
|
|
|
ImmortalAllocation
|
2007-10-14 01:18:25 +00:00
|
|
|
};
|
|
|
|
|
2010-09-14 16:49:41 +00:00
|
|
|
enum Root {
|
|
|
|
BootLoader,
|
|
|
|
AppLoader,
|
|
|
|
BootstrapClassMap,
|
2010-09-22 19:58:46 +00:00
|
|
|
FindLoadedClassMethod,
|
|
|
|
LoadClassMethod,
|
2010-09-14 16:49:41 +00:00
|
|
|
MonitorMap,
|
|
|
|
StringMap,
|
|
|
|
ByteArrayMap,
|
2010-11-26 19:41:31 +00:00
|
|
|
ClassRuntimeDataTable,
|
|
|
|
MethodRuntimeDataTable,
|
2010-09-14 16:49:41 +00:00
|
|
|
JNIMethodTable,
|
|
|
|
ShutdownHooks,
|
|
|
|
ObjectsToFinalize,
|
|
|
|
NullPointerException,
|
2010-11-05 19:18:28 +00:00
|
|
|
ArrayIndexOutOfBoundsException,
|
|
|
|
VirtualFileFinders,
|
|
|
|
VirtualFiles
|
2010-09-14 16:49:41 +00:00
|
|
|
};
|
|
|
|
|
2010-11-05 19:18:28 +00:00
|
|
|
static const unsigned RootCount = VirtualFiles + 1;
|
2010-09-14 16:49:41 +00:00
|
|
|
|
|
|
|
Machine(System* system, Heap* heap, Finder* bootFinder, Finder* appFinder,
|
|
|
|
Processor* processor, Classpath* classpath, const char** properties,
|
2010-09-10 21:05:29 +00:00
|
|
|
unsigned propertyCount);
|
2007-07-06 15:24:06 +00:00
|
|
|
|
|
|
|
~Machine() {
|
|
|
|
dispose();
|
|
|
|
}
|
|
|
|
|
|
|
|
void dispose();
|
|
|
|
|
2007-09-10 23:33:58 +00:00
|
|
|
JavaVMVTable* vtable;
|
2007-07-06 15:24:06 +00:00
|
|
|
System* system;
|
2008-01-13 22:05:08 +00:00
|
|
|
Heap::Client* heapClient;
|
2007-07-06 15:24:06 +00:00
|
|
|
Heap* heap;
|
2010-09-14 16:49:41 +00:00
|
|
|
Finder* bootFinder;
|
|
|
|
Finder* appFinder;
|
2007-09-24 01:39:03 +00:00
|
|
|
Processor* processor;
|
2010-09-10 21:05:29 +00:00
|
|
|
Classpath* classpath;
|
2007-07-06 15:24:06 +00:00
|
|
|
Thread* rootThread;
|
|
|
|
Thread* exclusive;
|
2009-08-24 23:51:31 +00:00
|
|
|
Thread* finalizeThread;
|
2007-09-07 23:20:21 +00:00
|
|
|
Reference* jniReferences;
|
2008-11-11 15:20:49 +00:00
|
|
|
const char** properties;
|
|
|
|
unsigned propertyCount;
|
2007-07-06 15:24:06 +00:00
|
|
|
unsigned activeCount;
|
|
|
|
unsigned liveCount;
|
2009-08-19 20:27:03 +00:00
|
|
|
unsigned daemonCount;
|
2007-10-28 19:14:53 +00:00
|
|
|
unsigned fixedFootprint;
|
2007-09-10 23:33:58 +00:00
|
|
|
System::Local* localThread;
|
2007-07-06 15:24:06 +00:00
|
|
|
System::Monitor* stateLock;
|
|
|
|
System::Monitor* heapLock;
|
|
|
|
System::Monitor* classLock;
|
2007-07-20 03:18:25 +00:00
|
|
|
System::Monitor* referenceLock;
|
2009-08-19 20:27:03 +00:00
|
|
|
System::Monitor* shutdownLock;
|
2008-01-29 15:19:15 +00:00
|
|
|
System::Library* libraries;
|
2007-07-06 15:24:06 +00:00
|
|
|
object types;
|
2010-09-14 16:49:41 +00:00
|
|
|
object roots;
|
2007-07-06 15:24:06 +00:00
|
|
|
object finalizers;
|
2007-07-10 01:43:43 +00:00
|
|
|
object tenuredFinalizers;
|
|
|
|
object finalizeQueue;
|
2007-07-06 15:24:06 +00:00
|
|
|
object weakReferences;
|
2007-07-10 01:43:43 +00:00
|
|
|
object tenuredWeakReferences;
|
2007-07-06 15:24:06 +00:00
|
|
|
bool unsafe;
|
2009-06-11 00:15:00 +00:00
|
|
|
bool triedBuiltinOnLoad;
|
2007-09-10 23:33:58 +00:00
|
|
|
JavaVMVTable javaVMVTable;
|
2007-07-06 15:24:06 +00:00
|
|
|
JNIEnvVTable jniEnvVTable;
|
2008-11-22 21:47:18 +00:00
|
|
|
uintptr_t* heapPool[ThreadHeapPoolSize];
|
2007-08-22 14:50:29 +00:00
|
|
|
unsigned heapPoolIndex;
|
2007-07-06 15:24:06 +00:00
|
|
|
};
|
|
|
|
|
2007-07-28 21:28:25 +00:00
|
|
|
void
|
|
|
|
printTrace(Thread* t, object exception);
|
|
|
|
|
|
|
|
uint8_t&
|
|
|
|
threadInterrupted(Thread* t, object thread);
|
|
|
|
|
2007-10-23 22:21:28 +00:00
|
|
|
void
|
|
|
|
enterActiveState(Thread* t);
|
|
|
|
|
2008-01-13 22:05:08 +00:00
|
|
|
#ifdef VM_STRESS
|
|
|
|
|
|
|
|
inline void stress(Thread* t);
|
|
|
|
|
|
|
|
#else // not VM_STRESS
|
|
|
|
|
|
|
|
#define stress(t)
|
|
|
|
|
|
|
|
#endif // not VM_STRESS
|
|
|
|
|
2008-06-25 16:28:11 +00:00
|
|
|
void
|
|
|
|
runJavaThread(Thread* t);
|
|
|
|
|
2009-08-24 23:51:31 +00:00
|
|
|
void
|
|
|
|
runFinalizeThread(Thread* t);
|
|
|
|
|
2007-07-06 15:24:06 +00:00
|
|
|
class Thread {
|
|
|
|
public:
|
|
|
|
enum State {
|
|
|
|
NoState,
|
|
|
|
ActiveState,
|
|
|
|
IdleState,
|
|
|
|
ZombieState,
|
2007-07-18 01:33:00 +00:00
|
|
|
JoinedState,
|
2007-07-06 15:24:06 +00:00
|
|
|
ExclusiveState,
|
|
|
|
ExitState
|
|
|
|
};
|
|
|
|
|
2010-09-14 16:49:41 +00:00
|
|
|
static const unsigned UseBackupHeapFlag = 1 << 0;
|
|
|
|
static const unsigned WaitingFlag = 1 << 1;
|
|
|
|
static const unsigned TracingFlag = 1 << 2;
|
|
|
|
static const unsigned DaemonFlag = 1 << 3;
|
|
|
|
static const unsigned StressFlag = 1 << 4;
|
|
|
|
static const unsigned ActiveFlag = 1 << 5;
|
2010-11-16 17:50:19 +00:00
|
|
|
static const unsigned SystemFlag = 1 << 6;
|
2010-09-14 16:49:41 +00:00
|
|
|
|
2007-07-06 15:24:06 +00:00
|
|
|
class Protector {
|
|
|
|
public:
|
2007-10-12 17:56:43 +00:00
|
|
|
Protector(Thread* t): t(t), next(t->protector) {
|
2007-07-06 15:24:06 +00:00
|
|
|
t->protector = this;
|
|
|
|
}
|
|
|
|
|
2008-11-11 15:20:49 +00:00
|
|
|
~Protector() {
|
2007-07-06 15:24:06 +00:00
|
|
|
t->protector = next;
|
|
|
|
}
|
|
|
|
|
2007-10-12 17:56:43 +00:00
|
|
|
virtual void visit(Heap::Visitor* v) = 0;
|
|
|
|
|
2007-07-06 15:24:06 +00:00
|
|
|
Thread* t;
|
|
|
|
Protector* next;
|
|
|
|
};
|
|
|
|
|
2007-10-12 17:56:43 +00:00
|
|
|
class SingleProtector: public Protector {
|
|
|
|
public:
|
|
|
|
SingleProtector(Thread* t, object* p): Protector(t), p(p) { }
|
|
|
|
|
|
|
|
virtual void visit(Heap::Visitor* v) {
|
2007-10-28 19:14:53 +00:00
|
|
|
v->visit(p);
|
2007-10-12 17:56:43 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
object* p;
|
|
|
|
};
|
|
|
|
|
2009-07-20 20:12:38 +00:00
|
|
|
class ClassInitStack {
|
|
|
|
public:
|
|
|
|
ClassInitStack(Thread* t, object class_):
|
|
|
|
next(t->classInitStack),
|
|
|
|
class_(class_),
|
|
|
|
protector(t, &(this->class_))
|
|
|
|
{
|
|
|
|
t->classInitStack = this;
|
|
|
|
}
|
|
|
|
|
|
|
|
~ClassInitStack() {
|
|
|
|
protector.t->classInitStack = next;
|
|
|
|
}
|
|
|
|
|
|
|
|
ClassInitStack* next;
|
|
|
|
object class_;
|
|
|
|
SingleProtector protector;
|
|
|
|
};
|
|
|
|
|
2007-07-28 21:28:25 +00:00
|
|
|
class Runnable: public System::Runnable {
|
|
|
|
public:
|
|
|
|
Runnable(Thread* t): t(t) { }
|
|
|
|
|
|
|
|
virtual void attach(System::Thread* st) {
|
|
|
|
t->systemThread = st;
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void run() {
|
2007-10-23 22:21:28 +00:00
|
|
|
enterActiveState(t);
|
|
|
|
|
2007-09-24 01:39:03 +00:00
|
|
|
t->m->localThread->set(t);
|
2007-09-10 23:33:58 +00:00
|
|
|
|
2009-08-24 23:51:31 +00:00
|
|
|
if (t == t->m->finalizeThread) {
|
|
|
|
runFinalizeThread(t);
|
|
|
|
} else if (t->javaThread) {
|
|
|
|
runJavaThread(t);
|
2007-07-28 21:28:25 +00:00
|
|
|
|
2009-08-24 23:51:31 +00:00
|
|
|
if (t->exception) {
|
|
|
|
printTrace(t, t->exception);
|
|
|
|
}
|
2007-07-28 21:28:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
t->exit();
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual bool interrupted() {
|
|
|
|
return threadInterrupted(t, t->javaThread);
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void setInterrupted(bool v) {
|
|
|
|
threadInterrupted(t, t->javaThread) = v;
|
|
|
|
}
|
|
|
|
|
|
|
|
Thread* t;
|
|
|
|
};
|
|
|
|
|
2007-07-24 01:44:20 +00:00
|
|
|
Thread(Machine* m, object javaThread, Thread* parent);
|
2007-07-06 15:24:06 +00:00
|
|
|
|
2007-10-25 22:06:05 +00:00
|
|
|
void init();
|
2007-07-07 18:09:16 +00:00
|
|
|
void exit();
|
2007-07-06 15:24:06 +00:00
|
|
|
void dispose();
|
|
|
|
|
|
|
|
JNIEnvVTable* vtable;
|
2007-09-24 01:39:03 +00:00
|
|
|
Machine* m;
|
2007-07-07 18:09:16 +00:00
|
|
|
Thread* parent;
|
|
|
|
Thread* peer;
|
2007-07-06 15:24:06 +00:00
|
|
|
Thread* child;
|
2010-02-05 00:56:21 +00:00
|
|
|
Thread* waitNext;
|
2007-07-06 15:24:06 +00:00
|
|
|
State state;
|
2007-09-07 23:20:21 +00:00
|
|
|
unsigned criticalLevel;
|
2007-07-07 18:09:16 +00:00
|
|
|
System::Thread* systemThread;
|
2010-02-05 00:56:21 +00:00
|
|
|
System::Monitor* lock;
|
2007-07-07 18:09:16 +00:00
|
|
|
object javaThread;
|
2007-07-06 15:24:06 +00:00
|
|
|
object exception;
|
|
|
|
unsigned heapIndex;
|
2007-08-23 02:24:25 +00:00
|
|
|
unsigned heapOffset;
|
2007-07-06 15:24:06 +00:00
|
|
|
Protector* protector;
|
2009-07-20 20:12:38 +00:00
|
|
|
ClassInitStack* classInitStack;
|
2007-07-28 21:28:25 +00:00
|
|
|
Runnable runnable;
|
2007-09-07 00:21:52 +00:00
|
|
|
uintptr_t* defaultHeap;
|
2007-09-13 03:15:16 +00:00
|
|
|
uintptr_t* heap;
|
2010-06-19 22:40:21 +00:00
|
|
|
uintptr_t backupHeap[ThreadBackupHeapSizeInWords];
|
2008-04-09 19:08:13 +00:00
|
|
|
unsigned backupHeapIndex;
|
2010-09-14 16:49:41 +00:00
|
|
|
unsigned flags;
|
2007-07-06 15:24:06 +00:00
|
|
|
};
|
|
|
|
|
2010-09-10 21:05:29 +00:00
|
|
|
class Classpath {
|
|
|
|
public:
|
|
|
|
virtual object
|
|
|
|
makeJclass(Thread* t, object class_) = 0;
|
|
|
|
|
|
|
|
virtual object
|
|
|
|
makeString(Thread* t, object array, int32_t offset, int32_t length) = 0;
|
|
|
|
|
|
|
|
virtual object
|
|
|
|
makeThread(Thread* t, Thread* parent) = 0;
|
|
|
|
|
|
|
|
virtual void
|
|
|
|
runThread(Thread* t) = 0;
|
|
|
|
|
|
|
|
virtual object
|
|
|
|
makeThrowable
|
|
|
|
(Thread* t, Machine::Type type, object message = 0, object trace = 0,
|
|
|
|
object cause = 0) = 0;
|
|
|
|
|
|
|
|
virtual void
|
|
|
|
boot(Thread* t) = 0;
|
|
|
|
|
2010-09-20 23:31:23 +00:00
|
|
|
virtual const char*
|
|
|
|
bootClasspath() = 0;
|
|
|
|
|
2010-09-10 21:05:29 +00:00
|
|
|
virtual void
|
|
|
|
dispose() = 0;
|
|
|
|
};
|
|
|
|
|
|
|
|
inline void
|
|
|
|
runJavaThread(Thread* t)
|
|
|
|
{
|
|
|
|
t->m->classpath->runThread(t);
|
|
|
|
}
|
|
|
|
|
|
|
|
Classpath*
|
2010-11-05 19:18:28 +00:00
|
|
|
makeClasspath(System* system, Allocator* allocator, const char* javaHome,
|
|
|
|
const char* embedPrefix);
|
2010-09-10 21:05:29 +00:00
|
|
|
|
2009-06-01 18:12:29 +00:00
|
|
|
typedef uint64_t (JNICALL *FastNativeFunction)(Thread*, object, uintptr_t*);
|
2009-05-03 20:57:11 +00:00
|
|
|
|
2007-07-06 15:24:06 +00:00
|
|
|
inline object
|
|
|
|
objectClass(Thread*, object o)
|
|
|
|
{
|
|
|
|
return mask(cast<object>(o, 0));
|
|
|
|
}
|
|
|
|
|
2007-07-16 01:03:02 +00:00
|
|
|
void
|
|
|
|
enter(Thread* t, Thread::State state);
|
2007-07-06 15:24:06 +00:00
|
|
|
|
2007-10-23 22:21:28 +00:00
|
|
|
inline void
|
|
|
|
enterActiveState(Thread* t)
|
|
|
|
{
|
|
|
|
enter(t, Thread::ActiveState);
|
|
|
|
}
|
|
|
|
|
2007-07-06 15:24:06 +00:00
|
|
|
class StateResource {
|
|
|
|
public:
|
|
|
|
StateResource(Thread* t, Thread::State state): t(t), oldState(t->state) {
|
|
|
|
enter(t, state);
|
|
|
|
}
|
|
|
|
|
|
|
|
~StateResource() { enter(t, oldState); }
|
|
|
|
|
|
|
|
private:
|
|
|
|
Thread* t;
|
|
|
|
Thread::State oldState;
|
|
|
|
};
|
|
|
|
|
2007-09-30 03:33:38 +00:00
|
|
|
inline void
|
|
|
|
dispose(Thread* t, Reference* r)
|
|
|
|
{
|
|
|
|
*(r->handle) = r->next;
|
2007-10-08 21:41:41 +00:00
|
|
|
if (r->next) {
|
|
|
|
r->next->handle = r->handle;
|
|
|
|
}
|
2008-04-13 18:15:04 +00:00
|
|
|
t->m->heap->free(r, sizeof(*r));
|
2007-09-30 03:33:38 +00:00
|
|
|
}
|
|
|
|
|
2009-12-17 02:16:51 +00:00
|
|
|
inline void
|
|
|
|
acquire(Thread*, Reference* r)
|
|
|
|
{
|
|
|
|
++ r->count;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void
|
|
|
|
release(Thread* t, Reference* r)
|
|
|
|
{
|
|
|
|
if ((-- r->count) == 0) {
|
|
|
|
dispose(t, r);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-07-16 01:03:02 +00:00
|
|
|
void
|
|
|
|
collect(Thread* t, Heap::CollectionType type);
|
|
|
|
|
2009-08-19 20:27:03 +00:00
|
|
|
void
|
|
|
|
shutDown(Thread* t);
|
|
|
|
|
2007-07-16 01:03:02 +00:00
|
|
|
#ifdef VM_STRESS
|
|
|
|
|
|
|
|
inline void
|
|
|
|
stress(Thread* t)
|
|
|
|
{
|
2010-09-17 01:43:27 +00:00
|
|
|
if ((t->flags & (Thread::StressFlag | Thread::TracingFlag)) == 0
|
2007-07-17 00:23:23 +00:00
|
|
|
and t->state != Thread::NoState
|
|
|
|
and t->state != Thread::IdleState)
|
|
|
|
{
|
2010-09-17 01:43:27 +00:00
|
|
|
atomicOr(&(t->flags), Thread::StressFlag);
|
2007-07-16 01:03:02 +00:00
|
|
|
|
|
|
|
# ifdef VM_STRESS_MAJOR
|
2010-09-17 01:43:27 +00:00
|
|
|
collect(t, Heap::MajorCollection);
|
2007-07-16 01:03:02 +00:00
|
|
|
# else // not VM_STRESS_MAJOR
|
2010-09-17 01:43:27 +00:00
|
|
|
collect(t, Heap::MinorCollection);
|
2007-07-16 01:03:02 +00:00
|
|
|
# endif // not VM_STRESS_MAJOR
|
|
|
|
|
2010-09-17 01:43:27 +00:00
|
|
|
atomicAnd(&(t->flags), ~Thread::StressFlag);
|
2007-07-16 01:03:02 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif // not VM_STRESS
|
|
|
|
|
2007-08-01 23:48:36 +00:00
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
2007-07-06 15:24:06 +00:00
|
|
|
class MonitorResource {
|
|
|
|
public:
|
|
|
|
MonitorResource(Thread* t, System::Monitor* m): t(t), m(m) {
|
2007-08-01 23:48:36 +00:00
|
|
|
acquire(t, m);
|
2007-07-06 15:24:06 +00:00
|
|
|
}
|
|
|
|
|
2007-08-01 23:48:36 +00:00
|
|
|
~MonitorResource() {
|
|
|
|
release(t, m);
|
|
|
|
}
|
2007-07-06 15:24:06 +00:00
|
|
|
|
|
|
|
private:
|
|
|
|
Thread* t;
|
|
|
|
System::Monitor* m;
|
|
|
|
};
|
|
|
|
|
|
|
|
class RawMonitorResource {
|
|
|
|
public:
|
|
|
|
RawMonitorResource(Thread* t, System::Monitor* m): t(t), m(m) {
|
2007-08-14 00:37:00 +00:00
|
|
|
m->acquire(t->systemThread);
|
2007-07-06 15:24:06 +00:00
|
|
|
}
|
|
|
|
|
2007-08-01 23:48:36 +00:00
|
|
|
~RawMonitorResource() {
|
|
|
|
release(t, m);
|
|
|
|
}
|
2007-07-06 15:24:06 +00:00
|
|
|
|
|
|
|
private:
|
|
|
|
Thread* t;
|
|
|
|
System::Monitor* m;
|
|
|
|
};
|
|
|
|
|
|
|
|
inline void NO_RETURN
|
|
|
|
abort(Thread* t)
|
|
|
|
{
|
2007-09-24 01:39:03 +00:00
|
|
|
abort(t->m->system);
|
2007-07-06 15:24:06 +00:00
|
|
|
}
|
|
|
|
|
2007-08-19 19:45:51 +00:00
|
|
|
#ifndef NDEBUG
|
2007-07-06 15:24:06 +00:00
|
|
|
inline void
|
|
|
|
assert(Thread* t, bool v)
|
|
|
|
{
|
2007-09-24 01:39:03 +00:00
|
|
|
assert(t->m->system, v);
|
2007-07-06 15:24:06 +00:00
|
|
|
}
|
2007-08-19 19:45:51 +00:00
|
|
|
#endif // not NDEBUG
|
2007-07-06 15:24:06 +00:00
|
|
|
|
|
|
|
inline void
|
|
|
|
expect(Thread* t, bool v)
|
|
|
|
{
|
2007-09-24 01:39:03 +00:00
|
|
|
expect(t->m->system, v);
|
2007-07-06 15:24:06 +00:00
|
|
|
}
|
|
|
|
|
2008-12-02 02:38:00 +00:00
|
|
|
class FixedAllocator: public Allocator {
|
|
|
|
public:
|
2009-04-05 21:42:10 +00:00
|
|
|
FixedAllocator(System* s, uint8_t* base, unsigned capacity):
|
|
|
|
s(s), base(base), offset(0), capacity(capacity)
|
2008-12-02 02:38:00 +00:00
|
|
|
{ }
|
|
|
|
|
|
|
|
virtual void* tryAllocate(unsigned) {
|
2009-04-05 21:42:10 +00:00
|
|
|
abort(s);
|
2008-12-02 02:38:00 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
virtual void* allocate(unsigned size) {
|
|
|
|
unsigned paddedSize = pad(size);
|
2009-04-05 21:42:10 +00:00
|
|
|
expect(s, offset + paddedSize < capacity);
|
2008-12-02 02:38:00 +00:00
|
|
|
|
|
|
|
void* p = base + offset;
|
|
|
|
offset += paddedSize;
|
|
|
|
return p;
|
|
|
|
}
|
|
|
|
|
|
|
|
virtual void free(const void*, unsigned) {
|
2009-04-05 21:42:10 +00:00
|
|
|
abort(s);
|
2008-12-02 02:38:00 +00:00
|
|
|
}
|
|
|
|
|
2009-04-05 21:42:10 +00:00
|
|
|
System* s;
|
2008-12-02 02:38:00 +00:00
|
|
|
uint8_t* base;
|
|
|
|
unsigned offset;
|
|
|
|
unsigned capacity;
|
|
|
|
};
|
|
|
|
|
2010-06-19 22:40:21 +00:00
|
|
|
inline bool
|
2008-04-09 19:08:13 +00:00
|
|
|
ensure(Thread* t, unsigned sizeInBytes)
|
|
|
|
{
|
|
|
|
if (t->heapIndex + ceiling(sizeInBytes, BytesPerWord)
|
2008-11-22 21:47:18 +00:00
|
|
|
> ThreadHeapSizeInWords)
|
2008-04-09 19:08:13 +00:00
|
|
|
{
|
2010-06-19 22:40:21 +00:00
|
|
|
if (sizeInBytes <= ThreadBackupHeapSizeInBytes) {
|
2010-09-14 16:49:41 +00:00
|
|
|
expect(t, (t->flags & Thread::UseBackupHeapFlag) == 0);
|
2009-03-04 03:05:48 +00:00
|
|
|
|
2010-09-14 16:49:41 +00:00
|
|
|
atomicOr(&(t->flags), Thread::UseBackupHeapFlag);
|
2009-03-04 03:05:48 +00:00
|
|
|
|
2010-06-19 22:40:21 +00:00
|
|
|
return true;
|
|
|
|
} else {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
return true;
|
2008-04-09 19:08:13 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-10-28 01:54:30 +00:00
|
|
|
object
|
2008-01-10 01:20:36 +00:00
|
|
|
allocate2(Thread* t, unsigned sizeInBytes, bool objectMask);
|
|
|
|
|
|
|
|
object
|
|
|
|
allocate3(Thread* t, Allocator* allocator, Machine::AllocationType type,
|
2008-04-13 18:15:04 +00:00
|
|
|
unsigned sizeInBytes, bool objectMask);
|
2007-07-06 15:24:06 +00:00
|
|
|
|
|
|
|
inline object
|
|
|
|
allocateSmall(Thread* t, unsigned sizeInBytes)
|
|
|
|
{
|
2009-07-18 01:37:46 +00:00
|
|
|
assert(t, t->heapIndex + ceiling(sizeInBytes, BytesPerWord)
|
|
|
|
<= ThreadHeapSizeInWords);
|
|
|
|
|
2007-09-07 00:21:52 +00:00
|
|
|
object o = reinterpret_cast<object>(t->heap + t->heapIndex);
|
2007-07-26 00:48:28 +00:00
|
|
|
t->heapIndex += ceiling(sizeInBytes, BytesPerWord);
|
2007-10-28 01:54:30 +00:00
|
|
|
cast<object>(o, 0) = 0;
|
2007-07-06 15:24:06 +00:00
|
|
|
return o;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline object
|
2007-10-28 01:54:30 +00:00
|
|
|
allocate(Thread* t, unsigned sizeInBytes, bool objectMask)
|
2007-07-06 15:24:06 +00:00
|
|
|
{
|
2007-07-16 01:03:02 +00:00
|
|
|
stress(t);
|
|
|
|
|
2007-07-26 00:48:28 +00:00
|
|
|
if (UNLIKELY(t->heapIndex + ceiling(sizeInBytes, BytesPerWord)
|
2008-11-22 21:47:18 +00:00
|
|
|
> ThreadHeapSizeInWords
|
2007-09-24 01:39:03 +00:00
|
|
|
or t->m->exclusive))
|
2007-07-06 15:24:06 +00:00
|
|
|
{
|
2008-01-10 01:20:36 +00:00
|
|
|
return allocate2(t, sizeInBytes, objectMask);
|
2007-07-06 15:24:06 +00:00
|
|
|
} else {
|
|
|
|
return allocateSmall(t, sizeInBytes);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-10-28 19:14:53 +00:00
|
|
|
inline void
|
|
|
|
mark(Thread* t, object o, unsigned offset, unsigned count)
|
|
|
|
{
|
2009-11-20 01:13:00 +00:00
|
|
|
t->m->heap->mark(o, offset / BytesPerWord, count);
|
2007-10-28 19:14:53 +00:00
|
|
|
}
|
2007-07-06 15:24:06 +00:00
|
|
|
|
2007-10-28 19:14:53 +00:00
|
|
|
inline void
|
|
|
|
mark(Thread* t, object o, unsigned offset)
|
|
|
|
{
|
2009-11-20 01:13:00 +00:00
|
|
|
t->m->heap->mark(o, offset / BytesPerWord, 1);
|
2007-10-28 19:14:53 +00:00
|
|
|
}
|
2007-10-28 01:54:30 +00:00
|
|
|
|
2008-07-05 20:21:13 +00:00
|
|
|
inline void
|
2007-10-22 17:22:30 +00:00
|
|
|
set(Thread* t, object target, unsigned offset, object value)
|
2007-07-16 23:58:37 +00:00
|
|
|
{
|
2007-10-22 17:22:30 +00:00
|
|
|
cast<object>(target, offset) = value;
|
|
|
|
mark(t, target, offset);
|
2007-07-16 23:58:37 +00:00
|
|
|
}
|
|
|
|
|
2007-07-14 17:31:01 +00:00
|
|
|
inline void
|
2008-01-10 01:20:36 +00:00
|
|
|
setObjectClass(Thread*, object o, object value)
|
2007-07-14 17:31:01 +00:00
|
|
|
{
|
2008-01-10 01:20:36 +00:00
|
|
|
cast<object>(o, 0)
|
|
|
|
= reinterpret_cast<object>
|
|
|
|
(reinterpret_cast<uintptr_t>(value)
|
2008-03-10 19:49:10 +00:00
|
|
|
| (reinterpret_cast<uintptr_t>(cast<object>(o, 0)) & (~PointerMask)));
|
2007-07-14 17:31:01 +00:00
|
|
|
}
|
|
|
|
|
2009-08-19 20:27:03 +00:00
|
|
|
inline Thread*
|
|
|
|
startThread(Thread* t, object javaThread)
|
|
|
|
{
|
|
|
|
Thread* p = t->m->processor->makeThread(t->m, javaThread, t);
|
|
|
|
|
|
|
|
if (t->m->system->success(t->m->system->start(&(p->runnable)))) {
|
|
|
|
return p;
|
|
|
|
} else {
|
|
|
|
p->exit();
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2008-11-11 15:20:49 +00:00
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
2007-07-06 15:24:06 +00:00
|
|
|
object&
|
|
|
|
arrayBodyUnsafe(Thread*, object, unsigned);
|
|
|
|
|
2007-11-04 21:15:28 +00:00
|
|
|
bool
|
|
|
|
instanceOf(Thread* t, object class_, object o);
|
|
|
|
|
2007-07-06 15:24:06 +00:00
|
|
|
#include "type-declarations.cpp"
|
|
|
|
|
2010-09-14 16:49:41 +00:00
|
|
|
inline object&
|
|
|
|
root(Thread* t, Machine::Root root)
|
|
|
|
{
|
|
|
|
return arrayBody(t, t->m->roots, root);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void
|
|
|
|
setRoot(Thread* t, Machine::Root root, object value)
|
|
|
|
{
|
|
|
|
set(t, t->m->roots, ArrayBody + (root * BytesPerWord), value);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline object
|
|
|
|
type(Thread* t, Machine::Type type)
|
|
|
|
{
|
|
|
|
return arrayBody(t, t->m->types, type);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void
|
|
|
|
setType(Thread* t, Machine::Type type, object value)
|
|
|
|
{
|
|
|
|
set(t, t->m->types, ArrayBody + (type * BytesPerWord), value);
|
|
|
|
}
|
|
|
|
|
2007-10-22 17:22:30 +00:00
|
|
|
inline bool
|
|
|
|
objectFixed(Thread*, object o)
|
|
|
|
{
|
|
|
|
return (cast<uintptr_t>(o, 0) & (~PointerMask)) == FixedMark;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline bool
|
|
|
|
objectExtended(Thread*, object o)
|
|
|
|
{
|
|
|
|
return (cast<uintptr_t>(o, 0) & (~PointerMask)) == ExtendedMark;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline bool
|
|
|
|
hashTaken(Thread*, object o)
|
|
|
|
{
|
|
|
|
return (cast<uintptr_t>(o, 0) & (~PointerMask)) == HashTakenMark;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline unsigned
|
|
|
|
baseSize(Thread* t, object o, object class_)
|
|
|
|
{
|
|
|
|
return ceiling(classFixedSize(t, class_), BytesPerWord)
|
|
|
|
+ ceiling(classArrayElementSize(t, class_)
|
|
|
|
* cast<uintptr_t>(o, classFixedSize(t, class_) - BytesPerWord),
|
|
|
|
BytesPerWord);
|
|
|
|
}
|
|
|
|
|
2007-07-06 15:24:06 +00:00
|
|
|
object
|
2007-11-25 23:00:55 +00:00
|
|
|
makeTrace(Thread* t, Processor::StackWalker* walker);
|
2007-07-06 15:24:06 +00:00
|
|
|
|
2007-11-25 23:00:55 +00:00
|
|
|
object
|
2008-04-09 19:08:13 +00:00
|
|
|
makeTrace(Thread* t, Thread* target);
|
|
|
|
|
|
|
|
inline object
|
|
|
|
makeTrace(Thread* t)
|
|
|
|
{
|
|
|
|
return makeTrace(t, t);
|
|
|
|
}
|
2007-07-06 15:24:06 +00:00
|
|
|
|
2008-07-05 20:21:13 +00:00
|
|
|
inline object
|
2007-09-27 22:20:54 +00:00
|
|
|
makeNew(Thread* t, object class_)
|
|
|
|
{
|
2010-09-10 21:05:29 +00:00
|
|
|
assert(t, t->state == Thread::NoState or t->state == Thread::ActiveState);
|
2008-01-18 01:27:44 +00:00
|
|
|
|
2007-09-27 22:20:54 +00:00
|
|
|
PROTECT(t, class_);
|
|
|
|
unsigned sizeInBytes = pad(classFixedSize(t, class_));
|
2009-08-18 20:26:28 +00:00
|
|
|
assert(t, sizeInBytes);
|
2007-10-28 01:54:30 +00:00
|
|
|
object instance = allocate(t, sizeInBytes, classObjectMask(t, class_));
|
|
|
|
setObjectClass(t, instance, class_);
|
2007-09-27 22:20:54 +00:00
|
|
|
|
|
|
|
return instance;
|
|
|
|
}
|
|
|
|
|
2009-07-22 00:57:55 +00:00
|
|
|
object
|
|
|
|
makeNewGeneral(Thread* t, object class_);
|
2007-09-27 22:20:54 +00:00
|
|
|
|
|
|
|
inline object
|
|
|
|
make(Thread* t, object class_)
|
|
|
|
{
|
2009-07-22 00:57:55 +00:00
|
|
|
if (UNLIKELY(classVmFlags(t, class_)
|
|
|
|
& (WeakReferenceFlag | HasFinalizerFlag)))
|
|
|
|
{
|
|
|
|
return makeNewGeneral(t, class_);
|
2007-09-27 22:20:54 +00:00
|
|
|
} else {
|
|
|
|
return makeNew(t, class_);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-07-06 15:24:06 +00:00
|
|
|
object
|
|
|
|
makeByteArray(Thread* t, const char* format, ...);
|
|
|
|
|
|
|
|
object
|
|
|
|
makeString(Thread* t, const char* format, ...);
|
|
|
|
|
2010-04-20 16:03:07 +00:00
|
|
|
int
|
2010-09-10 21:05:29 +00:00
|
|
|
stringUTFLength(Thread* t, object string, unsigned start, unsigned length);
|
|
|
|
|
|
|
|
inline int
|
|
|
|
stringUTFLength(Thread* t, object string)
|
|
|
|
{
|
|
|
|
return stringUTFLength(t, string, 0, stringLength(t, string));
|
|
|
|
}
|
2010-04-20 16:03:07 +00:00
|
|
|
|
2007-07-08 01:06:32 +00:00
|
|
|
void
|
2010-09-10 21:05:29 +00:00
|
|
|
stringChars(Thread* t, object string, unsigned start, unsigned length,
|
|
|
|
char* chars);
|
|
|
|
|
|
|
|
inline void
|
|
|
|
stringChars(Thread* t, object string, char* chars)
|
|
|
|
{
|
|
|
|
stringChars(t, string, 0, stringLength(t, string), chars);
|
|
|
|
}
|
2007-07-08 01:06:32 +00:00
|
|
|
|
2008-04-01 23:24:43 +00:00
|
|
|
void
|
2010-09-10 21:05:29 +00:00
|
|
|
stringChars(Thread* t, object string, unsigned start, unsigned length,
|
|
|
|
uint16_t* chars);
|
|
|
|
|
|
|
|
inline void
|
|
|
|
stringChars(Thread* t, object string, uint16_t* chars)
|
|
|
|
{
|
|
|
|
stringChars(t, string, 0, stringLength(t, string), chars);
|
|
|
|
}
|
2008-04-01 23:24:43 +00:00
|
|
|
|
2010-04-20 16:03:07 +00:00
|
|
|
void
|
2010-09-10 21:05:29 +00:00
|
|
|
stringUTFChars(Thread* t, object string, unsigned start, unsigned length,
|
|
|
|
char* chars, unsigned charsLength);
|
|
|
|
|
|
|
|
inline void
|
|
|
|
stringUTFChars(Thread* t, object string, char* chars, unsigned charsLength)
|
|
|
|
{
|
|
|
|
stringUTFChars(t, string, 0, stringLength(t, string), chars, charsLength);
|
|
|
|
}
|
2010-04-20 16:03:07 +00:00
|
|
|
|
2007-07-24 01:44:20 +00:00
|
|
|
bool
|
|
|
|
isAssignableFrom(Thread* t, object a, object b);
|
|
|
|
|
2007-07-28 16:10:13 +00:00
|
|
|
object
|
|
|
|
classInitializer(Thread* t, object class_);
|
|
|
|
|
2007-09-13 00:21:37 +00:00
|
|
|
object
|
|
|
|
frameMethod(Thread* t, int frame);
|
|
|
|
|
2007-07-06 15:24:06 +00:00
|
|
|
inline uintptr_t&
|
2007-08-19 19:45:51 +00:00
|
|
|
extendedWord(Thread* t UNUSED, object o, unsigned baseSize)
|
2007-07-06 15:24:06 +00:00
|
|
|
{
|
|
|
|
assert(t, objectExtended(t, o));
|
|
|
|
return cast<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)
|
|
|
|
{
|
|
|
|
assert(t, not objectExtended(t, o));
|
2007-10-22 17:22:30 +00:00
|
|
|
assert(t, not objectFixed(t, o));
|
2007-08-18 21:24:29 +00:00
|
|
|
|
2007-09-24 01:39:03 +00:00
|
|
|
ACQUIRE_RAW(t, t->m->heapLock);
|
2007-10-28 01:54:30 +00:00
|
|
|
|
|
|
|
cast<uintptr_t>(o, 0) |= HashTakenMark;
|
2008-01-14 16:39:57 +00:00
|
|
|
t->m->heap->pad(o);
|
2007-07-06 15:24:06 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
inline uint32_t
|
|
|
|
takeHash(Thread*, object o)
|
|
|
|
{
|
|
|
|
return reinterpret_cast<uintptr_t>(o) / BytesPerWord;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline uint32_t
|
|
|
|
objectHash(Thread* t, object o)
|
|
|
|
{
|
|
|
|
if (objectExtended(t, o)) {
|
|
|
|
return extendedWord(t, o, baseSize(t, o, objectClass(t, o)));
|
|
|
|
} else {
|
2007-10-22 17:22:30 +00:00
|
|
|
if (not objectFixed(t, o)) {
|
|
|
|
markHashTaken(t, o);
|
|
|
|
}
|
2007-07-06 15:24:06 +00:00
|
|
|
return takeHash(t, o);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
inline bool
|
|
|
|
objectEqual(Thread*, object a, object b)
|
|
|
|
{
|
|
|
|
return a == b;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline uint32_t
|
|
|
|
byteArrayHash(Thread* t, object array)
|
|
|
|
{
|
|
|
|
return hash(&byteArrayBody(t, array, 0), byteArrayLength(t, array));
|
|
|
|
}
|
|
|
|
|
2007-07-29 00:02:32 +00:00
|
|
|
inline uint32_t
|
|
|
|
charArrayHash(Thread* t, object array)
|
|
|
|
{
|
|
|
|
return hash(&charArrayBody(t, array, 0), charArrayLength(t, array));
|
|
|
|
}
|
|
|
|
|
2007-07-06 15:24:06 +00:00
|
|
|
inline bool
|
|
|
|
byteArrayEqual(Thread* t, object a, object b)
|
|
|
|
{
|
|
|
|
return a == b or
|
|
|
|
((byteArrayLength(t, a) == byteArrayLength(t, b)) and
|
|
|
|
memcmp(&byteArrayBody(t, a, 0), &byteArrayBody(t, b, 0),
|
|
|
|
byteArrayLength(t, a)) == 0);
|
|
|
|
}
|
|
|
|
|
2007-07-29 00:02:32 +00:00
|
|
|
inline uint32_t
|
|
|
|
stringHash(Thread* t, object s)
|
|
|
|
{
|
2007-08-19 20:24:26 +00:00
|
|
|
if (stringHashCode(t, s) == 0 and stringLength(t, s)) {
|
2007-07-29 00:02:32 +00:00
|
|
|
object data = stringData(t, s);
|
2010-09-14 16:49:41 +00:00
|
|
|
if (objectClass(t, data) == type(t, Machine::ByteArrayType)) {
|
2007-08-15 01:14:55 +00:00
|
|
|
stringHashCode(t, s) = hash
|
|
|
|
(&byteArrayBody(t, data, stringOffset(t, s)), stringLength(t, s));
|
2007-07-29 00:02:32 +00:00
|
|
|
} else {
|
2007-08-15 01:14:55 +00:00
|
|
|
stringHashCode(t, s) = hash
|
|
|
|
(&charArrayBody(t, data, stringOffset(t, s)), stringLength(t, s));
|
2007-07-29 00:02:32 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return stringHashCode(t, s);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline uint16_t
|
|
|
|
stringCharAt(Thread* t, object s, int i)
|
|
|
|
{
|
|
|
|
object data = stringData(t, s);
|
2010-09-14 16:49:41 +00:00
|
|
|
if (objectClass(t, data) == type(t, Machine::ByteArrayType)) {
|
2009-06-06 22:36:07 +00:00
|
|
|
return byteArrayBody(t, data, stringOffset(t, s) + i);
|
2007-07-29 00:02:32 +00:00
|
|
|
} else {
|
2009-06-06 22:36:07 +00:00
|
|
|
return charArrayBody(t, data, stringOffset(t, s) + i);
|
2007-07-29 00:02:32 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
inline bool
|
|
|
|
stringEqual(Thread* t, object a, object b)
|
|
|
|
{
|
|
|
|
if (a == b) {
|
|
|
|
return true;
|
|
|
|
} else if (stringLength(t, a) == stringLength(t, b)) {
|
2007-11-04 21:15:28 +00:00
|
|
|
for (unsigned i = 0; i < stringLength(t, a); ++i) {
|
2007-07-29 00:02:32 +00:00
|
|
|
if (stringCharAt(t, a, i) != stringCharAt(t, b, i)) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return true;
|
|
|
|
} else {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-07-06 15:24:06 +00:00
|
|
|
inline uint32_t
|
|
|
|
methodHash(Thread* t, object method)
|
|
|
|
{
|
|
|
|
return byteArrayHash(t, methodName(t, method))
|
|
|
|
^ byteArrayHash(t, methodSpec(t, method));
|
|
|
|
}
|
|
|
|
|
|
|
|
inline bool
|
|
|
|
methodEqual(Thread* t, object a, object b)
|
|
|
|
{
|
|
|
|
return a == b or
|
|
|
|
(byteArrayEqual(t, methodName(t, a), methodName(t, b)) and
|
|
|
|
byteArrayEqual(t, methodSpec(t, a), methodSpec(t, b)));
|
|
|
|
}
|
|
|
|
|
2007-09-28 23:41:03 +00:00
|
|
|
class MethodSpecIterator {
|
|
|
|
public:
|
|
|
|
MethodSpecIterator(Thread* t, const char* s):
|
|
|
|
t(t), s(s + 1)
|
|
|
|
{ }
|
|
|
|
|
|
|
|
const char* next() {
|
|
|
|
assert(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() {
|
|
|
|
assert(t, *s == ')');
|
|
|
|
return s + 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
Thread* t;
|
|
|
|
const char* s;
|
|
|
|
};
|
|
|
|
|
2007-07-14 17:31:01 +00:00
|
|
|
unsigned
|
|
|
|
fieldCode(Thread* t, unsigned javaCode);
|
|
|
|
|
|
|
|
unsigned
|
|
|
|
fieldType(Thread* t, unsigned code);
|
|
|
|
|
|
|
|
unsigned
|
|
|
|
primitiveSize(Thread* t, unsigned code);
|
|
|
|
|
|
|
|
inline unsigned
|
2007-11-02 21:08:14 +00:00
|
|
|
fieldSize(Thread* t, unsigned code)
|
2007-07-14 17:31:01 +00:00
|
|
|
{
|
|
|
|
if (code == ObjectField) {
|
|
|
|
return BytesPerWord;
|
|
|
|
} else {
|
|
|
|
return primitiveSize(t, code);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-11-02 21:08:14 +00:00
|
|
|
inline unsigned
|
|
|
|
fieldSize(Thread* t, object field)
|
|
|
|
{
|
|
|
|
return fieldSize(t, fieldCode(t, field));
|
|
|
|
}
|
|
|
|
|
2007-07-30 23:19:05 +00:00
|
|
|
object
|
2010-09-14 16:49:41 +00:00
|
|
|
findLoadedClass(Thread* t, object loader, object spec);
|
2007-07-30 23:19:05 +00:00
|
|
|
|
2009-07-22 00:57:55 +00:00
|
|
|
inline bool
|
|
|
|
emptyMethod(Thread* t, object method)
|
|
|
|
{
|
|
|
|
return ((methodFlags(t, method) & ACC_NATIVE) == 0)
|
|
|
|
and (codeLength(t, methodCode(t, method)) == 1)
|
|
|
|
and (codeBody(t, methodCode(t, method), 0) == return_);
|
|
|
|
}
|
|
|
|
|
2010-09-10 21:05:29 +00:00
|
|
|
object
|
|
|
|
parseUtf8(Thread* t, const char* data, unsigned length);
|
|
|
|
|
2007-07-30 23:19:05 +00:00
|
|
|
object
|
2009-08-10 13:56:16 +00:00
|
|
|
parseClass(Thread* t, object loader, const uint8_t* data, unsigned length);
|
2007-07-30 23:19:05 +00:00
|
|
|
|
2007-07-14 17:31:01 +00:00
|
|
|
object
|
2010-09-14 16:49:41 +00:00
|
|
|
resolveClass(Thread* t, object loader, object name, bool throw_ = true);
|
2009-06-11 00:15:00 +00:00
|
|
|
|
|
|
|
inline object
|
2010-11-04 17:02:09 +00:00
|
|
|
resolveClass(Thread* t, object loader, const char* name, bool throw_ = true)
|
2009-06-11 00:15:00 +00:00
|
|
|
{
|
2009-08-10 13:56:16 +00:00
|
|
|
PROTECT(t, loader);
|
|
|
|
object n = makeByteArray(t, "%s", name);
|
2010-11-04 17:02:09 +00:00
|
|
|
return resolveClass(t, loader, n, throw_);
|
2009-08-10 13:56:16 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
object
|
2010-09-14 16:49:41 +00:00
|
|
|
resolveSystemClass(Thread* t, object loader, object name, bool throw_ = true);
|
2009-08-10 13:56:16 +00:00
|
|
|
|
|
|
|
inline object
|
2010-09-14 16:49:41 +00:00
|
|
|
resolveSystemClass(Thread* t, object loader, const char* name)
|
2009-08-10 13:56:16 +00:00
|
|
|
{
|
2010-09-14 16:49:41 +00:00
|
|
|
return resolveSystemClass(t, loader, makeByteArray(t, "%s", name));
|
2009-06-11 00:15:00 +00:00
|
|
|
}
|
2007-07-14 17:31:01 +00:00
|
|
|
|
2009-08-18 20:26:28 +00:00
|
|
|
void
|
|
|
|
linkClass(Thread* t, object loader, object class_);
|
|
|
|
|
2007-09-24 01:39:03 +00:00
|
|
|
object
|
2009-06-11 00:15:00 +00:00
|
|
|
resolveMethod(Thread* t, object class_, const char* methodName,
|
2007-09-24 01:39:03 +00:00
|
|
|
const char* methodSpec);
|
|
|
|
|
2009-06-11 00:15:00 +00:00
|
|
|
inline object
|
2009-08-10 13:56:16 +00:00
|
|
|
resolveMethod(Thread* t, object loader, const char* className,
|
|
|
|
const char* methodName, const char* methodSpec)
|
2009-06-11 00:15:00 +00:00
|
|
|
{
|
2009-08-10 13:56:16 +00:00
|
|
|
object class_ = resolveClass(t, loader, className);
|
2009-06-11 00:15:00 +00:00
|
|
|
if (LIKELY(t->exception == 0)) {
|
|
|
|
return resolveMethod(t, class_, methodName, methodSpec);
|
|
|
|
} else {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
object
|
|
|
|
resolveField(Thread* t, object class_, const char* fieldName,
|
|
|
|
const char* fieldSpec);
|
|
|
|
|
|
|
|
inline object
|
2009-08-10 13:56:16 +00:00
|
|
|
resolveField(Thread* t, object loader, const char* className,
|
|
|
|
const char* fieldName, const char* fieldSpec)
|
2009-06-11 00:15:00 +00:00
|
|
|
{
|
2009-08-10 13:56:16 +00:00
|
|
|
object class_ = resolveClass(t, loader, className);
|
2009-06-11 00:15:00 +00:00
|
|
|
if (LIKELY(t->exception == 0)) {
|
|
|
|
return resolveField(t, class_, fieldName, fieldSpec);
|
|
|
|
} else {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2009-07-20 20:12:38 +00:00
|
|
|
bool
|
|
|
|
classNeedsInit(Thread* t, object c);
|
2008-11-30 04:58:09 +00:00
|
|
|
|
2009-07-20 20:12:38 +00:00
|
|
|
bool
|
|
|
|
preInitClass(Thread* t, object c);
|
|
|
|
|
|
|
|
void
|
|
|
|
postInitClass(Thread* t, object c);
|
|
|
|
|
|
|
|
void
|
|
|
|
initClass(Thread* t, object c);
|
2007-08-23 03:30:37 +00:00
|
|
|
|
2007-07-14 17:31:01 +00:00
|
|
|
object
|
2010-09-14 16:49:41 +00:00
|
|
|
makeObjectArray(Thread* t, object elementClass, unsigned count);
|
2007-07-14 17:31:01 +00:00
|
|
|
|
2009-09-19 00:01:54 +00:00
|
|
|
inline object
|
|
|
|
makeObjectArray(Thread* t, unsigned count)
|
|
|
|
{
|
2010-09-14 16:49:41 +00:00
|
|
|
return makeObjectArray(t, type(t, Machine::JobjectType), count);
|
2009-09-19 00:01:54 +00:00
|
|
|
}
|
|
|
|
|
2007-09-07 00:21:52 +00:00
|
|
|
object
|
|
|
|
findInTable(Thread* t, object table, object name, object spec,
|
|
|
|
object& (*getName)(Thread*, object),
|
|
|
|
object& (*getSpec)(Thread*, object));
|
|
|
|
|
|
|
|
inline object
|
|
|
|
findFieldInClass(Thread* t, object class_, object name, object spec)
|
|
|
|
{
|
|
|
|
return findInTable
|
|
|
|
(t, classFieldTable(t, class_), name, spec, fieldName, fieldSpec);
|
|
|
|
}
|
|
|
|
|
2010-11-04 17:02:09 +00:00
|
|
|
inline object
|
|
|
|
findFieldInClass2(Thread* t, object class_, const char* name, const char* spec)
|
|
|
|
{
|
|
|
|
PROTECT(t, class_);
|
|
|
|
object n = makeByteArray(t, "%s", name);
|
|
|
|
PROTECT(t, n);
|
|
|
|
object s = makeByteArray(t, "%s", spec);
|
|
|
|
return findFieldInClass(t, class_, n, s);
|
|
|
|
}
|
|
|
|
|
2007-09-07 00:21:52 +00:00
|
|
|
inline object
|
|
|
|
findMethodInClass(Thread* t, object class_, object name, object spec)
|
|
|
|
{
|
|
|
|
return findInTable
|
|
|
|
(t, classMethodTable(t, class_), name, spec, methodName, methodSpec);
|
|
|
|
}
|
|
|
|
|
|
|
|
object
|
2010-11-04 17:02:09 +00:00
|
|
|
findInHierarchyOrNull(Thread* t, object class_, object name, object spec,
|
|
|
|
object (*find)(Thread*, object, object, object));
|
|
|
|
|
|
|
|
inline object
|
2007-09-07 00:21:52 +00:00
|
|
|
findInHierarchy(Thread* t, object class_, object name, object spec,
|
|
|
|
object (*find)(Thread*, object, object, object),
|
2010-11-04 17:02:09 +00:00
|
|
|
Machine::Type errorType)
|
|
|
|
{
|
|
|
|
object o = findInHierarchyOrNull(t, class_, name, spec, find);
|
|
|
|
|
|
|
|
if (o == 0) {
|
|
|
|
object message = makeString
|
|
|
|
(t, "%s %s not found in %s",
|
|
|
|
&byteArrayBody(t, name, 0),
|
|
|
|
&byteArrayBody(t, spec, 0),
|
|
|
|
&byteArrayBody(t, className(t, class_), 0));
|
|
|
|
t->exception = t->m->classpath->makeThrowable(t, errorType, message);
|
|
|
|
}
|
|
|
|
|
|
|
|
return o;
|
|
|
|
}
|
2007-09-07 00:21:52 +00:00
|
|
|
|
|
|
|
inline object
|
|
|
|
findMethod(Thread* t, object class_, object name, object spec)
|
|
|
|
{
|
|
|
|
return findInHierarchy
|
2010-09-10 21:05:29 +00:00
|
|
|
(t, class_, name, spec, findMethodInClass, Machine::NoSuchMethodErrorType);
|
2007-09-07 00:21:52 +00:00
|
|
|
}
|
|
|
|
|
2010-11-04 17:02:09 +00:00
|
|
|
inline object
|
|
|
|
findMethodOrNull(Thread* t, object class_, const char* name, const char* spec)
|
|
|
|
{
|
|
|
|
PROTECT(t, class_);
|
|
|
|
object n = makeByteArray(t, "%s", name);
|
|
|
|
PROTECT(t, n);
|
|
|
|
object s = makeByteArray(t, "%s", spec);
|
|
|
|
return findInHierarchyOrNull(t, class_, n, s, findMethodInClass);
|
|
|
|
}
|
|
|
|
|
2008-06-25 16:28:11 +00:00
|
|
|
inline object
|
2009-08-13 15:17:05 +00:00
|
|
|
findVirtualMethod(Thread* t, object method, object class_)
|
2008-06-25 16:28:11 +00:00
|
|
|
{
|
|
|
|
return arrayBody(t, classVirtualTable(t, class_),
|
|
|
|
methodOffset(t, method));
|
|
|
|
}
|
|
|
|
|
2009-09-19 00:01:54 +00:00
|
|
|
inline object
|
|
|
|
findInterfaceMethod(Thread* t, object method, object class_)
|
|
|
|
{
|
|
|
|
assert(t, (classVmFlags(t, class_) & BootstrapFlag) == 0);
|
|
|
|
|
|
|
|
object interface = methodClass(t, method);
|
|
|
|
object itable = classInterfaceTable(t, class_);
|
|
|
|
for (unsigned i = 0; i < arrayLength(t, itable); i += 2) {
|
|
|
|
if (arrayBody(t, itable, i) == interface) {
|
|
|
|
return arrayBody(t, arrayBody(t, itable, i + 1),
|
|
|
|
methodOffset(t, method));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
abort(t);
|
|
|
|
}
|
|
|
|
|
2007-07-14 17:31:01 +00:00
|
|
|
inline unsigned
|
2007-08-19 19:45:51 +00:00
|
|
|
objectArrayLength(Thread* t UNUSED, object array)
|
2007-07-14 17:31:01 +00:00
|
|
|
{
|
|
|
|
assert(t, classFixedSize(t, objectClass(t, array)) == BytesPerWord * 2);
|
|
|
|
assert(t, classArrayElementSize(t, objectClass(t, array)) == BytesPerWord);
|
|
|
|
return cast<uintptr_t>(array, BytesPerWord);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline object&
|
2007-08-19 19:45:51 +00:00
|
|
|
objectArrayBody(Thread* t UNUSED, object array, unsigned index)
|
2007-07-14 17:31:01 +00:00
|
|
|
{
|
|
|
|
assert(t, classFixedSize(t, objectClass(t, array)) == BytesPerWord * 2);
|
|
|
|
assert(t, classArrayElementSize(t, objectClass(t, array)) == BytesPerWord);
|
|
|
|
assert(t, classObjectMask(t, objectClass(t, array))
|
|
|
|
== classObjectMask(t, arrayBody
|
2007-09-24 01:39:03 +00:00
|
|
|
(t, t->m->types, Machine::ArrayType)));
|
2009-09-19 00:01:54 +00:00
|
|
|
return cast<object>(array, ArrayBody + (index * BytesPerWord));
|
2007-07-14 17:31:01 +00:00
|
|
|
}
|
|
|
|
|
2007-11-20 22:24:02 +00:00
|
|
|
unsigned
|
|
|
|
parameterFootprint(Thread* t, const char* s, bool static_);
|
|
|
|
|
2007-07-06 23:50:26 +00:00
|
|
|
void
|
|
|
|
addFinalizer(Thread* t, object target, void (*finalize)(Thread*, object));
|
|
|
|
|
2010-02-05 00:56:21 +00:00
|
|
|
inline bool
|
|
|
|
atomicCompareAndSwapObject(Thread* t, object target, unsigned offset,
|
|
|
|
object old, object new_)
|
|
|
|
{
|
|
|
|
if (atomicCompareAndSwap(&cast<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, object monitor)
|
|
|
|
{
|
|
|
|
PROTECT(t, monitor);
|
|
|
|
|
|
|
|
object node = makeMonitorNode(t, t, 0);
|
|
|
|
|
|
|
|
while (true) {
|
|
|
|
object tail = monitorAcquireTail(t, monitor);
|
|
|
|
|
|
|
|
loadMemoryBarrier();
|
|
|
|
|
|
|
|
object next = monitorNodeNext(t, tail);
|
|
|
|
|
|
|
|
loadMemoryBarrier();
|
|
|
|
|
|
|
|
if (tail == monitorAcquireTail(t, monitor)) {
|
|
|
|
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, object monitor, bool remove)
|
|
|
|
{
|
|
|
|
while (true) {
|
|
|
|
object head = monitorAcquireHead(t, monitor);
|
|
|
|
|
|
|
|
loadMemoryBarrier();
|
|
|
|
|
|
|
|
object tail = monitorAcquireTail(t, monitor);
|
|
|
|
|
|
|
|
loadMemoryBarrier();
|
|
|
|
|
|
|
|
object next = monitorNodeNext(t, head);
|
|
|
|
|
|
|
|
loadMemoryBarrier();
|
|
|
|
|
|
|
|
if (head == monitorAcquireHead(t, monitor)) {
|
|
|
|
if (head == tail) {
|
|
|
|
if (next) {
|
|
|
|
atomicCompareAndSwapObject
|
|
|
|
(t, monitor, MonitorAcquireTail, tail, next);
|
|
|
|
} else {
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
Thread* value = static_cast<Thread*>(monitorNodeValue(t, next));
|
|
|
|
if ((not remove)
|
|
|
|
or atomicCompareAndSwapObject
|
|
|
|
(t, monitor, MonitorAcquireHead, head, next))
|
|
|
|
{
|
|
|
|
return value;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
inline bool
|
|
|
|
monitorTryAcquire(Thread* t, object monitor)
|
|
|
|
{
|
|
|
|
if (monitorOwner(t, monitor) == t
|
|
|
|
or (monitorAtomicPollAcquire(t, monitor, false) == 0
|
|
|
|
and atomicCompareAndSwap
|
|
|
|
(reinterpret_cast<uintptr_t*>(&monitorOwner(t, monitor)), 0,
|
|
|
|
reinterpret_cast<uintptr_t>(t))))
|
|
|
|
{
|
|
|
|
++ monitorDepth(t, monitor);
|
|
|
|
return true;
|
|
|
|
} else {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void
|
|
|
|
monitorAcquire(Thread* t, object monitor)
|
|
|
|
{
|
|
|
|
if (not monitorTryAcquire(t, monitor)) {
|
|
|
|
PROTECT(t, monitor);
|
|
|
|
|
|
|
|
ACQUIRE(t, t->lock);
|
|
|
|
|
|
|
|
monitorAtomicAppendAcquire(t, monitor);
|
|
|
|
|
|
|
|
// 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*>(&monitorOwner(t, monitor)), 0,
|
|
|
|
reinterpret_cast<uintptr_t>(t))))
|
|
|
|
{
|
|
|
|
ENTER(t, Thread::IdleState);
|
|
|
|
|
|
|
|
t->lock->wait(t->systemThread, 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
expect(t, t == monitorAtomicPollAcquire(t, monitor, true));
|
|
|
|
|
|
|
|
++ monitorDepth(t, monitor);
|
|
|
|
}
|
|
|
|
|
|
|
|
assert(t, monitorOwner(t, monitor) == t);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void
|
|
|
|
monitorRelease(Thread* t, object monitor)
|
|
|
|
{
|
|
|
|
expect(t, monitorOwner(t, monitor) == t);
|
|
|
|
|
|
|
|
if (-- monitorDepth(t, monitor) == 0) {
|
|
|
|
monitorOwner(t, monitor) = 0;
|
|
|
|
|
|
|
|
storeLoadMemoryBarrier();
|
|
|
|
|
|
|
|
Thread* next = monitorAtomicPollAcquire(t, monitor, false);
|
|
|
|
|
|
|
|
if (next) {
|
|
|
|
ACQUIRE(t, next->lock);
|
|
|
|
|
|
|
|
next->lock->notify(t->systemThread);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void
|
|
|
|
monitorAppendWait(Thread* t, object monitor)
|
|
|
|
{
|
|
|
|
assert(t, monitorOwner(t, monitor) == t);
|
|
|
|
|
2010-09-14 16:49:41 +00:00
|
|
|
expect(t, (t->flags & Thread::WaitingFlag) == 0);
|
2010-02-05 00:56:21 +00:00
|
|
|
expect(t, t->waitNext == 0);
|
|
|
|
|
2010-09-14 16:49:41 +00:00
|
|
|
atomicOr(&(t->flags), Thread::WaitingFlag);
|
2010-02-05 00:56:21 +00:00
|
|
|
|
|
|
|
if (monitorWaitTail(t, monitor)) {
|
|
|
|
static_cast<Thread*>(monitorWaitTail(t, monitor))->waitNext = t;
|
|
|
|
} else {
|
|
|
|
monitorWaitHead(t, monitor) = t;
|
|
|
|
}
|
|
|
|
|
|
|
|
monitorWaitTail(t, monitor) = t;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void
|
|
|
|
monitorRemoveWait(Thread* t, object monitor)
|
|
|
|
{
|
|
|
|
assert(t, monitorOwner(t, monitor) == t);
|
|
|
|
|
|
|
|
Thread* previous = 0;
|
|
|
|
for (Thread* current = static_cast<Thread*>(monitorWaitHead(t, monitor));
|
|
|
|
current; current = current->waitNext)
|
|
|
|
{
|
|
|
|
if (t == current) {
|
|
|
|
if (t == monitorWaitHead(t, monitor)) {
|
|
|
|
monitorWaitHead(t, monitor) = t->waitNext;
|
|
|
|
} else {
|
|
|
|
previous->waitNext = t->waitNext;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (t == monitorWaitTail(t, monitor)) {
|
|
|
|
assert(t, t->waitNext == 0);
|
|
|
|
monitorWaitTail(t, monitor) = previous;
|
|
|
|
}
|
|
|
|
|
|
|
|
t->waitNext = 0;
|
2010-09-14 16:49:41 +00:00
|
|
|
atomicAnd(&(t->flags), ~Thread::WaitingFlag);
|
2010-02-05 00:56:21 +00:00
|
|
|
|
|
|
|
return;
|
|
|
|
} else {
|
|
|
|
previous = current;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
abort(t);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline bool
|
|
|
|
monitorFindWait(Thread* t, object monitor)
|
|
|
|
{
|
|
|
|
assert(t, monitorOwner(t, monitor) == t);
|
|
|
|
|
|
|
|
for (Thread* current = static_cast<Thread*>(monitorWaitHead(t, monitor));
|
|
|
|
current; current = current->waitNext)
|
|
|
|
{
|
|
|
|
if (t == current) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline bool
|
|
|
|
monitorWait(Thread* t, object monitor, int64_t time)
|
|
|
|
{
|
|
|
|
expect(t, monitorOwner(t, monitor) == t);
|
|
|
|
|
|
|
|
bool interrupted;
|
|
|
|
unsigned depth;
|
|
|
|
|
|
|
|
PROTECT(t, monitor);
|
|
|
|
|
|
|
|
{ ACQUIRE(t, t->lock);
|
|
|
|
|
|
|
|
monitorAppendWait(t, monitor);
|
|
|
|
|
|
|
|
depth = monitorDepth(t, monitor);
|
|
|
|
monitorDepth(t, monitor) = 1;
|
|
|
|
|
|
|
|
monitorRelease(t, monitor);
|
|
|
|
|
|
|
|
ENTER(t, Thread::IdleState);
|
|
|
|
|
|
|
|
interrupted = t->lock->wait(t->systemThread, time);
|
|
|
|
}
|
|
|
|
|
|
|
|
monitorAcquire(t, monitor);
|
|
|
|
|
|
|
|
monitorDepth(t, monitor) = depth;
|
|
|
|
|
2010-09-14 16:49:41 +00:00
|
|
|
if (t->flags & Thread::WaitingFlag) {
|
2010-02-05 00:56:21 +00:00
|
|
|
monitorRemoveWait(t, monitor);
|
|
|
|
} else {
|
|
|
|
expect(t, not monitorFindWait(t, monitor));
|
|
|
|
}
|
|
|
|
|
|
|
|
assert(t, monitorOwner(t, monitor) == t);
|
|
|
|
|
|
|
|
return interrupted;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline Thread*
|
|
|
|
monitorPollWait(Thread* t, object monitor)
|
|
|
|
{
|
|
|
|
assert(t, monitorOwner(t, monitor) == t);
|
|
|
|
|
|
|
|
Thread* next = static_cast<Thread*>(monitorWaitHead(t, monitor));
|
|
|
|
|
|
|
|
if (next) {
|
|
|
|
monitorWaitHead(t, monitor) = next->waitNext;
|
2010-09-14 16:49:41 +00:00
|
|
|
atomicAnd(&(next->flags), ~Thread::WaitingFlag);
|
2010-02-05 00:56:21 +00:00
|
|
|
next->waitNext = 0;
|
|
|
|
if (next == monitorWaitTail(t, monitor)) {
|
|
|
|
monitorWaitTail(t, monitor) = 0;
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
assert(t, monitorWaitTail(t, monitor) == 0);
|
|
|
|
}
|
|
|
|
|
|
|
|
return next;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline bool
|
|
|
|
monitorNotify(Thread* t, object monitor)
|
|
|
|
{
|
|
|
|
expect(t, monitorOwner(t, monitor) == 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, object monitor)
|
|
|
|
{
|
|
|
|
PROTECT(t, monitor);
|
|
|
|
|
|
|
|
while (monitorNotify(t, monitor)) { }
|
|
|
|
}
|
|
|
|
|
2010-09-17 01:43:27 +00:00
|
|
|
class ObjectMonitorResource {
|
|
|
|
public:
|
|
|
|
ObjectMonitorResource(Thread* t, object o): o(o), protector(t, &(this->o)) {
|
|
|
|
monitorAcquire(protector.t, o);
|
|
|
|
}
|
|
|
|
|
|
|
|
~ObjectMonitorResource() {
|
|
|
|
monitorRelease(protector.t, o);
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
object o;
|
|
|
|
Thread::SingleProtector protector;
|
|
|
|
};
|
|
|
|
|
2010-02-05 00:56:21 +00:00
|
|
|
object
|
2007-11-27 22:23:00 +00:00
|
|
|
objectMonitor(Thread* t, object o, bool createNew);
|
2007-07-06 23:50:26 +00:00
|
|
|
|
2007-07-07 23:47:35 +00:00
|
|
|
inline void
|
|
|
|
acquire(Thread* t, object o)
|
|
|
|
{
|
2008-01-19 20:12:16 +00:00
|
|
|
unsigned hash;
|
|
|
|
if (DebugMonitors) {
|
|
|
|
hash = objectHash(t, o);
|
|
|
|
}
|
|
|
|
|
2010-02-05 00:56:21 +00:00
|
|
|
object m = objectMonitor(t, o, true);
|
2007-07-11 04:19:26 +00:00
|
|
|
|
|
|
|
if (DebugMonitors) {
|
2010-09-17 01:43:27 +00:00
|
|
|
fprintf(stderr, "thread %p acquires %p for %x\n", t, m, hash);
|
2007-07-11 04:19:26 +00:00
|
|
|
}
|
|
|
|
|
2010-02-05 00:56:21 +00:00
|
|
|
monitorAcquire(t, m);
|
2007-07-07 23:47:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
inline void
|
|
|
|
release(Thread* t, object o)
|
|
|
|
{
|
2008-01-19 20:12:16 +00:00
|
|
|
unsigned hash;
|
|
|
|
if (DebugMonitors) {
|
|
|
|
hash = objectHash(t, o);
|
|
|
|
}
|
|
|
|
|
2010-02-05 00:56:21 +00:00
|
|
|
object m = objectMonitor(t, o, false);
|
2007-07-11 04:19:26 +00:00
|
|
|
|
|
|
|
if (DebugMonitors) {
|
2010-09-17 01:43:27 +00:00
|
|
|
fprintf(stderr, "thread %p releases %p for %x\n", t, m, hash);
|
2007-07-11 04:19:26 +00:00
|
|
|
}
|
|
|
|
|
2010-02-05 00:56:21 +00:00
|
|
|
monitorRelease(t, m);
|
2007-07-07 23:47:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
inline void
|
|
|
|
wait(Thread* t, object o, int64_t milliseconds)
|
|
|
|
{
|
2008-01-19 20:12:16 +00:00
|
|
|
unsigned hash;
|
|
|
|
if (DebugMonitors) {
|
|
|
|
hash = objectHash(t, o);
|
|
|
|
}
|
|
|
|
|
2010-02-05 00:56:21 +00:00
|
|
|
object m = objectMonitor(t, o, false);
|
2007-07-11 04:19:26 +00:00
|
|
|
|
|
|
|
if (DebugMonitors) {
|
2009-10-14 16:01:37 +00:00
|
|
|
fprintf(stderr, "thread %p waits %d millis on %p for %x\n",
|
|
|
|
t, static_cast<int>(milliseconds), m, hash);
|
2007-07-11 04:19:26 +00:00
|
|
|
}
|
|
|
|
|
2010-02-05 00:56:21 +00:00
|
|
|
if (m and monitorOwner(t, m) == t) {
|
|
|
|
PROTECT(t, m);
|
|
|
|
|
|
|
|
bool interrupted = monitorWait(t, m, milliseconds);
|
2007-07-28 21:28:25 +00:00
|
|
|
|
|
|
|
if (interrupted) {
|
2010-09-10 21:05:29 +00:00
|
|
|
t->exception = t->m->classpath->makeThrowable
|
|
|
|
(t, Machine::InterruptedExceptionType);
|
2007-07-28 21:28:25 +00:00
|
|
|
}
|
2007-07-07 23:47:35 +00:00
|
|
|
} else {
|
2010-09-10 21:05:29 +00:00
|
|
|
t->exception = t->m->classpath->makeThrowable
|
|
|
|
(t, Machine::IllegalMonitorStateExceptionType);
|
2007-07-07 23:47:35 +00:00
|
|
|
}
|
2007-07-11 04:19:26 +00:00
|
|
|
|
|
|
|
if (DebugMonitors) {
|
|
|
|
fprintf(stderr, "thread %p wakes up on %p for %x\n",
|
2008-01-19 20:12:16 +00:00
|
|
|
t, m, hash);
|
2007-07-11 04:19:26 +00:00
|
|
|
}
|
2007-07-19 23:45:44 +00:00
|
|
|
|
|
|
|
stress(t);
|
2007-07-07 23:47:35 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
inline void
|
|
|
|
notify(Thread* t, object o)
|
|
|
|
{
|
2008-01-19 20:12:16 +00:00
|
|
|
unsigned hash;
|
|
|
|
if (DebugMonitors) {
|
|
|
|
hash = objectHash(t, o);
|
|
|
|
}
|
|
|
|
|
2010-02-05 00:56:21 +00:00
|
|
|
object m = objectMonitor(t, o, false);
|
2007-07-11 04:19:26 +00:00
|
|
|
|
|
|
|
if (DebugMonitors) {
|
|
|
|
fprintf(stderr, "thread %p notifies on %p for %x\n",
|
2008-01-19 20:12:16 +00:00
|
|
|
t, m, hash);
|
2007-07-11 04:19:26 +00:00
|
|
|
}
|
|
|
|
|
2010-02-05 00:56:21 +00:00
|
|
|
if (m and monitorOwner(t, m) == t) {
|
|
|
|
monitorNotify(t, m);
|
2007-07-07 23:47:35 +00:00
|
|
|
} else {
|
2010-09-10 21:05:29 +00:00
|
|
|
t->exception = t->m->classpath->makeThrowable
|
|
|
|
(t, Machine::IllegalMonitorStateExceptionType);
|
2007-07-07 23:47:35 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void
|
|
|
|
notifyAll(Thread* t, object o)
|
|
|
|
{
|
2010-02-05 00:56:21 +00:00
|
|
|
object m = objectMonitor(t, o, false);
|
2007-07-11 04:19:26 +00:00
|
|
|
|
|
|
|
if (DebugMonitors) {
|
|
|
|
fprintf(stderr, "thread %p notifies all on %p for %x\n",
|
|
|
|
t, m, objectHash(t, o));
|
|
|
|
}
|
|
|
|
|
2010-02-05 00:56:21 +00:00
|
|
|
if (m and monitorOwner(t, m) == t) {
|
|
|
|
monitorNotifyAll(t, m);
|
2007-07-07 23:47:35 +00:00
|
|
|
} else {
|
2010-09-10 21:05:29 +00:00
|
|
|
t->exception = t->m->classpath->makeThrowable
|
|
|
|
(t, Machine::IllegalMonitorStateExceptionType);
|
2007-07-07 23:47:35 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-11-16 17:50:19 +00:00
|
|
|
inline bool
|
|
|
|
zombified(Thread* t)
|
|
|
|
{
|
|
|
|
return t->state == Thread::ZombieState
|
|
|
|
or t->state == Thread::JoinedState;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline bool
|
|
|
|
acquireSystem(Thread* t, Thread* target)
|
|
|
|
{
|
|
|
|
ACQUIRE_RAW(t, t->m->stateLock);
|
|
|
|
|
|
|
|
if (not zombified(target)) {
|
|
|
|
atomicOr(&(target->flags), Thread::SystemFlag);
|
|
|
|
return true;
|
|
|
|
} else {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-07-20 14:36:31 +00:00
|
|
|
inline void
|
2010-11-16 17:50:19 +00:00
|
|
|
releaseSystem(Thread* t, Thread* target)
|
2007-07-20 14:36:31 +00:00
|
|
|
{
|
2010-11-16 17:50:19 +00:00
|
|
|
ACQUIRE_RAW(t, t->m->stateLock);
|
|
|
|
|
|
|
|
assert(t, not zombified(target));
|
|
|
|
|
|
|
|
atomicAnd(&(target->flags), ~Thread::SystemFlag);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void
|
|
|
|
interrupt(Thread* t, Thread* target)
|
|
|
|
{
|
|
|
|
if (acquireSystem(t, target)) {
|
2010-11-09 22:46:16 +00:00
|
|
|
target->systemThread->interrupt();
|
2010-11-16 17:50:19 +00:00
|
|
|
releaseSystem(t, target);
|
2010-11-09 22:46:16 +00:00
|
|
|
}
|
2007-07-20 14:36:31 +00:00
|
|
|
}
|
|
|
|
|
2009-08-24 23:51:31 +00:00
|
|
|
inline void
|
|
|
|
setDaemon(Thread* t, object thread, bool daemon)
|
|
|
|
{
|
|
|
|
ACQUIRE_RAW(t, t->m->stateLock);
|
|
|
|
|
2009-08-27 00:26:44 +00:00
|
|
|
if ((threadDaemon(t, thread) != 0) != daemon) {
|
2009-08-24 23:51:31 +00:00
|
|
|
threadDaemon(t, thread) = daemon;
|
2010-09-14 16:49:41 +00:00
|
|
|
|
|
|
|
Thread* p = reinterpret_cast<Thread*>(threadPeer(t, thread));
|
|
|
|
if (daemon) {
|
|
|
|
atomicOr(&(p->flags), Thread::DaemonFlag);
|
|
|
|
} else {
|
|
|
|
atomicAnd(&(p->flags), ~Thread::DaemonFlag);
|
|
|
|
}
|
2009-08-24 23:51:31 +00:00
|
|
|
|
|
|
|
if (daemon) {
|
|
|
|
++ t->m->daemonCount;
|
|
|
|
} else {
|
|
|
|
expect(t, t->m->daemonCount);
|
|
|
|
-- t->m->daemonCount;
|
|
|
|
}
|
|
|
|
|
|
|
|
t->m->stateLock->notifyAll(t->systemThread);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2007-07-29 00:02:32 +00:00
|
|
|
object
|
|
|
|
intern(Thread* t, object s);
|
|
|
|
|
2008-11-21 23:20:35 +00:00
|
|
|
void
|
|
|
|
walk(Thread* t, Heap::Walker* w, object o, unsigned start);
|
|
|
|
|
2008-11-11 15:20:49 +00:00
|
|
|
int
|
|
|
|
walkNext(Thread* t, object o, int previous);
|
|
|
|
|
|
|
|
void
|
|
|
|
visitRoots(Machine* m, Heap::Visitor* v);
|
|
|
|
|
2007-09-24 13:46:48 +00:00
|
|
|
inline jobject
|
|
|
|
makeLocalReference(Thread* t, object o)
|
|
|
|
{
|
|
|
|
return t->m->processor->makeLocalReference(t, o);
|
|
|
|
}
|
2007-09-24 01:39:03 +00:00
|
|
|
|
2007-09-24 13:46:48 +00:00
|
|
|
inline void
|
|
|
|
disposeLocalReference(Thread* t, jobject r)
|
|
|
|
{
|
|
|
|
t->m->processor->disposeLocalReference(t, r);
|
|
|
|
}
|
2007-09-24 01:39:03 +00:00
|
|
|
|
2007-11-05 14:28:46 +00:00
|
|
|
inline bool
|
|
|
|
methodVirtual(Thread* t, object method)
|
|
|
|
{
|
|
|
|
return (methodFlags(t, method) & (ACC_STATIC | ACC_PRIVATE)) == 0
|
|
|
|
and byteArrayBody(t, methodName(t, method), 0) != '<';
|
|
|
|
}
|
|
|
|
|
2007-11-05 21:40:17 +00:00
|
|
|
inline unsigned
|
|
|
|
singletonMaskSize(unsigned count)
|
|
|
|
{
|
|
|
|
if (count) {
|
|
|
|
return ceiling(count + 2, BitsPerWord);
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline unsigned
|
|
|
|
singletonMaskSize(Thread* t, object singleton)
|
|
|
|
{
|
|
|
|
unsigned length = singletonLength(t, singleton);
|
|
|
|
if (length) {
|
|
|
|
return ceiling(length + 2, BitsPerWord + 1);
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline unsigned
|
|
|
|
singletonCount(Thread* t, object singleton)
|
|
|
|
{
|
|
|
|
return singletonLength(t, singleton) - singletonMaskSize(t, singleton);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline uint32_t*
|
|
|
|
singletonMask(Thread* t, object singleton)
|
|
|
|
{
|
|
|
|
assert(t, singletonLength(t, singleton));
|
|
|
|
return reinterpret_cast<uint32_t*>
|
|
|
|
(&singletonBody(t, singleton, singletonCount(t, singleton)));
|
|
|
|
}
|
|
|
|
|
|
|
|
inline void
|
|
|
|
singletonMarkObject(Thread* t, object singleton, unsigned index)
|
|
|
|
{
|
|
|
|
singletonMask(t, singleton)[(index + 2) / 32]
|
|
|
|
|= (static_cast<uint32_t>(1) << ((index + 2) % 32));
|
|
|
|
}
|
|
|
|
|
|
|
|
inline bool
|
|
|
|
singletonIsObject(Thread* t, object singleton, unsigned index)
|
|
|
|
{
|
|
|
|
assert(t, index < singletonCount(t, singleton));
|
|
|
|
|
|
|
|
return (singletonMask(t, singleton)[(index + 2) / 32]
|
|
|
|
& (static_cast<uint32_t>(1) << ((index + 2) % 32))) != 0;
|
|
|
|
}
|
|
|
|
|
2007-11-06 15:29:05 +00:00
|
|
|
inline object&
|
2007-11-05 21:40:17 +00:00
|
|
|
singletonObject(Thread* t, object singleton, unsigned index)
|
|
|
|
{
|
|
|
|
assert(t, singletonIsObject(t, singleton, index));
|
2007-11-06 15:29:05 +00:00
|
|
|
return reinterpret_cast<object&>(singletonBody(t, singleton, index));
|
2007-11-05 21:40:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
inline uintptr_t&
|
|
|
|
singletonValue(Thread* t, object singleton, unsigned index)
|
|
|
|
{
|
|
|
|
assert(t, not singletonIsObject(t, singleton, index));
|
|
|
|
return singletonBody(t, singleton, index);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline object
|
2009-03-04 03:05:48 +00:00
|
|
|
makeSingletonOfSize(Thread* t, unsigned count)
|
2007-11-05 21:40:17 +00:00
|
|
|
{
|
2009-03-04 03:05:48 +00:00
|
|
|
object o = makeSingleton(t, count + singletonMaskSize(count));
|
2008-04-11 21:00:18 +00:00
|
|
|
assert(t, singletonLength(t, o) == count + singletonMaskSize(t, o));
|
2007-11-05 21:40:17 +00:00
|
|
|
if (count) {
|
|
|
|
singletonMask(t, o)[0] = 1;
|
|
|
|
}
|
|
|
|
return o;
|
|
|
|
}
|
|
|
|
|
2009-08-10 19:20:23 +00:00
|
|
|
inline void
|
2009-10-18 02:11:03 +00:00
|
|
|
singletonSetBit(Thread* t, object singleton, unsigned start, unsigned index)
|
2009-08-10 19:20:23 +00:00
|
|
|
{
|
2009-10-18 02:11:03 +00:00
|
|
|
singletonValue(t, singleton, start + (index / BitsPerWord))
|
|
|
|
|= static_cast<uintptr_t>(1) << (index % BitsPerWord);
|
2009-08-10 19:20:23 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
inline bool
|
2009-10-18 02:11:03 +00:00
|
|
|
singletonBit(Thread* t, object singleton, unsigned start, unsigned index)
|
2009-08-10 19:20:23 +00:00
|
|
|
{
|
2009-10-18 02:11:03 +00:00
|
|
|
return (singletonValue(t, singleton, start + (index / BitsPerWord))
|
|
|
|
& (static_cast<uintptr_t>(1) << (index % BitsPerWord))) != 0;
|
2009-08-10 19:20:23 +00:00
|
|
|
}
|
|
|
|
|
2009-10-18 02:11:03 +00:00
|
|
|
inline unsigned
|
|
|
|
poolMaskSize(unsigned count)
|
|
|
|
{
|
|
|
|
return ceiling(count, BitsPerWord);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline unsigned
|
|
|
|
poolMaskSize(Thread* t, object pool)
|
|
|
|
{
|
|
|
|
return ceiling(singletonCount(t, pool), BitsPerWord + 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline unsigned
|
|
|
|
poolSize(Thread* t, object pool)
|
2009-08-10 19:20:23 +00:00
|
|
|
{
|
2009-10-18 02:11:03 +00:00
|
|
|
return singletonCount(t, pool) - poolMaskSize(t, pool);
|
2009-08-10 19:20:23 +00:00
|
|
|
}
|
|
|
|
|
2009-08-18 20:26:28 +00:00
|
|
|
inline object
|
|
|
|
resolveClassInObject(Thread* t, object loader, object container,
|
|
|
|
unsigned classOffset)
|
|
|
|
{
|
|
|
|
object o = cast<object>(container, classOffset);
|
2010-09-14 16:49:41 +00:00
|
|
|
if (objectClass(t, o) == type(t, Machine::ByteArrayType)) {
|
2009-08-18 20:26:28 +00:00
|
|
|
PROTECT(t, container);
|
|
|
|
|
|
|
|
o = resolveClass(t, loader, o);
|
|
|
|
if (UNLIKELY(t->exception)) return 0;
|
|
|
|
|
|
|
|
set(t, container, classOffset, o);
|
|
|
|
}
|
|
|
|
return o;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline object
|
|
|
|
resolveClassInPool(Thread* t, object loader, object method, unsigned index)
|
|
|
|
{
|
|
|
|
object o = singletonObject(t, codePool(t, methodCode(t, method)), index);
|
2010-09-14 16:49:41 +00:00
|
|
|
if (objectClass(t, o) == type(t, Machine::ReferenceType)) {
|
2009-08-18 20:26:28 +00:00
|
|
|
PROTECT(t, method);
|
|
|
|
|
|
|
|
o = resolveClass(t, loader, referenceName(t, o));
|
|
|
|
if (UNLIKELY(t->exception)) return 0;
|
|
|
|
|
|
|
|
set(t, codePool(t, methodCode(t, method)),
|
|
|
|
SingletonBody + (index * BytesPerWord), o);
|
|
|
|
}
|
|
|
|
return o;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline object
|
|
|
|
resolveClassInPool(Thread* t, object method, unsigned index)
|
|
|
|
{
|
|
|
|
return resolveClassInPool(t, classLoader(t, methodClass(t, method)),
|
|
|
|
method, index);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline object
|
|
|
|
resolve(Thread* t, object loader, object method, unsigned index,
|
|
|
|
object (*find)(vm::Thread*, object, object, object),
|
2010-09-10 21:05:29 +00:00
|
|
|
Machine::Type errorType)
|
2009-08-18 20:26:28 +00:00
|
|
|
{
|
|
|
|
object o = singletonObject(t, codePool(t, methodCode(t, method)), index);
|
2010-09-14 16:49:41 +00:00
|
|
|
if (objectClass(t, o) == type(t, Machine::ReferenceType))
|
2009-08-18 20:26:28 +00:00
|
|
|
{
|
|
|
|
PROTECT(t, method);
|
|
|
|
|
|
|
|
object reference = o;
|
|
|
|
PROTECT(t, reference);
|
|
|
|
|
|
|
|
object class_ = resolveClassInObject(t, loader, o, ReferenceClass);
|
|
|
|
if (UNLIKELY(t->exception)) return 0;
|
|
|
|
|
|
|
|
o = findInHierarchy
|
|
|
|
(t, class_, referenceName(t, reference), referenceSpec(t, reference),
|
2010-09-10 21:05:29 +00:00
|
|
|
find, errorType);
|
2009-08-18 20:26:28 +00:00
|
|
|
if (UNLIKELY(t->exception)) return 0;
|
|
|
|
|
|
|
|
set(t, codePool(t, methodCode(t, method)),
|
|
|
|
SingletonBody + (index * BytesPerWord), o);
|
|
|
|
}
|
|
|
|
|
|
|
|
return o;
|
|
|
|
}
|
|
|
|
|
|
|
|
inline object
|
|
|
|
resolveField(Thread* t, object loader, object method, unsigned index)
|
|
|
|
{
|
|
|
|
return resolve(t, loader, method, index, findFieldInClass,
|
2010-09-10 21:05:29 +00:00
|
|
|
Machine::NoSuchFieldErrorType);
|
2009-08-18 20:26:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
inline object
|
|
|
|
resolveField(Thread* t, object method, unsigned index)
|
|
|
|
{
|
|
|
|
return resolveField
|
|
|
|
(t, classLoader(t, methodClass(t, method)), method, index);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline object
|
|
|
|
resolveMethod(Thread* t, object loader, object method, unsigned index)
|
|
|
|
{
|
|
|
|
return resolve(t, loader, method, index, findMethodInClass,
|
2010-09-10 21:05:29 +00:00
|
|
|
Machine::NoSuchMethodErrorType);
|
2009-08-18 20:26:28 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
inline object
|
|
|
|
resolveMethod(Thread* t, object method, unsigned index)
|
|
|
|
{
|
|
|
|
return resolveMethod
|
|
|
|
(t, classLoader(t, methodClass(t, method)), method, index);
|
|
|
|
}
|
|
|
|
|
2010-11-26 19:41:31 +00:00
|
|
|
object
|
|
|
|
vectorAppend(Thread*, object, object);
|
|
|
|
|
2010-09-01 16:13:52 +00:00
|
|
|
inline object
|
2010-11-26 19:41:31 +00:00
|
|
|
getClassRuntimeData(Thread* t, object c)
|
2010-09-01 16:13:52 +00:00
|
|
|
{
|
2010-11-26 19:41:31 +00:00
|
|
|
if (classRuntimeDataIndex(t, c) == 0) {
|
2010-09-10 21:05:29 +00:00
|
|
|
PROTECT(t, c);
|
|
|
|
|
2010-09-01 16:13:52 +00:00
|
|
|
ACQUIRE(t, t->m->classLock);
|
|
|
|
|
2010-11-26 19:41:31 +00:00
|
|
|
if (classRuntimeDataIndex(t, c) == 0) {
|
|
|
|
object runtimeData = makeClassRuntimeData(t, 0, 0, 0);
|
|
|
|
|
|
|
|
setRoot(t, Machine::ClassRuntimeDataTable, vectorAppend
|
|
|
|
(t, root(t, Machine::ClassRuntimeDataTable), runtimeData));
|
|
|
|
|
|
|
|
classRuntimeDataIndex(t, c) = vectorSize
|
|
|
|
(t, root(t, Machine::ClassRuntimeDataTable));
|
|
|
|
}
|
2010-09-01 16:13:52 +00:00
|
|
|
}
|
|
|
|
|
2010-11-26 19:41:31 +00:00
|
|
|
return vectorBody(t, root(t, Machine::ClassRuntimeDataTable),
|
|
|
|
classRuntimeDataIndex(t, c) - 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline object
|
|
|
|
getMethodRuntimeData(Thread* t, object method)
|
|
|
|
{
|
|
|
|
if (methodRuntimeDataIndex(t, method) == 0) {
|
|
|
|
PROTECT(t, method);
|
|
|
|
|
|
|
|
ACQUIRE(t, t->m->classLock);
|
|
|
|
|
|
|
|
if (methodRuntimeDataIndex(t, method) == 0) {
|
|
|
|
object runtimeData = makeMethodRuntimeData(t, 0);
|
|
|
|
|
|
|
|
setRoot(t, Machine::MethodRuntimeDataTable, vectorAppend
|
|
|
|
(t, root(t, Machine::MethodRuntimeDataTable), runtimeData));
|
|
|
|
|
|
|
|
methodRuntimeDataIndex(t, method) = vectorSize
|
|
|
|
(t, root(t, Machine::MethodRuntimeDataTable));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return vectorBody(t, root(t, Machine::MethodRuntimeDataTable),
|
|
|
|
methodRuntimeDataIndex(t, method) - 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
inline object
|
|
|
|
getJClass(Thread* t, object c)
|
|
|
|
{
|
|
|
|
object jclass = classRuntimeDataJclass(t, getClassRuntimeData(t, c));
|
2010-09-01 16:13:52 +00:00
|
|
|
if (jclass == 0) {
|
2010-09-10 21:05:29 +00:00
|
|
|
PROTECT(t, c);
|
|
|
|
|
2010-09-01 16:13:52 +00:00
|
|
|
ACQUIRE(t, t->m->classLock);
|
|
|
|
|
2010-11-26 19:41:31 +00:00
|
|
|
jclass = classRuntimeDataJclass(t, getClassRuntimeData(t, c));
|
|
|
|
if (jclass == 0) {
|
|
|
|
jclass = t->m->classpath->makeJclass(t, c);
|
2010-09-01 16:13:52 +00:00
|
|
|
|
2010-11-26 19:41:31 +00:00
|
|
|
set(t, getClassRuntimeData(t, c), ClassRuntimeDataJclass, jclass);
|
|
|
|
}
|
2010-09-01 16:13:52 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
return jclass;
|
|
|
|
}
|
|
|
|
|
2010-09-10 21:05:29 +00:00
|
|
|
inline object
|
|
|
|
primitiveClass(Thread* t, char name)
|
|
|
|
{
|
|
|
|
switch (name) {
|
2010-09-14 16:49:41 +00:00
|
|
|
case 'B': return type(t, Machine::JbyteType);
|
|
|
|
case 'C': return type(t, Machine::JcharType);
|
|
|
|
case 'D': return type(t, Machine::JdoubleType);
|
|
|
|
case 'F': return type(t, Machine::JfloatType);
|
|
|
|
case 'I': return type(t, Machine::JintType);
|
|
|
|
case 'J': return type(t, Machine::JlongType);
|
|
|
|
case 'S': return type(t, Machine::JshortType);
|
|
|
|
case 'V': return type(t, Machine::JvoidType);
|
|
|
|
case 'Z': return type(t, Machine::JbooleanType);
|
2010-09-10 21:05:29 +00:00
|
|
|
default:
|
|
|
|
t->exception = t->m->classpath->makeThrowable
|
|
|
|
(t, Machine::IllegalArgumentExceptionType);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
2010-09-14 16:49:41 +00:00
|
|
|
|
|
|
|
inline void
|
|
|
|
registerNative(Thread* t, object method, void* function)
|
|
|
|
{
|
|
|
|
PROTECT(t, method);
|
|
|
|
|
|
|
|
expect(t, methodFlags(t, method) & ACC_NATIVE);
|
|
|
|
|
|
|
|
object native = makeNative(t, function, false);
|
2010-11-26 19:41:31 +00:00
|
|
|
PROTECT(t, native);
|
|
|
|
|
|
|
|
object runtimeData = getMethodRuntimeData(t, method);
|
2010-09-14 16:49:41 +00:00
|
|
|
|
2010-11-26 19:41:31 +00:00
|
|
|
// ensure other threads only see the methodRuntimeDataNative field
|
|
|
|
// populated once the object it points to has been populated:
|
2010-09-14 16:49:41 +00:00
|
|
|
storeStoreMemoryBarrier();
|
|
|
|
|
2010-11-26 19:41:31 +00:00
|
|
|
set(t, runtimeData, MethodRuntimeDataNative, native);
|
2010-09-14 16:49:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
inline void
|
2010-09-25 21:54:01 +00:00
|
|
|
unregisterNatives(Thread* t, object c)
|
|
|
|
{
|
2010-09-14 16:49:41 +00:00
|
|
|
if (classMethodTable(t, c)) {
|
|
|
|
for (unsigned i = 0; i < arrayLength(t, classMethodTable(t, c)); ++i) {
|
|
|
|
object method = arrayBody(t, classMethodTable(t, c), i);
|
|
|
|
if (methodFlags(t, method) & ACC_NATIVE) {
|
2010-11-26 19:41:31 +00:00
|
|
|
set(t, getMethodRuntimeData(t, method), MethodRuntimeDataNative, 0);
|
2010-09-14 16:49:41 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2010-09-10 21:05:29 +00:00
|
|
|
|
2010-09-25 21:54:01 +00:00
|
|
|
object
|
|
|
|
getCaller(Thread* t, unsigned target);
|
|
|
|
|
2010-09-10 21:05:29 +00:00
|
|
|
object
|
|
|
|
defineClass(Thread* t, object loader, const uint8_t* buffer, unsigned length);
|
|
|
|
|
2008-11-11 15:20:49 +00:00
|
|
|
void
|
|
|
|
dumpHeap(Thread* t, FILE* out);
|
|
|
|
|
2010-11-04 17:02:09 +00:00
|
|
|
inline object
|
|
|
|
methodClone(Thread* t, object method)
|
|
|
|
{
|
|
|
|
return makeMethod
|
|
|
|
(t, methodVmFlags(t, method),
|
|
|
|
methodReturnCode(t, method),
|
|
|
|
methodParameterCount(t, method),
|
|
|
|
methodParameterFootprint(t, method),
|
|
|
|
methodFlags(t, method),
|
|
|
|
methodOffset(t, method),
|
|
|
|
methodNativeID(t, method),
|
2010-11-26 19:41:31 +00:00
|
|
|
methodRuntimeDataIndex(t, method),
|
2010-11-04 17:02:09 +00:00
|
|
|
methodName(t, method),
|
|
|
|
methodSpec(t, method),
|
|
|
|
methodAddendum(t, method),
|
|
|
|
methodClass(t, method),
|
|
|
|
methodCode(t, method));
|
|
|
|
}
|
|
|
|
|
2007-06-25 02:02:24 +00:00
|
|
|
} // namespace vm
|
|
|
|
|
2008-01-28 15:12:06 +00:00
|
|
|
void
|
|
|
|
vmPrintTrace(vm::Thread* t);
|
|
|
|
|
2009-06-11 15:45:35 +00:00
|
|
|
void*
|
|
|
|
vmAddressFromLine(vm::Thread* t, vm::object m, unsigned line);
|
|
|
|
|
2007-07-06 23:50:26 +00:00
|
|
|
#endif//MACHINE_H
|