/* Copyright (c) 2010, Avian Contributors Permission to use, copy, modify, and/or distribute this software for any purpose with or without fee is hereby granted, provided that the above copyright notice and this permission notice appear in all copies. There is NO WARRANTY for this software. See license.txt for details. */ #include "machine.h" #include "classpath-common.h" #include "util.h" // todo: move platform-specific stuff into system.h and implementations #ifdef PLATFORM_WINDOWS # include # include # include # include # define CLOSE _close # define READ _read # define WRITE _write # ifdef _MSC_VER # define S_ISREG(x) ((x) | _S_IFREG) # define S_ISDIR(x) ((x) | _S_IFDIR) # define S_IRUSR _S_IREAD # define S_IWUSR _S_IWRITE # else # define OPEN _wopen # define CREAT _wcreat # endif # define LIBRARY_PREFIX "" # define LIBRARY_SUFFIX ".dll" #else // not PLATFORM_WINDOWS # include # include # include # include # include # include # include # include # define OPEN open # define CLOSE close # define READ read # define WRITE write # define FSTAT fstat # define LSEEK lseek # define LIBRARY_PREFIX "lib" # define LIBRARY_SUFFIX ".so" #endif // not PLATFORM_WINDOWS using namespace vm; namespace { #ifdef _MSC_VER inline int OPEN(string_t path, int mask, int mode) { int fd; if (_wsopen_s(&fd, path, mask, _SH_DENYNO, mode) == 0) { return fd; } else { return -1; } } inline int CREAT(string_t path, int mode) { return OPEN(path, _O_CREAT, mode); } #endif namespace local { const unsigned InterfaceVersion = 4; const unsigned PageSize = 4 * 1024; const int VirtualFileBase = 1000000000; Machine* globalMachine; const char* primitiveName(Thread* t, object c) { if (c == primitiveClass(t, 'V')) { return "void"; } else if (c == primitiveClass(t, 'Z')) { return "boolean"; } else if (c == primitiveClass(t, 'B')) { return "byte"; } else if (c == primitiveClass(t, 'C')) { return "char"; } else if (c == primitiveClass(t, 'S')) { return "short"; } else if (c == primitiveClass(t, 'I')) { return "int"; } else if (c == primitiveClass(t, 'F')) { return "float"; } else if (c == primitiveClass(t, 'J')) { return "long"; } else if (c == primitiveClass(t, 'D')) { return "double"; } else { abort(t); } } object getClassName(Thread* t, object c) { if (className(t, c) == 0) { if (classVmFlags(t, c) & PrimitiveFlag) { PROTECT(t, c); object name = makeByteArray(t, primitiveName(t, c)); set(t, c, ClassName, name); } else { abort(t); } } return className(t, c); } object makeClassNameString(Thread* t, object name) { RUNTIME_ARRAY(char, s, byteArrayLength(t, name)); replace('/', '.', RUNTIME_ARRAY_BODY(s), reinterpret_cast(&byteArrayBody(t, name, 0))); return makeString(t, "%s", s); } // only safe to call during bootstrap when there's only one thread // running: void intercept(Thread* t, object c, const char* name, const char* spec, void* function) { object m = findMethodOrNull(t, c, name, spec); if (m) { PROTECT(t, m); object clone = methodClone(t, m); // make clone private to prevent vtable updates at compilation // time. Otherwise, our interception might be bypassed by calls // through the vtable. methodFlags(t, clone) |= ACC_PRIVATE; methodFlags(t, m) |= ACC_NATIVE; object native = makeNativeIntercept(t, function, true, clone); set(t, m, MethodCode, native); } } int64_t JNICALL getFileAttributes (Thread* t, object method, uintptr_t* arguments); int64_t JNICALL getLength (Thread* t, object method, uintptr_t* arguments); class MyClasspath : public Classpath { public: static const unsigned BufferSize = 1024; MyClasspath(System* s, Allocator* allocator, const char* javaHome, const char* embedPrefix): allocator(allocator) { class StringBuilder { public: StringBuilder(System* s, char* pointer, unsigned remaining): s(s), pointer(pointer), remaining(remaining) { } void append(const char* append) { unsigned length = strlen(append); expect(s, remaining > length); strncpy(pointer, append, remaining); remaining -= length; pointer += length; } void append(char c) { assert(s, remaining > 1); pointer[0] = c; pointer[1] = 0; -- remaining; ++ pointer; } System* s; char* pointer; unsigned remaining; } sb(s, buffer, BufferSize); this->javaHome = sb.pointer; sb.append(javaHome); sb.append('\0'); this->classpath = sb.pointer; sb.append(BOOT_CLASSPATH); sb.append(s->pathSeparator()); sb.append(javaHome); sb.append("/lib/rt.jar"); sb.append(s->pathSeparator()); sb.append(javaHome); sb.append("/lib/jsse.jar"); sb.append(s->pathSeparator()); sb.append(javaHome); sb.append("/lib/jce.jar"); sb.append(s->pathSeparator()); sb.append(javaHome); sb.append("/lib/resources.jar"); sb.append('\0'); this->libraryPath = sb.pointer; sb.append(javaHome); #ifdef ARCH_x86_64 sb.append("/lib/amd64"); #else // todo: handle other architectures sb.append("/lib/i386"); #endif sb.append('\0'); this->zipLibrary = sb.pointer; sb.append(this->libraryPath); sb.append("/"); sb.append(LIBRARY_PREFIX); sb.append("zip"); sb.append(LIBRARY_SUFFIX); sb.append('\0'); this->embedPrefix = sb.pointer; sb.append(embedPrefix); this->embedPrefixLength = sb.pointer - this->embedPrefix; } virtual object makeJclass(Thread* t, object class_) { PROTECT(t, class_); object name = makeClassNameString(t, getClassName(t, class_)); return vm::makeJclass (t, 0, 0, name, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, class_); } virtual object makeString(Thread* t, object array, int32_t offset, int32_t length) { if (objectClass(t, array) == type(t, Machine::ByteArrayType)) { PROTECT(t, array); object charArray = makeCharArray(t, length); for (int i = 0; i < length; ++i) { charArrayBody(t, charArray, i) = byteArrayBody(t, array, offset + i); } array = charArray; } return vm::makeString(t, array, offset, length, 0); } virtual object makeThread(Thread* t, Thread* parent) { const unsigned MaxPriority = 10; const unsigned NormalPriority = 5; object group; if (parent) { group = threadGroup(t, parent->javaThread); } else { group = makeThreadGroup (t, 0, 0, MaxPriority, false, false, false, 0, 0, 0, 0, 0); } return vm::makeThread (t, 0, NormalPriority, 0, 0, false, false, false, 0, group, root(t, Machine::BootLoader), 0, 0, 0, 0, 0, 0, 0, 0, 0, 0, false, 0, 0, 0, 0, false); } virtual void runThread(Thread* t) { object method = resolveMethod (t, root(t, Machine::BootLoader), "java/lang/Thread", "run", "()V"); if (LIKELY(t->exception == 0)) { t->m->processor->invoke(t, method, t->javaThread); } acquire(t, t->javaThread); t->flags &= ~Thread::ActiveFlag; notifyAll(t, t->javaThread); release(t, t->javaThread); } virtual object makeThrowable (Thread* t, Machine::Type type, object message, object trace, object cause) { PROTECT(t, message); PROTECT(t, trace); PROTECT(t, cause); if (trace == 0) { trace = makeTrace(t); } object result = make(t, vm::type(t, type)); set(t, result, ThrowableMessage, message); set(t, result, ThrowableTrace, trace); set(t, result, ThrowableCause, cause); return result; } virtual void boot(Thread* t) { globalMachine = t->m; #ifdef AVIAN_OPENJDK_SRC { object ufsClass = resolveClass (t, root(t, Machine::BootLoader), "java/io/UnixFileSystem"); if (ufsClass) { PROTECT(t, ufsClass); object fileClass = resolveClass (t, root(t, Machine::BootLoader), "java/io/File"); if (fileClass) { object pathField = findFieldInClass2 (t, fileClass, "path", "Ljava/lang/String;"); if (pathField) { this->pathField = fieldOffset(t, pathField); intercept(t, ufsClass, "getBooleanAttributes0", "(Ljava/io/File;)I", voidPointer(getFileAttributes)); intercept(t, ufsClass, "getLength", "(Ljava/io/File;)J", voidPointer(getLength)); } } } if (UNLIKELY(t->exception)) return; } #else // not AVIAN_OPENJDK_SRC if (loadLibrary(t, libraryPath, "java", true, true) == 0) { abort(t); } #endif // not AVIAN_OPENJDK_SRC resolveSystemClass(t, root(t, Machine::BootLoader), className(t, type(t, Machine::ClassLoaderType))); if (UNLIKELY(t->exception)) return; object constructor = resolveMethod (t, type(t, Machine::ClassLoaderType), "", "(Ljava/lang/ClassLoader;)V"); if (UNLIKELY(t->exception)) return; PROTECT(t, constructor); t->m->processor->invoke(t, constructor, root(t, Machine::BootLoader), 0); if (UNLIKELY(t->exception)) return; t->m->processor->invoke (t, constructor, root(t, Machine::AppLoader), root(t, Machine::BootLoader)); if (UNLIKELY(t->exception)) return; object scl = resolveField (t, type(t, Machine::ClassLoaderType), "scl", "Ljava/lang/ClassLoader;"); if (UNLIKELY(t->exception)) return; PROTECT(t, scl); object sclSet = resolveField (t, type(t, Machine::ClassLoaderType), "sclSet", "Z"); if (UNLIKELY(t->exception)) return; set(t, classStaticTable(t, type(t, Machine::ClassLoaderType)), fieldOffset(t, scl), root(t, Machine::AppLoader)); cast(classStaticTable(t, type(t, Machine::ClassLoaderType)), fieldOffset(t, sclSet)) = true; t->m->processor->invoke (t, root(t, Machine::BootLoader), "java/lang/System", "initializeSystemClass", "()V", 0); } virtual const char* bootClasspath() { return classpath; } virtual void dispose() { allocator->free(this, sizeof(*this)); } Allocator* allocator; const char* javaHome; const char* classpath; const char* libraryPath; const char* zipLibrary; const char* embedPrefix; unsigned embedPrefixLength; unsigned pathField; char buffer[BufferSize]; }; struct JVM_ExceptionTableEntryType { jint start_pc; jint end_pc; jint handler_pc; jint catchType; }; struct jvm_version_info { unsigned jvm_version; unsigned update_version: 8; unsigned special_update_version: 8; unsigned reserved1: 16; unsigned reserved2; unsigned is_attach_supported: 1; unsigned is_kernel_jvm: 1; unsigned: 30; unsigned: 32; unsigned: 32; }; Finder* getFinder(Thread* t, const char* name, unsigned nameLength) { ACQUIRE(t, t->m->referenceLock); for (object p = root(t, Machine::VirtualFileFinders); p; p = finderNext(t, p)) { if (byteArrayLength(t, finderName(t, p)) == nameLength and strncmp(reinterpret_cast (&byteArrayBody(t, finderName(t, p), 0)), name, nameLength)) { return static_cast(finderFinder(t, p)); } } object n = makeByteArray(t, nameLength + 1); memcpy(&byteArrayBody(t, n, 0), name, nameLength); void* p = t->m->libraries->resolve (reinterpret_cast(&byteArrayBody(t, n, 0))); if (p) { uint8_t* (*function)(unsigned*); memcpy(&function, &p, BytesPerWord); unsigned size; uint8_t* data = function(&size); if (data) { Finder* f = makeFinder(t->m->system, t->m->heap, data, size); object finder = makeFinder (t, f, n, root(t, Machine::VirtualFileFinders)); setRoot(t, Machine::VirtualFileFinders, finder); return f; } } return 0; } class EmbeddedFile { public: EmbeddedFile(MyClasspath* cp, const char* path, unsigned pathLength) { if (strncmp(cp->embedPrefix, path, cp->embedPrefixLength) == 0) { const char* p = path + cp->embedPrefixLength; while (*p == '/') ++ p; this->jar = p; if (*p == 0) { this->jarLength = 0; this->path = 0; this->pathLength = 0; return; } while (*p and *p != '/') ++p; this->jarLength = p - this->jar; while (*p == '/') ++p; this->path = p; this->pathLength = pathLength - (p - path); } else { this->jar = 0; this->jarLength =0; this->path = 0; this->pathLength = 0; } } const char* jar; const char* path; unsigned jarLength; unsigned pathLength; }; int64_t JNICALL getFileAttributes (Thread* t, object method, uintptr_t* arguments) { const unsigned Exists = 1; const unsigned Regular = 2; const unsigned Directory = 4; MyClasspath* cp = static_cast(t->m->classpath); object file = reinterpret_cast(arguments[1]); object path = cast(file, cp->pathField); RUNTIME_ARRAY(char, p, stringLength(t, path) + 1); stringChars(t, path, RUNTIME_ARRAY_BODY(p)); if (strcmp(cp->zipLibrary, RUNTIME_ARRAY_BODY(p)) == 0) { return Exists | Regular; } else { EmbeddedFile ef(cp, RUNTIME_ARRAY_BODY(p), stringLength(t, path)); if (ef.jar) { if (ef.jarLength == 0) { return Exists | Directory; } Finder* finder = getFinder(t, ef.jar, ef.jarLength); if (finder) { if (ef.pathLength == 0) { return Exists | Directory; } unsigned length; System::FileType type = finder->stat(ef.path, &length, true); switch (type) { case System::TypeUnknown: return Exists; case System::TypeDoesNotExist: return 0; case System::TypeFile: return Exists | Regular; case System::TypeDirectory: return Exists | Directory; default: abort(t); } } else { return 0; } } else { object r = t->m->processor->invoke (t, nativeInterceptOriginal(t, methodCode(t, method)), reinterpret_cast(arguments[0]), file); return (r ? intValue(t, r) : 0); } } } int64_t JNICALL getLength (Thread* t, object method, uintptr_t* arguments) { MyClasspath* cp = static_cast(t->m->classpath); object file = reinterpret_cast(arguments[1]); object path = cast(file, cp->pathField); RUNTIME_ARRAY(char, p, stringLength(t, path) + 1); stringChars(t, path, RUNTIME_ARRAY_BODY(p)); EmbeddedFile ef(cp, RUNTIME_ARRAY_BODY(p), stringLength(t, path)); if (ef.jar) { if (ef.jarLength == 0) { return 0; } Finder* finder = getFinder(t, ef.jar, ef.jarLength); if (finder) { if (ef.pathLength == 0) { return 0; } unsigned fileLength; finder->stat(ef.path, &fileLength); return fileLength; } return 0; } else { object r = t->m->processor->invoke (t, nativeInterceptOriginal(t, methodCode(t, method)), reinterpret_cast(arguments[0]), file); return (r ? longValue(t, r) : 0); } } unsigned countMethods(Thread* t, object c, bool publicOnly) { object table = classMethodTable(t, c); unsigned count = 0; for (unsigned i = 0; i < arrayLength(t, table); ++i) { object vmMethod = arrayBody(t, table, i); if (((not publicOnly) or (methodFlags(t, vmMethod) & ACC_PUBLIC)) and byteArrayBody(t, methodName(t, vmMethod), 0) != '<') { ++ count; } } return count; } unsigned countFields(Thread* t, object c, bool publicOnly) { object table = classFieldTable(t, c); if (publicOnly) { unsigned count = 0; for (unsigned i = 0; i < arrayLength(t, table); ++i) { object vmField = arrayBody(t, table, i); if (fieldFlags(t, vmField) & ACC_PUBLIC) { ++ count; } } return count; } else { return objectArrayLength(t, table); } } unsigned countConstructors(Thread* t, object c, bool publicOnly) { object table = classMethodTable(t, c); unsigned count = 0; for (unsigned i = 0; i < arrayLength(t, table); ++i) { object vmMethod = arrayBody(t, table, i); if (((not publicOnly) or (methodFlags(t, vmMethod) & ACC_PUBLIC)) and strcmp(reinterpret_cast (&byteArrayBody(t, methodName(t, vmMethod), 0)), "") == 0) { ++ count; } } return count; } object resolveClassBySpec(Thread* t, object loader, const char* spec, unsigned specLength) { switch (*spec) { case 'L': { RUNTIME_ARRAY(char, s, specLength - 1); memcpy(RUNTIME_ARRAY_BODY(s), spec + 1, specLength - 2); RUNTIME_ARRAY_BODY(s)[specLength - 2] = 0; return resolveClass(t, loader, s); } case '[': { RUNTIME_ARRAY(char, s, specLength + 1); memcpy(RUNTIME_ARRAY_BODY(s), spec, specLength); RUNTIME_ARRAY_BODY(s)[specLength] = 0; return resolveClass(t, loader, s); } default: return primitiveClass(t, *spec); } } object resolveJType(Thread* t, object loader, const char* spec, unsigned specLength) { object c = resolveClassBySpec(t, loader, spec, specLength); if (UNLIKELY(t->exception)) return 0; return getJClass(t, c); } object resolveParameterTypes(Thread* t, object loader, object spec, unsigned* parameterCount, unsigned* returnTypeSpec) { PROTECT(t, loader); PROTECT(t, spec); object list = 0; PROTECT(t, list); unsigned offset = 1; unsigned count = 0; while (byteArrayBody(t, spec, offset) != ')') { switch (byteArrayBody(t, spec, offset)) { case 'L': { unsigned start = offset; ++ offset; while (byteArrayBody(t, spec, offset) != ';') ++ offset; ++ offset; object type = resolveClassBySpec (t, loader, reinterpret_cast(&byteArrayBody(t, spec, start)), offset - start); if (UNLIKELY(t->exception)) { return 0; } list = makePair(t, type, list); ++ count; } break; case '[': { unsigned start = offset; while (byteArrayBody(t, spec, offset) == '[') ++ offset; switch (byteArrayBody(t, spec, offset)) { case 'L': ++ offset; while (byteArrayBody(t, spec, offset) != ';') ++ offset; ++ offset; break; default: ++ offset; break; } object type = resolveClassBySpec (t, loader, reinterpret_cast(&byteArrayBody(t, spec, start)), offset - start); if (UNLIKELY(t->exception)) { return 0; } list = makePair(t, type, list); ++ count; } break; default: list = makePair (t, primitiveClass(t, byteArrayBody(t, spec, offset)), list); ++ offset; ++ count; break; } } *parameterCount = count; *returnTypeSpec = offset + 1; return list; } object resolveParameterJTypes(Thread* t, object loader, object spec, unsigned* parameterCount, unsigned* returnTypeSpec) { object list = resolveParameterTypes (t, loader, spec, parameterCount, returnTypeSpec); if (UNLIKELY(t->exception)) return 0; PROTECT(t, list); object array = makeObjectArray (t, type(t, Machine::JclassType), *parameterCount); PROTECT(t, array); for (int i = *parameterCount - 1; i >= 0; --i) { object c = getJClass(t, pairFirst(t, list)); set(t, array, ArrayBody + (i * BytesPerWord), c); list = pairSecond(t, list); } return array; } object resolveExceptionJTypes(Thread* t, object loader, object addendum) { if (addendum == 0) { return makeObjectArray(t, type(t, Machine::JclassType), 0); } PROTECT(t, loader); PROTECT(t, addendum); object array = makeObjectArray (t, type(t, Machine::JclassType), shortArrayLength(t, methodAddendumExceptionTable(t, addendum))); PROTECT(t, array); for (unsigned i = 0; i < shortArrayLength (t, methodAddendumExceptionTable(t, addendum)); ++i) { uint16_t index = shortArrayBody (t, methodAddendumExceptionTable(t, addendum), i) - 1; object o = singletonObject(t, addendumPool(t, addendum), index); if (objectClass(t, o) == type(t, Machine::ReferenceType)) { o = resolveClass(t, loader, referenceName(t, o)); if (UNLIKELY(t->exception)) return 0; set(t, addendumPool(t, addendum), SingletonBody + (index * BytesPerWord), o); } o = getJClass(t, o); set(t, array, ArrayBody + (i * BytesPerWord), o); } return array; } void setProperty(Thread* t, object method, object properties, const char* name, const void* value, const char* format = "%s") { PROTECT(t, method); PROTECT(t, properties); object n = makeString(t, "%s", name); PROTECT(t, n); object v = makeString(t, format, value); t->m->processor->invoke(t, method, properties, n, v); } object interruptLock(Thread* t, object thread) { if (threadInterruptLock(t, thread) == 0) { PROTECT(t, thread); ACQUIRE(t, t->m->referenceLock); if (threadInterruptLock(t, thread) == 0) { object head = makeMonitorNode(t, 0, 0); object lock = makeMonitor(t, 0, 0, 0, head, head, 0); set(t, thread, ThreadInterruptLock, lock); } } return threadInterruptLock(t, thread); } } // namespace local } // namespace namespace vm { Classpath* makeClasspath(System* s, Allocator* allocator, const char* javaHome, const char* embedPrefix) { return new (allocator->allocate(sizeof(local::MyClasspath))) local::MyClasspath(s, allocator, javaHome, embedPrefix); } } // namespace vm extern "C" JNIEXPORT int64_t JNICALL Avian_java_lang_Class_getSuperclass (Thread* t, object, uintptr_t* arguments) { object super = classSuper (t, jclassVmClass(t, reinterpret_cast(arguments[0]))); return super ? reinterpret_cast(getJClass(t, super)) : 0; } extern "C" JNIEXPORT void Avian_sun_misc_Unsafe_registerNatives (Thread*, object, uintptr_t*) { // ignore } extern "C" JNIEXPORT int64_t Avian_sun_misc_Unsafe_defineClass__Ljava_lang_String_2_3BIILjava_lang_ClassLoader_2Ljava_security_ProtectionDomain_2 (Thread* t, object, uintptr_t* arguments) { //object name = reinterpret_cast(arguments[1]); object data = reinterpret_cast(arguments[2]); int32_t offset = arguments[3]; int32_t length = arguments[4]; object loader = reinterpret_cast(arguments[5]); //object domain = reinterpret_cast(arguments[6]); uint8_t* buffer = static_cast (t->m->heap->allocate(length)); memcpy(buffer, &byteArrayBody(t, data, offset), length); object c = defineClass(t, loader, buffer, length); return c ? reinterpret_cast(getJClass(t, c)) : 0; } extern "C" JNIEXPORT int64_t Avian_sun_misc_Unsafe_allocateInstance (Thread* t, object, uintptr_t* arguments) { object c = jclassVmClass(t, reinterpret_cast(arguments[1])); PROTECT(t, c); initClass(t, c); if (UNLIKELY(t->exception)) return 0; return reinterpret_cast(make(t, c)); } extern "C" JNIEXPORT int64_t JNICALL Avian_sun_misc_Unsafe_staticFieldOffset (Thread* t, object, uintptr_t* arguments) { object jfield = reinterpret_cast(arguments[1]); return fieldOffset (t, arrayBody (t, classFieldTable (t, jclassVmClass(t, jfieldClazz(t, jfield))), jfieldSlot(t, jfield))); } extern "C" JNIEXPORT int64_t JNICALL Avian_sun_misc_Unsafe_arrayBaseOffset (Thread*, object, uintptr_t*) { return BytesPerWord * 2; } extern "C" JNIEXPORT int64_t JNICALL Avian_sun_misc_Unsafe_arrayIndexScale (Thread* t, object, uintptr_t* arguments) { object c = jclassVmClass(t, reinterpret_cast(arguments[1])); if (classVmFlags(t, c) & PrimitiveFlag) { const char* name = reinterpret_cast (&byteArrayBody(t, local::getClassName(t, c), 0)); switch (*name) { case 'b': return 1; case 's': case 'c': return 2; case 'l': case 'd': return 8; case 'i': case 'f': return 4; default: abort(t); } } else { return BytesPerWord; } } extern "C" JNIEXPORT int64_t JNICALL Avian_sun_misc_Unsafe_staticFieldBase (Thread* t, object, uintptr_t* arguments) { return reinterpret_cast (classStaticTable (t, jclassVmClass (t, jfieldClazz(t, reinterpret_cast(arguments[1]))))); } extern "C" JNIEXPORT int64_t JNICALL Avian_sun_misc_Unsafe_objectFieldOffset (Thread* t, object, uintptr_t* arguments) { object jfield = reinterpret_cast(arguments[1]); return fieldOffset (t, arrayBody (t, classFieldTable (t, jclassVmClass(t, jfieldClazz(t, jfield))), jfieldSlot(t, jfield))); } extern "C" JNIEXPORT int64_t JNICALL Avian_sun_misc_Unsafe_getObject (Thread*, object, uintptr_t* arguments) { object o = reinterpret_cast(arguments[1]); int64_t offset; memcpy(&offset, arguments + 2, 8); return cast(o, offset); } extern "C" JNIEXPORT void JNICALL Avian_sun_misc_Unsafe_putObject (Thread* t, object, uintptr_t* arguments) { object o = reinterpret_cast(arguments[1]); int64_t offset; memcpy(&offset, arguments + 2, 8); uintptr_t value = arguments[4]; set(t, o, offset, reinterpret_cast(value)); } extern "C" JNIEXPORT int64_t JNICALL Avian_sun_misc_Unsafe_getInt__Ljava_lang_Object_2J (Thread*, object, uintptr_t* arguments) { object o = reinterpret_cast(arguments[1]); int64_t offset; memcpy(&offset, arguments + 2, 8); return cast(o, offset); } extern "C" JNIEXPORT int64_t JNICALL Avian_sun_misc_Unsafe_getIntVolatile (Thread*, object, uintptr_t* arguments) { object o = reinterpret_cast(arguments[1]); int64_t offset; memcpy(&offset, arguments + 2, 8); int32_t result = cast(o, offset); loadMemoryBarrier(); return result; } extern "C" JNIEXPORT void JNICALL Avian_sun_misc_Unsafe_putInt__Ljava_lang_Object_2JI (Thread*, object, uintptr_t* arguments) { object o = reinterpret_cast(arguments[1]); int64_t offset; memcpy(&offset, arguments + 2, 8); int32_t value = arguments[4]; cast(o, offset) = value; } extern "C" JNIEXPORT int64_t JNICALL Avian_sun_misc_Unsafe_getBoolean (Thread*, object, uintptr_t* arguments) { object o = reinterpret_cast(arguments[1]); int64_t offset; memcpy(&offset, arguments + 2, 8); return cast(o, offset); } extern "C" JNIEXPORT void JNICALL Avian_sun_misc_Unsafe_putBoolean (Thread*, object, uintptr_t* arguments) { object o = reinterpret_cast(arguments[1]); int64_t offset; memcpy(&offset, arguments + 2, 8); uint8_t value = arguments[4]; cast(o, offset) = value; } extern "C" JNIEXPORT void JNICALL Avian_sun_misc_Unsafe_putLong__Ljava_lang_Object_2JJ (Thread*, object, uintptr_t* arguments) { object o = reinterpret_cast(arguments[1]); int64_t offset; memcpy(&offset, arguments + 2, 8); int64_t value; memcpy(&value, arguments + 4, 8); cast(o, offset) = value; } extern "C" JNIEXPORT int64_t JNICALL Avian_sun_misc_Unsafe_getObjectVolatile (Thread*, object, uintptr_t* arguments) { object o = reinterpret_cast(arguments[1]); int64_t offset; memcpy(&offset, arguments + 2, 8); uintptr_t value = cast(o, offset); loadMemoryBarrier(); return value; } extern "C" JNIEXPORT int64_t JNICALL Avian_sun_misc_Unsafe_compareAndSwapInt (Thread*, object, uintptr_t* arguments) { object target = reinterpret_cast(arguments[1]); int64_t offset; memcpy(&offset, arguments + 2, 8); int32_t expect = arguments[4]; int32_t update = arguments[5]; return __sync_bool_compare_and_swap (&cast(target, offset), expect, update); } extern "C" JNIEXPORT int64_t JNICALL Avian_sun_misc_Unsafe_compareAndSwapObject (Thread* t, object, uintptr_t* arguments) { object target = reinterpret_cast(arguments[1]); int64_t offset; memcpy(&offset, arguments + 2, 8); intptr_t expect = arguments[4]; intptr_t update = arguments[5]; bool success = __sync_bool_compare_and_swap (&cast(target, offset), expect, update); if (success) { mark(t, target, offset); } return success; } extern "C" JNIEXPORT int64_t JNICALL Avian_sun_misc_Unsafe_compareAndSwapLong (Thread*, object, uintptr_t* arguments) { object target = reinterpret_cast(arguments[1]); int64_t offset; memcpy(&offset, arguments + 2, 8); int64_t expect; memcpy(&expect, arguments + 4, 8); int64_t update; memcpy(&update, arguments + 6, 8); return __sync_bool_compare_and_swap (&cast(target, offset), expect, update); } extern "C" JNIEXPORT int64_t JNICALL Avian_sun_misc_Unsafe_allocateMemory (Thread* t, object, uintptr_t* arguments) { void* p = malloc(arguments[1]); if (p) { return reinterpret_cast(p); } else { t->exception = t->m->classpath->makeThrowable (t, Machine::OutOfMemoryErrorType); return 0; } } extern "C" JNIEXPORT void JNICALL Avian_sun_misc_Unsafe_freeMemory (Thread*, object, uintptr_t* arguments) { void* p = reinterpret_cast(arguments[1]); if (p) { free(p); } } extern "C" JNIEXPORT void JNICALL Avian_sun_misc_Unsafe_setMemory (Thread*, object, uintptr_t* arguments) { int64_t p; memcpy(&p, arguments + 1, 8); int64_t count; memcpy(&count, arguments + 3, 8); int8_t v = arguments[5]; memset(reinterpret_cast(p), v, count); } extern "C" JNIEXPORT void JNICALL Avian_sun_misc_Unsafe_putLong__JJ (Thread*, object, uintptr_t* arguments) { int64_t p; memcpy(&p, arguments + 1, 8); int64_t v; memcpy(&v, arguments + 3, 8); *reinterpret_cast(p) = v; } extern "C" JNIEXPORT void JNICALL Avian_sun_misc_Unsafe_putInt__JI (Thread*, object, uintptr_t* arguments) { int64_t p; memcpy(&p, arguments + 1, 8); int32_t v = arguments[3]; *reinterpret_cast(p) = v; } extern "C" JNIEXPORT int64_t JNICALL Avian_sun_misc_Unsafe_getByte__J (Thread*, object, uintptr_t* arguments) { int64_t p; memcpy(&p, arguments + 1, 8); return *reinterpret_cast(p); } extern "C" JNIEXPORT int64_t JNICALL Avian_sun_misc_Unsafe_getInt__J (Thread*, object, uintptr_t* arguments) { int64_t p; memcpy(&p, arguments + 1, 8); return *reinterpret_cast(p); } extern "C" JNIEXPORT int64_t JNICALL Avian_sun_misc_Unsafe_pageSize (Thread*, object, uintptr_t*) { return local::PageSize; } extern "C" JNIEXPORT void JNICALL Avian_sun_misc_Unsafe_ensureClassInitialized (Thread* t, object, uintptr_t* arguments) { initClass(t, jclassVmClass(t, reinterpret_cast(arguments[1]))); } extern "C" JNIEXPORT jint JNICALL JVM_GetInterfaceVersion() { return local::InterfaceVersion; } extern "C" JNIEXPORT jint JNICALL JVM_IHashCode(Thread* t, jobject o) { ENTER(t, Thread::ActiveState); return objectHash(t, *o); } extern "C" JNIEXPORT void JNICALL JVM_MonitorWait(Thread* t, jobject o, jlong milliseconds) { ENTER(t, Thread::ActiveState); vm::wait(t, *o, milliseconds); } extern "C" JNIEXPORT void JNICALL JVM_MonitorNotify(Thread* t, jobject o) { ENTER(t, Thread::ActiveState); notify(t, *o); } extern "C" JNIEXPORT void JNICALL JVM_MonitorNotifyAll(Thread* t, jobject o) { ENTER(t, Thread::ActiveState); notifyAll(t, *o); } extern "C" JNIEXPORT jobject JNICALL JVM_Clone(Thread* t, jobject o) { ENTER(t, Thread::ActiveState); return makeLocalReference(t, clone(t, *o)); } extern "C" JNIEXPORT jstring JNICALL JVM_InternString(Thread* t, jstring s) { ENTER(t, Thread::ActiveState); return makeLocalReference(t, intern(t, *s)); } extern "C" JNIEXPORT jlong JNICALL JVM_CurrentTimeMillis(Thread* t, jclass) { return t->m->system->now(); } extern "C" JNIEXPORT jlong JNICALL JVM_NanoTime(Thread* t, jclass) { return t->m->system->now() * 1000 * 1000; } extern "C" JNIEXPORT void JNICALL JVM_ArrayCopy(Thread* t, jclass, jobject src, jint srcOffset, jobject dst, jint dstOffset, jint length) { ENTER(t, Thread::ActiveState); arrayCopy(t, *src, srcOffset, *dst, dstOffset, length); } extern "C" JNIEXPORT jobject JNICALL JVM_InitProperties(Thread* t, jobject properties) { ENTER(t, Thread::ActiveState); object method = resolveMethod (t, root(t, Machine::BootLoader), "java/util/Properties", "setProperty", "(Ljava/lang/String;Ljava/lang/String;)Ljava/lang/Object;"); if (UNLIKELY(t->exception)) { return 0; } PROTECT(t, method); #ifdef PLATFORM_WINDOWS local::setProperty(t, method, *properties, "line.separator", "\r\n"); local::setProperty(t, method, *properties, "file.separator", "\\"); local::setProperty(t, method, *properties, "path.separator", ";"); local::setProperty(t, method, *properties, "os.name", "Windows"); TCHAR buffer[MAX_PATH]; GetTempPath(MAX_PATH, buffer); local::setProperty(t, method, *properties, "java.io.tmpdir", buffer); local::setProperty(t, method, *properties, "java.home", buffer); local::setProperty(t, method, *properties, "user.home", _wgetenv(L"USERPROFILE"), "%ls"); GetCurrentDirectory(MAX_PATH, buffer); local::setProperty(t, method, *properties, "user.dir", buffer); #else local::setProperty(t, method, *properties, "line.separator", "\n"); local::setProperty(t, method, *properties, "file.separator", "/"); local::setProperty(t, method, *properties, "path.separator", ":"); # ifdef __APPLE__ local::setProperty(t, method, *properties, "os.name", "Mac OS X"); # else local::setProperty(t, method, *properties, "os.name", "Linux"); # endif local::setProperty(t, method, *properties, "java.io.tmpdir", "/tmp"); local::setProperty(t, method, *properties, "user.home", getenv("HOME")); local::setProperty(t, method, *properties, "user.dir", getenv("PWD")); local::setProperty(t, method, *properties, "java.protocol.handler.pkgs", "avian"); #endif local::setProperty (t, method, *properties, "java.home", static_cast(t->m->classpath)->javaHome); local::setProperty (t, method, *properties, "sun.boot.library.path", static_cast(t->m->classpath)->libraryPath); local::setProperty(t, method, *properties, "file.encoding", "ASCII"); #ifdef ARCH_x86_32 local::setProperty(t, method, *properties, "os.arch", "x86"); #elif defined ARCH_x86_64 local::setProperty(t, method, *properties, "os.arch", "x86_64"); #elif defined ARCH_powerpc local::setProperty(t, method, *properties, "os.arch", "ppc"); #elif defined ARCH_arm local::setProperty(t, method, *properties, "os.arch", "arm"); #else local::setProperty(t, method, *properties, "os.arch", "unknown"); #endif for (unsigned i = 0; i < t->m->propertyCount; ++i) { const char* start = t->m->properties[i]; const char* p = start; while (*p and *p != '=') ++p; if (*p == '=') { RUNTIME_ARRAY(char, name, (p - start) + 1); memcpy(name, start, p - start); name[p - start] = 0; local::setProperty (t, method, *properties, RUNTIME_ARRAY_BODY(name), p + 1); } } return properties; } extern "C" JNIEXPORT void JNICALL JVM_OnExit(void (*)(void)) { abort(); } extern "C" JNIEXPORT void JNICALL JVM_Exit(jint code) { exit(code); } extern "C" JNIEXPORT void JNICALL JVM_Halt(jint code) { exit(code); } extern "C" JNIEXPORT void JNICALL JVM_GC() { Thread* t = static_cast(local::globalMachine->localThread->get()); ENTER(t, Thread::ActiveState); collect(t, Heap::MajorCollection); } extern "C" JNIEXPORT jlong JNICALL JVM_MaxObjectInspectionAge(void) { abort(); } extern "C" JNIEXPORT void JNICALL JVM_TraceInstructions(jboolean) { abort(); } extern "C" JNIEXPORT void JNICALL JVM_TraceMethodCalls(jboolean) { abort(); } extern "C" JNIEXPORT jlong JNICALL JVM_TotalMemory() { return 0; } extern "C" JNIEXPORT jlong JNICALL JVM_FreeMemory() { return 0; } extern "C" JNIEXPORT jlong JNICALL JVM_MaxMemory() { return 0; } extern "C" JNIEXPORT jint JNICALL JVM_ActiveProcessorCount(void) { abort(); } extern "C" JNIEXPORT void* JNICALL JVM_LoadLibrary(const char* name) { Thread* t = static_cast(local::globalMachine->localThread->get()); #ifdef AVIAN_OPENJDK_SRC if (strcmp(static_cast(t->m->classpath)->zipLibrary, name) == 0) { return t->m->libraries; } #endif // AVIAN_OPENJDK_SRC ENTER(t, Thread::ActiveState); return loadLibrary (t, static_cast(t->m->classpath)->libraryPath, name, false, false); } extern "C" JNIEXPORT void JNICALL JVM_UnloadLibrary(void*) { abort(); } extern "C" JNIEXPORT void* JNICALL JVM_FindLibraryEntry(void* library, const char* name) { Thread* t = static_cast(local::globalMachine->localThread->get()); ENTER(t, Thread::ActiveState); return static_cast(library)->resolve(name); } extern "C" JNIEXPORT jboolean JNICALL JVM_IsSupportedJNIVersion(jint version) { return version <= JNI_VERSION_1_4; } extern "C" JNIEXPORT jboolean JNICALL JVM_IsNaN(jdouble) { abort(); } extern "C" JNIEXPORT void JNICALL JVM_FillInStackTrace(Thread* t, jobject throwable) { ENTER(t, Thread::ActiveState); object trace = getTrace(t, 1); set(t, *throwable, ThrowableTrace, trace); } extern "C" JNIEXPORT void JNICALL JVM_PrintStackTrace(Thread*, jobject, jobject) { abort(); } extern "C" JNIEXPORT jint JNICALL JVM_GetStackTraceDepth(Thread* t, jobject throwable) { ENTER(t, Thread::ActiveState); return objectArrayLength(t, throwableTrace(t, *throwable)); } extern "C" JNIEXPORT jobject JNICALL JVM_GetStackTraceElement(Thread* t, jobject throwable, jint index) { ENTER(t, Thread::ActiveState); return makeLocalReference (t, makeStackTraceElement (t, objectArrayBody(t, throwableTrace(t, *throwable), index))); } extern "C" JNIEXPORT void JNICALL JVM_InitializeCompiler (Thread*, jclass) { abort(); } extern "C" JNIEXPORT jboolean JNICALL JVM_IsSilentCompiler(Thread*, jclass) { abort(); } extern "C" JNIEXPORT jboolean JNICALL JVM_CompileClass(Thread*, jclass, jclass) { abort(); } extern "C" JNIEXPORT jboolean JNICALL JVM_CompileClasses(Thread*, jclass, jstring) { abort(); } extern "C" JNIEXPORT jobject JNICALL JVM_CompilerCommand(Thread*, jclass, jobject) { abort(); } extern "C" JNIEXPORT void JNICALL JVM_EnableCompiler(Thread*, jclass) { abort(); } extern "C" JNIEXPORT void JNICALL JVM_DisableCompiler(Thread*, jclass) { abort(); } extern "C" JNIEXPORT void JNICALL JVM_StartThread(Thread* t, jobject thread) { ENTER(t, Thread::ActiveState); startThread(t, *thread); } extern "C" JNIEXPORT void JNICALL JVM_StopThread(Thread*, jobject, jobject) { abort(); } extern "C" JNIEXPORT jboolean JNICALL JVM_IsThreadAlive(Thread* t, jobject thread) { ENTER(t, Thread::ActiveState); Thread* p = reinterpret_cast(threadPeer(t, *thread)); return p and (p->flags & Thread::ActiveFlag) != 0; } extern "C" JNIEXPORT void JNICALL JVM_SuspendThread(Thread*, jobject) { abort(); } extern "C" JNIEXPORT void JNICALL JVM_ResumeThread(Thread*, jobject) { abort(); } extern "C" JNIEXPORT void JNICALL JVM_SetThreadPriority(Thread*, jobject, jint) { // ignore } extern "C" JNIEXPORT void JNICALL JVM_Yield(Thread*, jclass) { sched_yield(); } extern "C" JNIEXPORT void JNICALL JVM_Sleep(Thread* t, jclass, jlong milliseconds) { ENTER(t, Thread::ActiveState); if (threadSleepLock(t, t->javaThread) == 0) { object lock = makeJobject(t); set(t, t->javaThread, ThreadSleepLock, lock); } acquire(t, threadSleepLock(t, t->javaThread)); vm::wait(t, threadSleepLock(t, t->javaThread), milliseconds); release(t, threadSleepLock(t, t->javaThread)); } extern "C" JNIEXPORT jobject JNICALL JVM_CurrentThread(Thread* t, jclass) { ENTER(t, Thread::ActiveState); return makeLocalReference(t, t->javaThread); } extern "C" JNIEXPORT jint JNICALL JVM_CountStackFrames(Thread*, jobject) { abort(); } extern "C" JNIEXPORT void JNICALL JVM_Interrupt(Thread* t, jobject thread) { ENTER(t, Thread::ActiveState); monitorAcquire(t, local::interruptLock(t, *thread)); Thread* p = reinterpret_cast(threadPeer(t, *thread)); if (p) { interrupt(t, p); } else { threadInterrupted(t, *thread) = true; } monitorRelease(t, local::interruptLock(t, *thread)); } extern "C" JNIEXPORT jboolean JNICALL JVM_IsInterrupted(Thread* t, jobject thread, jboolean clear) { ENTER(t, Thread::ActiveState); monitorAcquire(t, local::interruptLock(t, *thread)); bool v = threadInterrupted(t, *thread); if (clear) { threadInterrupted(t, *thread) = false; } monitorRelease(t, local::interruptLock(t, *thread)); return v; } extern "C" JNIEXPORT jboolean JNICALL JVM_HoldsLock(Thread*, jclass, jobject) { abort(); } extern "C" JNIEXPORT void JNICALL JVM_DumpAllStacks(Thread*, jclass) { abort(); } extern "C" JNIEXPORT jobjectArray JNICALL JVM_GetAllThreads(Thread*, jclass) { abort(); } extern "C" JNIEXPORT jobjectArray JNICALL JVM_DumpThreads(Thread*, jclass, jobjectArray) { abort(); } extern "C" JNIEXPORT jclass JNICALL JVM_CurrentLoadedClass(Thread*) { abort(); } extern "C" JNIEXPORT jobject JNICALL JVM_CurrentClassLoader(Thread*) { abort(); } extern "C" JNIEXPORT jobjectArray JNICALL JVM_GetClassContext(Thread* t) { ENTER(t, Thread::ActiveState); object trace = getTrace(t, 1); PROTECT(t, trace); object context = makeObjectArray (t, type(t, Machine::JclassType), objectArrayLength(t, trace)); PROTECT(t, context); for (unsigned i = 0; i < objectArrayLength(t, trace); ++i) { object c = getJClass (t, methodClass(t, traceElementMethod(t, objectArrayBody(t, trace, i)))); set(t, context, ArrayBody + (i * BytesPerWord), c); } return makeLocalReference(t, context); } extern "C" JNIEXPORT jint JNICALL JVM_ClassDepth(Thread*, jstring) { abort(); } extern "C" JNIEXPORT jint JNICALL JVM_ClassLoaderDepth(Thread*) { abort(); } extern "C" JNIEXPORT jstring JNICALL JVM_GetSystemPackage(Thread*, jstring) { return 0; } extern "C" JNIEXPORT jobjectArray JNICALL JVM_GetSystemPackages(Thread*) { abort(); } extern "C" JNIEXPORT jobject JNICALL JVM_AllocateNewObject(Thread*, jobject, jclass, jclass) { abort(); } extern "C" JNIEXPORT jobject JNICALL JVM_AllocateNewArray(Thread*, jobject, jclass, jint) { abort(); } extern "C" JNIEXPORT jobject JNICALL JVM_LatestUserDefinedLoader(Thread*) { abort(); } extern "C" JNIEXPORT jclass JNICALL JVM_LoadClass0(Thread*, jobject, jclass, jstring) { abort(); } extern "C" JNIEXPORT jint JNICALL JVM_GetArrayLength(Thread* t, jobject array) { ENTER(t, Thread::ActiveState); return cast(*array, BytesPerWord); } extern "C" JNIEXPORT jobject JNICALL JVM_GetArrayElement(Thread* t, jobject array, jint index) { ENTER(t, Thread::ActiveState); return makeLocalReference(t, objectArrayBody(t, *array, index)); } extern "C" JNIEXPORT jvalue JNICALL JVM_GetPrimitiveArrayElement(Thread*, jobject, jint, jint) { abort(); } extern "C" JNIEXPORT void JNICALL JVM_SetArrayElement(Thread* t, jobject array, jint index, jobject value) { ENTER(t, Thread::ActiveState); set(t, *array, ArrayBody + (index * BytesPerWord), (value ? *value : 0)); } extern "C" JNIEXPORT void JNICALL JVM_SetPrimitiveArrayElement(Thread*, jobject, jint, jvalue, unsigned char) { abort(); } extern "C" JNIEXPORT jobject JNICALL JVM_NewArray(Thread* t, jclass elementClass, jint length) { ENTER(t, Thread::ActiveState); object c = jclassVmClass(t, *elementClass); if (classVmFlags(t, c) & PrimitiveFlag) { const char* name = reinterpret_cast (&byteArrayBody(t, local::getClassName(t, c), 0)); switch (*name) { case 'b': if (name[1] == 'o') { return makeLocalReference(t, makeBooleanArray(t, length)); } else { return makeLocalReference(t, makeByteArray(t, length)); } case 'c': return makeLocalReference(t, makeCharArray(t, length)); case 'd': return makeLocalReference(t, makeDoubleArray(t, length)); case 'f': return makeLocalReference(t, makeFloatArray(t, length)); case 'i': return makeLocalReference(t, makeIntArray(t, length)); case 'l': return makeLocalReference(t, makeLongArray(t, length)); case 's': return makeLocalReference(t, makeShortArray(t, length)); default: abort(t); } } else { return makeLocalReference(t, makeObjectArray(t, c, length)); } } extern "C" JNIEXPORT jobject JNICALL JVM_NewMultiArray(Thread*, jclass, jintArray) { abort(); } extern "C" JNIEXPORT jclass JNICALL JVM_GetCallerClass(Thread* t, int target) { ENTER(t, Thread::ActiveState); return makeLocalReference (t, getJClass(t, methodClass(t, getCaller(t, target)))); } extern "C" JNIEXPORT jclass JNICALL JVM_FindPrimitiveClass(Thread* t, const char* name) { ENTER(t, Thread::ActiveState); switch (*name) { case 'b': if (name[1] == 'o') { return makeLocalReference (t, getJClass(t, type(t, Machine::JbooleanType))); } else { return makeLocalReference (t, getJClass(t, type(t, Machine::JbyteType))); } case 'c': return makeLocalReference (t, getJClass(t, type(t, Machine::JcharType))); case 'd': return makeLocalReference (t, getJClass(t, type(t, Machine::JdoubleType))); case 'f': return makeLocalReference (t, getJClass(t, type(t, Machine::JfloatType))); case 'i': return makeLocalReference (t, getJClass(t, type(t, Machine::JintType))); case 'l': return makeLocalReference (t, getJClass(t, type(t, Machine::JlongType))); case 's': return makeLocalReference (t, getJClass(t, type(t, Machine::JshortType))); case 'v': return makeLocalReference (t, getJClass(t, type(t, Machine::JvoidType))); default: t->exception = t->m->classpath->makeThrowable (t, Machine::IllegalArgumentExceptionType); return 0; } } extern "C" JNIEXPORT void JNICALL JVM_ResolveClass(Thread*, jclass) { abort(); } extern "C" JNIEXPORT jclass JNICALL JVM_FindClassFromClassLoader(Thread* t, const char* name, jboolean init, jobject loader, jboolean throwError) { ENTER(t, Thread::ActiveState); object c = resolveClass (t, loader ? *loader : root(t, Machine::BootLoader), name); if (t->exception) { if (throwError) { t->exception = t->m->classpath->makeThrowable (t, Machine::NoClassDefFoundErrorType, throwableMessage(t, t->exception), throwableTrace(t, t->exception), throwableCause(t, t->exception)); } return 0; } if (init) { PROTECT(t, c); initClass(t, c); } return makeLocalReference(t, getJClass(t, c)); } extern "C" JNIEXPORT jclass JNICALL JVM_FindClassFromClass(Thread*, const char*, jboolean, jclass) { abort(); } extern "C" JNIEXPORT jclass JNICALL JVM_FindLoadedClass(Thread* t, jobject loader, jstring name) { ENTER(t, Thread::ActiveState); object spec = makeByteArray(t, stringLength(t, *name) + 1); { char* s = reinterpret_cast(&byteArrayBody(t, spec, 0)); stringChars(t, *name, s); replace('.', '/', s); } object c = findLoadedClass(t, *loader, spec); return c ? makeLocalReference(t, getJClass(t, c)) : 0; } extern "C" JNIEXPORT jclass JNICALL JVM_DefineClass(Thread* t, const char*, jobject loader, const uint8_t* data, jsize length, jobject) { ENTER(t, Thread::ActiveState); object c = defineClass(t, *loader, data, length); return c ? makeLocalReference(t, getJClass(t, c)) : 0; } extern "C" JNIEXPORT jclass JNICALL JVM_DefineClassWithSource(Thread* t, const char*, jobject loader, const uint8_t* data, jsize length, jobject, const char*) { return JVM_DefineClass(t, 0, loader, data, length, 0); } extern "C" JNIEXPORT jstring JNICALL JVM_GetClassName(Thread* t, jclass c) { ENTER(t, Thread::ActiveState); return makeLocalReference(t, jclassName(t, *c)); } extern "C" JNIEXPORT jobjectArray JNICALL JVM_GetClassInterfaces(Thread* t, jclass c) { ENTER(t, Thread::ActiveState); object table = classInterfaceTable(t, jclassVmClass(t, *c)); if (table) { unsigned stride = (classFlags(t, jclassVmClass(t, *c)) & ACC_INTERFACE) == 0 ? 2 : 1; object array = makeObjectArray (t, type(t, Machine::JclassType), arrayLength(t, table) / stride); PROTECT(t, array); for (unsigned i = 0; i < objectArrayLength(t, array); ++i) { object interface = getJClass(t, arrayBody(t, table, i * stride)); set(t, array, ArrayBody + (i * BytesPerWord), interface); } return makeLocalReference(t, array); } else { return makeLocalReference (t, makeObjectArray(t, type(t, Machine::JclassType), 0)); } } extern "C" JNIEXPORT jobject JNICALL JVM_GetClassLoader(Thread* t, jclass c) { ENTER(t, Thread::ActiveState); object loader = classLoader(t, jclassVmClass(t, *c)); if (loader == root(t, Machine::BootLoader)) { // sun.misc.Unsafe.getUnsafe expects a null result if the class // loader is the boot classloader and will throw a // SecurityException otherwise. object caller = getCaller(t, 2); if (caller and strcmp (reinterpret_cast (&byteArrayBody(t, className(t, methodClass(t, caller)), 0)), "sun/misc/Unsafe") == 0) { return 0; } else { return makeLocalReference(t, root(t, Machine::BootLoader)); } } else { return makeLocalReference(t, loader); } } extern "C" JNIEXPORT jboolean JNICALL JVM_IsInterface(Thread* t, jclass c) { ENTER(t, Thread::ActiveState); return (classFlags(t, jclassVmClass(t, *c)) & ACC_INTERFACE) != 0; } extern "C" JNIEXPORT jobjectArray JNICALL JVM_GetClassSigners(Thread*, jclass) { abort(); } extern "C" JNIEXPORT void JNICALL JVM_SetClassSigners(Thread*, jclass, jobjectArray) { abort(); } extern "C" JNIEXPORT jobject JNICALL JVM_GetProtectionDomain(Thread* t, jclass) { ENTER(t, Thread::ActiveState); object openJDK = resolveClass (t, root(t, Machine::BootLoader), "avian/OpenJDK"); if (UNLIKELY(t->exception)) return 0; object method = resolveMethod (t, openJDK, "getProtectionDomain", "()Ljava/security/ProtectionDomain;"); if (UNLIKELY(t->exception)) return 0; return makeLocalReference(t, t->m->processor->invoke(t, method, 0)); } extern "C" JNIEXPORT void JNICALL JVM_SetProtectionDomain(Thread*, jclass, jobject) { abort(); } extern "C" JNIEXPORT jboolean JNICALL JVM_IsArrayClass(Thread* t, jclass c) { ENTER(t, Thread::ActiveState); return classArrayDimensions(t, jclassVmClass(t, *c)) != 0; } extern "C" JNIEXPORT jboolean JNICALL JVM_IsPrimitiveClass(Thread* t, jclass c) { ENTER(t, Thread::ActiveState); return (classVmFlags(t, jclassVmClass(t, *c)) & PrimitiveFlag) != 0; } extern "C" JNIEXPORT jclass JNICALL JVM_GetComponentType(Thread* t, jclass c) { ENTER(t, Thread::ActiveState); uint8_t n = byteArrayBody(t, className(t, jclassVmClass(t, *c)), 1); if (n != 'L' and n != '[') { return makeLocalReference(t, getJClass(t, primitiveClass(t, n))); } else { return makeLocalReference (t, getJClass(t, classStaticTable(t, jclassVmClass(t, *c)))); } } extern "C" JNIEXPORT jint JNICALL JVM_GetClassModifiers(Thread* t, jclass c) { ENTER(t, Thread::ActiveState); return classFlags(t, jclassVmClass(t, *c)); } extern "C" JNIEXPORT jobjectArray JNICALL JVM_GetDeclaredClasses(Thread*, jclass) { abort(); } extern "C" JNIEXPORT jclass JNICALL JVM_GetDeclaringClass(Thread*, jclass) { abort(); } extern "C" JNIEXPORT jstring JNICALL JVM_GetClassSignature(Thread*, jclass) { abort(); } extern "C" JNIEXPORT jbyteArray JNICALL JVM_GetClassAnnotations(Thread* t, jclass c) { ENTER(t, Thread::ActiveState); return makeLocalReference (t, addendumAnnotationTable(t, classAddendum(t, jclassVmClass(t, *c)))); } extern "C" JNIEXPORT jobjectArray JNICALL JVM_GetClassDeclaredMethods(Thread* t, jclass c, jboolean publicOnly) { ENTER(t, Thread::ActiveState); object table = classMethodTable(t, jclassVmClass(t, *c)); if (table) { PROTECT(t, table); object array = makeObjectArray (t, type(t, Machine::JmethodType), local::countMethods(t, jclassVmClass(t, *c), publicOnly)); PROTECT(t, array); unsigned ai = 0; for (unsigned i = 0; i < arrayLength(t, table); ++i) { object vmMethod = arrayBody(t, table, i); PROTECT(t, vmMethod); if (((not publicOnly) or (methodFlags(t, vmMethod) & ACC_PUBLIC)) and byteArrayBody(t, methodName(t, vmMethod), 0) != '<') { object name = intern (t, t->m->classpath->makeString (t, methodName(t, vmMethod), 0, byteArrayLength (t, methodName(t, vmMethod)) - 1)); PROTECT(t, name); unsigned parameterCount; unsigned returnTypeSpec; object parameterTypes = local::resolveParameterJTypes (t, classLoader(t, jclassVmClass(t, *c)), methodSpec(t, vmMethod), ¶meterCount, &returnTypeSpec); if (UNLIKELY(t->exception)) return 0; PROTECT(t, parameterTypes); object returnType = local::resolveJType (t, classLoader(t, jclassVmClass(t, *c)), reinterpret_cast (&byteArrayBody(t, methodSpec(t, vmMethod), returnTypeSpec)), byteArrayLength(t, methodSpec(t, vmMethod)) - 1 - returnTypeSpec); if (UNLIKELY(t->exception)) return 0; PROTECT(t, returnType); object exceptionTypes = local::resolveExceptionJTypes (t, classLoader(t, jclassVmClass(t, *c)), methodAddendum(t, vmMethod)); if (UNLIKELY(t->exception)) return 0; PROTECT(t, exceptionTypes); object signature = t->m->classpath->makeString (t, methodSpec(t, vmMethod), 0, byteArrayLength (t, methodSpec(t, vmMethod)) - 1); object annotationTable = methodAddendum(t, vmMethod) == 0 ? 0 : addendumAnnotationTable(t, methodAddendum(t, vmMethod)); if (annotationTable) { set(t, classAddendum(t, jclassVmClass(t, *c)), AddendumPool, addendumPool(t, methodAddendum(t, vmMethod))); } object method = makeJmethod (t, true, *c, i, name, returnType, parameterTypes, exceptionTypes, methodFlags(t, vmMethod), signature, 0, annotationTable, 0, 0, 0, 0, 0, 0, 0); assert(t, ai < objectArrayLength(t, array)); set(t, array, ArrayBody + ((ai++) * BytesPerWord), method); } } return makeLocalReference(t, array); } else { return makeLocalReference (t, makeObjectArray(t, type(t, Machine::JmethodType), 0)); } } extern "C" JNIEXPORT jobjectArray JNICALL JVM_GetClassDeclaredFields(Thread* t, jclass c, jboolean publicOnly) { ENTER(t, Thread::ActiveState); object table = classFieldTable(t, jclassVmClass(t, *c)); if (table) { PROTECT(t, table); object array = makeObjectArray (t, type(t, Machine::JfieldType), local::countFields(t, jclassVmClass(t, *c), publicOnly)); PROTECT(t, array); unsigned ai = 0; for (unsigned i = 0; i < arrayLength(t, table); ++i) { object vmField = arrayBody(t, table, i); PROTECT(t, vmField); if ((not publicOnly) or (fieldFlags(t, vmField) & ACC_PUBLIC)) { object name = intern (t, t->m->classpath->makeString (t, fieldName(t, vmField), 0, byteArrayLength (t, fieldName(t, vmField)) - 1)); PROTECT(t, name); object type = local::resolveClassBySpec (t, classLoader(t, jclassVmClass(t, *c)), reinterpret_cast (&byteArrayBody(t, fieldSpec(t, vmField), 0)), byteArrayLength(t, fieldSpec(t, vmField)) - 1); if (UNLIKELY(t->exception)) { return 0; } PROTECT(t, type); type = getJClass(t, type); object signature = t->m->classpath->makeString (t, fieldSpec(t, vmField), 0, byteArrayLength (t, fieldSpec(t, vmField)) - 1); object annotationTable = fieldAddendum(t, vmField) == 0 ? 0 : addendumAnnotationTable(t, fieldAddendum(t, vmField)); if (annotationTable) { set(t, classAddendum(t, jclassVmClass(t, *c)), AddendumPool, addendumPool(t, fieldAddendum(t, vmField))); } object field = makeJfield (t, true, *c, i, name, type, fieldFlags (t, vmField), signature, 0, annotationTable, 0, 0, 0, 0, 0, 0); assert(t, ai < objectArrayLength(t, array)); set(t, array, ArrayBody + ((ai++) * BytesPerWord), field); } } assert(t, ai == objectArrayLength(t, array)); return makeLocalReference(t, array); } else { return makeLocalReference (t, makeObjectArray(t, type(t, Machine::JfieldType), 0)); } } extern "C" JNIEXPORT jobjectArray JNICALL JVM_GetClassDeclaredConstructors(Thread* t, jclass c, jboolean publicOnly) { ENTER(t, Thread::ActiveState); object table = classMethodTable(t, jclassVmClass(t, *c)); if (table) { PROTECT(t, table); object array = makeObjectArray (t, type(t, Machine::JconstructorType), local::countConstructors(t, jclassVmClass(t, *c), publicOnly)); PROTECT(t, array); unsigned ai = 0; for (unsigned i = 0; i < arrayLength(t, table); ++i) { object vmMethod = arrayBody(t, table, i); PROTECT(t, vmMethod); if (((not publicOnly) or (methodFlags(t, vmMethod) & ACC_PUBLIC)) and strcmp(reinterpret_cast (&byteArrayBody(t, methodName(t, vmMethod), 0)), "") == 0) { unsigned parameterCount; unsigned returnTypeSpec; object parameterTypes = local::resolveParameterJTypes (t, classLoader(t, jclassVmClass(t, *c)), methodSpec(t, vmMethod), ¶meterCount, &returnTypeSpec); if (UNLIKELY(t->exception)) return 0; PROTECT(t, parameterTypes); object exceptionTypes = local::resolveExceptionJTypes (t, classLoader(t, jclassVmClass(t, *c)), methodAddendum(t, vmMethod)); if (UNLIKELY(t->exception)) return 0; PROTECT(t, exceptionTypes); object signature = t->m->classpath->makeString (t, methodSpec(t, vmMethod), 0, byteArrayLength (t, methodSpec(t, vmMethod)) - 1); object annotationTable = methodAddendum(t, vmMethod) == 0 ? 0 : addendumAnnotationTable(t, methodAddendum(t, vmMethod)); if (annotationTable) { set(t, classAddendum(t, jclassVmClass(t, *c)), AddendumPool, addendumPool(t, methodAddendum(t, vmMethod))); } object method = makeJconstructor (t, true, *c, i, parameterTypes, exceptionTypes, methodFlags (t, vmMethod), signature, 0, annotationTable, 0, 0, 0, 0, 0); assert(t, ai < objectArrayLength(t, array)); set(t, array, ArrayBody + ((ai++) * BytesPerWord), method); } } return makeLocalReference(t, array); } else { return makeLocalReference (t, makeObjectArray(t, type(t, Machine::JconstructorType), 0)); } } extern "C" JNIEXPORT jint JNICALL JVM_GetClassAccessFlags(Thread* t, jclass c) { return JVM_GetClassModifiers(t, c); } extern "C" JNIEXPORT jobject JNICALL JVM_InvokeMethod(Thread* t, jobject method, jobject instance, jobjectArray arguments) { ENTER(t, Thread::ActiveState); object vmMethod = arrayBody (t, classMethodTable (t, jclassVmClass(t, jmethodClazz(t, *method))), jmethodSlot(t, *method)); if (methodFlags(t, vmMethod) & ACC_STATIC) { instance = 0; } object result; if (arguments) { result = t->m->processor->invokeArray (t, vmMethod, instance ? *instance : 0, *arguments); } else { result = t->m->processor->invoke(t, vmMethod, instance ? *instance : 0); } return result ? makeLocalReference(t, result) : 0; } extern "C" JNIEXPORT jobject JNICALL JVM_NewInstanceFromConstructor(Thread* t, jobject constructor, jobjectArray arguments) { ENTER(t, Thread::ActiveState); object instance = make (t, jclassVmClass(t, jconstructorClazz(t, *constructor))); PROTECT(t, instance); object method = arrayBody (t, classMethodTable (t, jclassVmClass(t, jconstructorClazz(t, *constructor))), jconstructorSlot(t, *constructor)); if (arguments) { t->m->processor->invokeArray(t, method, instance, *arguments); } else { t->m->processor->invoke(t, method, instance); } if (UNLIKELY(t->exception)) { return 0; } else { return makeLocalReference(t, instance); } } extern "C" JNIEXPORT jobject JNICALL JVM_GetClassConstantPool(Thread* t, jclass c) { ENTER(t, Thread::ActiveState); return makeLocalReference (t, makeConstantPool (t, addendumPool(t, classAddendum(t, jclassVmClass(t, *c))))); } extern "C" JNIEXPORT jint JNICALL JVM_ConstantPoolGetSize(Thread* t, jobject, jobject pool) { if (pool == 0) return 0; ENTER(t, Thread::ActiveState); return singletonCount(t, *pool); } extern "C" JNIEXPORT jclass JNICALL JVM_ConstantPoolGetClassAt(Thread*, jobject, jobject, jint) { abort(); } extern "C" JNIEXPORT jclass JNICALL JVM_ConstantPoolGetClassAtIfLoaded(Thread*, jobject, jobject, jint) { abort(); } extern "C" JNIEXPORT jobject JNICALL JVM_ConstantPoolGetMethodAt(Thread*, jobject, jobject, jint) { abort(); } extern "C" JNIEXPORT jobject JNICALL JVM_ConstantPoolGetMethodAtIfLoaded(Thread*, jobject, jobject, jint) { abort(); } extern "C" JNIEXPORT jobject JNICALL JVM_ConstantPoolGetFieldAt(Thread*, jobject, jobject, jint) { abort(); } extern "C" JNIEXPORT jobject JNICALL JVM_ConstantPoolGetFieldAtIfLoaded(Thread*, jobject, jobject, jint) { abort(); } extern "C" JNIEXPORT jobjectArray JNICALL JVM_ConstantPoolGetMemberRefInfoAt(Thread*, jobject, jobject, jint) { abort(); } extern "C" JNIEXPORT jint JNICALL JVM_ConstantPoolGetIntAt(Thread* t, jobject, jobject pool, jint index) { ENTER(t, Thread::ActiveState); return singletonValue(t, *pool, index - 1); } extern "C" JNIEXPORT jlong JNICALL JVM_ConstantPoolGetLongAt(Thread*, jobject, jobject, jint) { abort(); } extern "C" JNIEXPORT jfloat JNICALL JVM_ConstantPoolGetFloatAt(Thread*, jobject, jobject, jint) { abort(); } extern "C" JNIEXPORT jdouble JNICALL JVM_ConstantPoolGetDoubleAt(Thread*, jobject, jobject, jint) { abort(); } extern "C" JNIEXPORT jstring JNICALL JVM_ConstantPoolGetStringAt(Thread*, jobject, jobject, jint) { abort(); } extern "C" JNIEXPORT jstring JNICALL JVM_ConstantPoolGetUTF8At(Thread* t, jobject, jobject pool, jint index) { ENTER(t, Thread::ActiveState); object array = singletonObject(t, *pool, index - 1); return makeLocalReference (t, t->m->classpath->makeString (t, array, 0, cast(array, BytesPerWord) - 1)); } extern "C" JNIEXPORT jobject JNICALL JVM_DoPrivileged (Thread* t, jclass, jobject action, jobject, jboolean wrapException) { ENTER(t, Thread::ActiveState); // todo: cache these class and method lookups in the t->m->classpath // object: object privilegedAction = resolveClass (t, root(t, Machine::BootLoader), "java/security/PrivilegedAction"); if (UNLIKELY(t->exception)) { return 0; } object method; if (instanceOf(t, privilegedAction, *action)) { method = resolveMethod (t, privilegedAction, "run", "()Ljava/lang/Object;"); } else { object privilegedExceptionAction = resolveClass (t, root(t, Machine::BootLoader), "java/security/PrivilegedExceptionAction"); if (UNLIKELY(t->exception)) { return 0; } method = resolveMethod (t, privilegedExceptionAction, "run", "()Ljava/lang/Object;"); } if (LIKELY(t->exception == 0)) { object result = t->m->processor->invoke(t, method, *action); if (LIKELY(t->exception == 0)) { return makeLocalReference(t, result); } else { if (wrapException and not (instanceOf(t, type(t, Machine::ErrorType), t->exception) or instanceOf(t, type(t, Machine::RuntimeExceptionType), t->exception))) { object cause = t->exception; PROTECT(t, cause); t->exception = 0; object paeClass = resolveClass (t, root(t, Machine::BootLoader), "java/security/PrivilegedActionException"); if (LIKELY(t->exception == 0)) { PROTECT(t, paeClass); object paeConstructor = resolveMethod (t, paeClass, "", "(Ljava/lang/Exception;)V"); PROTECT(t, paeConstructor); if (LIKELY(t->exception == 0)) { object result = make(t, paeClass); PROTECT(t, result); t->m->processor->invoke(t, paeConstructor, result, cause); if (LIKELY(t->exception == 0)) { t->exception = result; } } } } } } return 0; } extern "C" JNIEXPORT jobject JNICALL JVM_GetInheritedAccessControlContext(Thread*, jclass) { abort(); } extern "C" JNIEXPORT jobject JNICALL JVM_GetStackAccessControlContext(Thread*, jclass) { return 0; } extern "C" JNIEXPORT void* JNICALL JVM_RegisterSignal(jint, void*) { abort(); } extern "C" JNIEXPORT jboolean JNICALL JVM_RaiseSignal(jint) { abort(); } extern "C" JNIEXPORT jint JNICALL JVM_FindSignal(const char*) { return -1; } extern "C" JNIEXPORT jboolean JNICALL JVM_DesiredAssertionStatus(Thread*, jclass, jclass) { return false; } extern "C" JNIEXPORT jobject JNICALL JVM_AssertionStatusDirectives(Thread*, jclass) { abort(); } extern "C" JNIEXPORT jboolean JNICALL JVM_SupportsCX8() { return true; } extern "C" JNIEXPORT const char* JNICALL JVM_GetClassNameUTF(Thread*, jclass) { abort(); } extern "C" JNIEXPORT void JNICALL JVM_GetClassCPTypes(Thread*, jclass, unsigned char*) { abort(); } extern "C" JNIEXPORT jint JNICALL JVM_GetClassCPEntriesCount(Thread*, jclass) { abort(); } extern "C" JNIEXPORT jint JNICALL JVM_GetClassFieldsCount(Thread*, jclass) { abort(); } extern "C" JNIEXPORT jint JNICALL JVM_GetClassMethodsCount(Thread*, jclass) { abort(); } extern "C" JNIEXPORT void JNICALL JVM_GetMethodIxExceptionIndexes(Thread*, jclass, jint, unsigned short*) { abort(); } extern "C" JNIEXPORT jint JNICALL JVM_GetMethodIxExceptionsCount(Thread*, jclass, jint) { abort(); } extern "C" JNIEXPORT void JNICALL JVM_GetMethodIxByteCode(Thread*, jclass, jint, unsigned char*) { abort(); } extern "C" JNIEXPORT jint JNICALL JVM_GetMethodIxByteCodeLength(Thread*, jclass, jint) { abort(); } extern "C" JNIEXPORT void JNICALL JVM_GetMethodIxExceptionTableEntry(Thread*, jclass, jint, jint, local::JVM_ExceptionTableEntryType*) { abort(); } extern "C" JNIEXPORT jint JNICALL JVM_GetMethodIxExceptionTableLength(Thread*, jclass, int) { abort(); } extern "C" JNIEXPORT jint JNICALL JVM_GetFieldIxModifiers(Thread*, jclass, int) { abort(); } extern "C" JNIEXPORT jint JNICALL JVM_GetMethodIxModifiers(Thread*, jclass, int) { abort(); } extern "C" JNIEXPORT jint JNICALL JVM_GetMethodIxLocalsCount(Thread*, jclass, int) { abort(); } extern "C" JNIEXPORT jint JNICALL JVM_GetMethodIxArgsSize(Thread*, jclass, int) { abort(); } extern "C" JNIEXPORT jint JNICALL JVM_GetMethodIxMaxStack(Thread*, jclass, int) { abort(); } extern "C" JNIEXPORT jboolean JNICALL JVM_IsConstructorIx(Thread*, jclass, int) { abort(); } extern "C" JNIEXPORT const char* JNICALL JVM_GetMethodIxNameUTF(Thread*, jclass, jint) { abort(); } extern "C" JNIEXPORT const char* JNICALL JVM_GetMethodIxSignatureUTF(Thread*, jclass, jint) { abort(); } extern "C" JNIEXPORT const char* JNICALL JVM_GetCPFieldNameUTF(Thread*, jclass, jint) { abort(); } extern "C" JNIEXPORT const char* JNICALL JVM_GetCPMethodNameUTF(Thread*, jclass, jint) { abort(); } extern "C" JNIEXPORT const char* JNICALL JVM_GetCPMethodSignatureUTF(Thread*, jclass, jint) { abort(); } extern "C" JNIEXPORT const char* JNICALL JVM_GetCPFieldSignatureUTF(Thread*, jclass, jint) { abort(); } extern "C" JNIEXPORT const char* JNICALL JVM_GetCPClassNameUTF(Thread*, jclass, jint) { abort(); } extern "C" JNIEXPORT const char* JNICALL JVM_GetCPFieldClassNameUTF(Thread*, jclass, jint) { abort(); } extern "C" JNIEXPORT const char* JNICALL JVM_GetCPMethodClassNameUTF(Thread*, jclass, jint) { abort(); } extern "C" JNIEXPORT jint JNICALL JVM_GetCPFieldModifiers(Thread*, jclass, int, jclass) { abort(); } extern "C" JNIEXPORT jint JNICALL JVM_GetCPMethodModifiers(Thread*, jclass, int, jclass) { abort(); } extern "C" JNIEXPORT void JNICALL JVM_ReleaseUTF(const char*) { abort(); } extern "C" JNIEXPORT jboolean JNICALL JVM_IsSameClassPackage(Thread*, jclass, jclass) { abort(); } extern "C" JNIEXPORT jint JNICALL JVM_GetLastErrorString(char* dst, int length) { strncpy(dst, strerror(errno), length); return strlen(dst); } extern "C" JNIEXPORT char* JNICALL JVM_NativePath(char* path) { return path; } extern "C" JNIEXPORT jint JNICALL JVM_Open(const char* path, jint flags, jint mode) { Thread* t = static_cast(local::globalMachine->localThread->get()); local::MyClasspath* cp = static_cast(t->m->classpath); local::EmbeddedFile ef(cp, path, strlen(path)); if (ef.jar) { if (flags != O_RDONLY) { errno = EACCES; return -1; } if (ef.jarLength == 0 or ef.pathLength == 0) { errno = ENOENT; return -1; } Finder* finder = local::getFinder(t, ef.jar, ef.jarLength); if (finder == 0) { errno = ENOENT; return -1; } System::Region* r = finder->find(ef.path); if (r == 0) { errno = ENOENT; return -1; } ENTER(t, Thread::ActiveState); ACQUIRE(t, t->m->referenceLock); int index = -1; unsigned oldLength = root(t, Machine::VirtualFiles) ? arrayLength(t, root(t, Machine::VirtualFiles)) : 0; for (unsigned i = 0; i < oldLength; ++i) { if (arrayBody(t, root(t, Machine::VirtualFiles), i) == 0) { index = i; break; } } if (index == -1) { object newArray = growArray(t, root(t, Machine::VirtualFiles)); setRoot(t, Machine::VirtualFiles, newArray); index = oldLength; } object region = makeRegion(t, r, 0); set(t, root(t, Machine::VirtualFiles), ArrayBody + (index * BytesPerWord), region); return index + local::VirtualFileBase; } else { int r = OPEN(path, flags, mode); expect(t, r < local::VirtualFileBase); return r; } } extern "C" JNIEXPORT jint JNICALL JVM_Close(jint fd) { if (fd >= local::VirtualFileBase) { Thread* t = static_cast(local::globalMachine->localThread->get()); unsigned index = fd - local::VirtualFileBase; ENTER(t, Thread::ActiveState); ACQUIRE(t, t->m->referenceLock); object region = arrayBody(t, root(t, Machine::VirtualFiles), index); if (region) { static_cast(regionRegion(t, region))->dispose(); } set(t, root(t, Machine::VirtualFiles), ArrayBody + (index * BytesPerWord), 0); return 0; } else { return CLOSE(fd); } } extern "C" JNIEXPORT jint JNICALL JVM_Read(jint fd, char* dst, jint length) { if (fd >= local::VirtualFileBase) { Thread* t = static_cast(local::globalMachine->localThread->get()); unsigned index = fd - local::VirtualFileBase; ENTER(t, Thread::ActiveState); ACQUIRE(t, t->m->referenceLock); object region = arrayBody(t, root(t, Machine::VirtualFiles), index); if (region) { System::Region* r = static_cast (regionRegion(t, region)); int available = r->length() - regionPosition(t, region); if (length > available) { length = available; } memcpy(dst, r->start(), length); regionPosition(t, region) += length; return length; } else { errno = EINVAL; return -1; } } else { return READ(fd, dst, length); } } extern "C" JNIEXPORT jint JNICALL JVM_Write(jint fd, char* src, jint length) { return WRITE(fd, src, length); } extern "C" JNIEXPORT jint JNICALL JVM_Available(jint fd, jlong* result) { if (fd >= local::VirtualFileBase) { Thread* t = static_cast(local::globalMachine->localThread->get()); unsigned index = fd - local::VirtualFileBase; ENTER(t, Thread::ActiveState); ACQUIRE(t, t->m->referenceLock); object region = arrayBody(t, root(t, Machine::VirtualFiles), index); if (region) { return static_cast(regionRegion(t, region))->length() - regionPosition(t, region); } else { return 0; } } else { struct stat buffer; int n; if (FSTAT(fd, &buffer) >= 0 and (S_ISCHR(buffer.st_mode) or S_ISFIFO(buffer.st_mode) or S_ISSOCK(buffer.st_mode)) and ioctl(fd, FIONREAD, &n) >= 0) { *result = n; return 1; } int current = LSEEK(fd, 0, SEEK_CUR); if (current == -1) return 0; int end = LSEEK(fd, 0, SEEK_END); if (end == -1) return 0; if (LSEEK(fd, current, SEEK_SET) == -1) return 0; *result = end - current; return 1; } } extern "C" JNIEXPORT jlong JNICALL JVM_Lseek(jint fd, jlong offset, jint seek) { if (fd >= local::VirtualFileBase) { Thread* t = static_cast(local::globalMachine->localThread->get()); unsigned index = fd - local::VirtualFileBase; ENTER(t, Thread::ActiveState); ACQUIRE(t, t->m->referenceLock); object region = arrayBody(t, root(t, Machine::VirtualFiles), index); if (region) { System::Region* r = static_cast (regionRegion(t, region)); switch (seek) { case SEEK_SET: break; case SEEK_CUR: offset += regionPosition(t, region); break; case SEEK_END: offset += r->length(); break; default: errno = EINVAL; return -1; } if (offset >= 0 and offset <= static_cast(r->length())) { regionPosition(t, region) = offset; return offset; } else { errno = EINVAL; return -1; } } else { errno = EINVAL; return -1; } } else { return LSEEK(fd, offset, seek); } } extern "C" JNIEXPORT jint JNICALL JVM_SetLength(jint, jlong) { abort(); } extern "C" JNIEXPORT jint JNICALL JVM_Sync(jint fd) { return fsync(fd); } extern "C" JNIEXPORT jint JNICALL JVM_InitializeSocketLibrary() { #ifdef PLATFORM_WINDOWS static bool wsaInitialized = false; if (not wsaInitialized) { WSADATA data; int r = WSAStartup(MAKEWORD(2, 2), &data); if (r or LOBYTE(data.wVersion) != 2 or HIBYTE(data.wVersion) != 2) { return -1; } else { wsaInitialized = true; } } #endif return 0; } extern "C" JNIEXPORT jint JNICALL JVM_Socket(jint domain, jint type, jint protocol) { return socket(domain, type, protocol); } extern "C" JNIEXPORT jint JNICALL JVM_SocketClose(jint socket) { #ifdef PLATFORM_WINDOWS return closesocket(socket); #else return close(socket); #endif } extern "C" JNIEXPORT jint JNICALL JVM_SocketShutdown(jint socket, jint how) { return shutdown(socket, how); } extern "C" JNIEXPORT jint JNICALL JVM_Recv(jint socket, char* dst, jint count, jint flags) { return recv(socket, dst, count, flags); } extern "C" JNIEXPORT jint JNICALL JVM_Send(jint socket, char* src, jint count, jint flags) { return send(socket, src, count, flags); } extern "C" JNIEXPORT jint JNICALL JVM_Timeout(int, long) { abort(); } extern "C" JNIEXPORT jint JNICALL JVM_Listen(jint socket, jint count) { return listen(socket, count); } extern "C" JNIEXPORT jint JNICALL JVM_Connect(jint socket, sockaddr* address, jint addressLength) { return connect(socket, address, addressLength); } extern "C" JNIEXPORT jint JNICALL JVM_Bind(jint, struct sockaddr*, jint) { abort(); } extern "C" JNIEXPORT jint JNICALL JVM_Accept(jint socket, struct sockaddr* address, jint* addressLength) { socklen_t length = *addressLength; int r = accept(socket, address, &length); *addressLength = length; return r; } extern "C" JNIEXPORT jint JNICALL JVM_RecvFrom(jint, char*, int, int, struct sockaddr*, int*) { abort(); } extern "C" JNIEXPORT jint JNICALL JVM_SendTo(jint, char*, int, int, struct sockaddr*, int) { abort(); } extern "C" JNIEXPORT jint JNICALL JVM_SocketAvailable(jint, jint*) { abort(); } extern "C" JNIEXPORT jint JNICALL JVM_GetSockName(jint socket, struct sockaddr* address, int* addressLength) { socklen_t length = *addressLength; int r = getsockname(socket, address, &length); *addressLength = length; return r; } extern "C" JNIEXPORT jint JNICALL JVM_GetSockOpt(jint, int, int, char*, int*) { abort(); } extern "C" JNIEXPORT jint JNICALL JVM_SetSockOpt(jint, int, int, const char*, int) { abort(); } extern "C" JNIEXPORT struct protoent* JNICALL JVM_GetProtoByName(char*) { abort(); } extern "C" JNIEXPORT struct hostent* JNICALL JVM_GetHostByAddr(const char*, int, int) { abort(); } extern "C" JNIEXPORT struct hostent* JNICALL JVM_GetHostByName(char*) { abort(); } extern "C" JNIEXPORT int JNICALL JVM_GetHostName(char* name, int length) { return gethostname(name, length); } extern "C" JNIEXPORT void* JNICALL JVM_RawMonitorCreate(void) { System* s = local::globalMachine->system; System::Monitor* lock; if (s->success(s->make(&lock))) { return lock; } else { return 0; } } extern "C" JNIEXPORT void JNICALL JVM_RawMonitorDestroy(void* lock) { static_cast(lock)->dispose(); } extern "C" JNIEXPORT jint JNICALL JVM_RawMonitorEnter(void* lock) { static_cast(lock)->acquire (static_cast (local::globalMachine->localThread->get())->systemThread); return 0; } extern "C" JNIEXPORT void JNICALL JVM_RawMonitorExit(void* lock) { static_cast(lock)->release (static_cast (local::globalMachine->localThread->get())->systemThread); } extern "C" JNIEXPORT void* JNICALL JVM_GetManagement(jint) { abort(); } extern "C" JNIEXPORT jobject JNICALL JVM_InitAgentProperties(Thread*, jobject) { abort(); } extern "C" JNIEXPORT jobjectArray JNICALL JVM_GetEnclosingMethodInfo(JNIEnv*, jclass) { abort(); } extern "C" JNIEXPORT jintArray JNICALL JVM_GetThreadStateValues(JNIEnv*, jint) { abort(); } extern "C" JNIEXPORT jobjectArray JNICALL JVM_GetThreadStateNames(JNIEnv*, jint, jintArray) { abort(); } extern "C" JNIEXPORT void JNICALL JVM_GetVersionInfo(JNIEnv*, local::jvm_version_info*, size_t) { abort(); } extern "C" JNIEXPORT int jio_vsnprintf(char* dst, size_t size, const char* format, va_list a) { return vm::vsnprintf(dst, size, format, a); } // extern "C" JNIEXPORT int // jio_snprintf(char* dst, size_t size, const char* format, ...) // { // va_list a; // va_start(a, format); // int r = jio_vsnprintf(dst, size, format, a); // va_end(a); // return r; // } extern "C" JNIEXPORT int jio_vfprintf(FILE* stream, const char* format, va_list a) { return vfprintf(stream, format, a); } // extern "C" JNIEXPORT int // jio_fprintf(FILE* stream, const char* format, ...) // { // va_list a; // va_start(a, format); // int r = jio_vfprintf(stream, format, a); // va_end(a); // return r; // }