📄 natclassloader.cc
字号:
// natClassLoader.cc - Implementation of java.lang.ClassLoader native methods./* Copyright (C) 1999, 2000, 2001, 2002 Free Software Foundation This file is part of libgcj.This software is copyrighted work licensed under the terms of theLibgcj License. Please consult the file "LIBGCJ_LICENSE" fordetails. *//* Author: Kresten Krab Thorup <krab@gnu.org> */#include <config.h>#include <stdlib.h>#include <string.h>#include <gcj/cni.h>#include <jvm.h>#include <java-threads.h>#include <java-interp.h>#include <java/lang/Character.h>#include <java/lang/Thread.h>#include <java/lang/ClassLoader.h>#include <gnu/gcj/runtime/VMClassLoader.h>#include <java/lang/InternalError.h>#include <java/lang/IllegalAccessError.h>#include <java/lang/LinkageError.h>#include <java/lang/ClassFormatError.h>#include <java/lang/NoClassDefFoundError.h>#include <java/lang/ClassNotFoundException.h>#include <java/lang/ClassCircularityError.h>#include <java/lang/IncompatibleClassChangeError.h>#include <java/lang/VirtualMachineError.h>#include <java/lang/VMClassLoader.h>#include <java/lang/reflect/Modifier.h>#include <java/lang/Runtime.h>#include <java/lang/StringBuffer.h>#include <java/io/Serializable.h>#include <java/lang/Cloneable.h>/////////// java.lang.ClassLoader native methods ////////////java::lang::Class *java::lang::ClassLoader::defineClass0 (jstring name, jbyteArray data, jint offset, jint length, java::security::ProtectionDomain *pd){#ifdef INTERPRETER jclass klass; klass = (jclass) JvAllocObject (&java::lang::Class::class$, sizeof (_Jv_InterpClass)); _Jv_InitNewClassFields (klass); // Synchronize on the class, so that it is not attempted initialized // until we're done loading. JvSynchronize sync (klass); // Record the defining loader. For the system class loader, we // record NULL. if (this != java::lang::ClassLoader::getSystemClassLoader()) klass->loader = this; if (name != 0) { _Jv_Utf8Const *name2 = _Jv_makeUtf8Const (name); if (! _Jv_VerifyClassName (name2)) throw new java::lang::ClassFormatError (JvNewStringLatin1 ("erroneous class name")); klass->name = name2; } try { _Jv_DefineClass (klass, data, offset, length); } catch (java::lang::Throwable *ex) { klass->state = JV_STATE_ERROR; klass->notifyAll (); _Jv_UnregisterClass (klass); // If EX is not a ClassNotFoundException, that's ok, because we // account for the possibility in defineClass(). throw ex; } klass->protectionDomain = pd; // if everything proceeded sucessfully, we're loaded. JvAssert (klass->state == JV_STATE_LOADED); return klass;#else // INTERPRETER return 0;#endif}void_Jv_WaitForState (jclass klass, int state){ if (klass->state >= state) return; _Jv_MonitorEnter (klass) ; if (state == JV_STATE_LINKED) { // Must call _Jv_PrepareCompiledClass while holding the class // mutex. _Jv_PrepareCompiledClass (klass); _Jv_MonitorExit (klass); return; } java::lang::Thread *self = java::lang::Thread::currentThread(); // this is similar to the strategy for class initialization. // if we already hold the lock, just leave. while (klass->state <= state && klass->thread && klass->thread != self) klass->wait (); _Jv_MonitorExit (klass); if (klass->state == JV_STATE_ERROR) throw new java::lang::LinkageError;}// Finish linking a class. Only called from ClassLoader::resolveClass.voidjava::lang::ClassLoader::linkClass0 (java::lang::Class *klass){ if (klass->state >= JV_STATE_LINKED) return;#ifdef INTERPRETER if (_Jv_IsInterpretedClass (klass)) _Jv_PrepareClass (klass);#endif _Jv_PrepareCompiledClass (klass);}voidjava::lang::ClassLoader::markClassErrorState0 (java::lang::Class *klass){ klass->state = JV_STATE_ERROR; klass->notifyAll ();}jclassjava::lang::VMClassLoader::defineClass (java::lang::ClassLoader *cl, jstring name, jbyteArray data, jint offset, jint length){ return cl->defineClass (name, data, offset, length);}jclassjava::lang::VMClassLoader::getPrimitiveClass (jchar type){ char sig[2]; sig[0] = (char) type; sig[1] = '\0'; return _Jv_FindClassFromSignature (sig, NULL);}/** This function does class-preparation for compiled classes. NOTE: It contains replicated functionality from _Jv_ResolvePoolEntry, and this is intentional, since that function lives in resolve.cc which is entirely conditionally compiled. */void_Jv_PrepareCompiledClass (jclass klass){ if (klass->state >= JV_STATE_LINKED) return; // Short-circuit, so that mutually dependent classes are ok. klass->state = JV_STATE_LINKED; _Jv_Constants *pool = &klass->constants; for (int index = 1; index < pool->size; ++index) { if (pool->tags[index] == JV_CONSTANT_Class) { _Jv_Utf8Const *name = pool->data[index].utf8; jclass found; if (name->data[0] == '[') found = _Jv_FindClassFromSignature (&name->data[0], klass->loader); else found = _Jv_FindClass (name, klass->loader); if (! found) { jstring str = _Jv_NewStringUTF (name->data); throw new java::lang::NoClassDefFoundError (str); } pool->data[index].clazz = found; pool->tags[index] |= JV_CONSTANT_ResolvedFlag; } else if (pool->tags[index] == JV_CONSTANT_String) { jstring str; str = _Jv_NewStringUtf8Const (pool->data[index].utf8); pool->data[index].o = str; pool->tags[index] |= JV_CONSTANT_ResolvedFlag; } }#ifdef INTERPRETER // FIXME: although the comment up top says that this function is // only called for compiled classes, it is actually called for every // class. if (! _Jv_IsInterpretedClass (klass)) {#endif /* INTERPRETER */ jfieldID f = JvGetFirstStaticField (klass); for (int n = JvNumStaticFields (klass); n > 0; --n) { int mod = f->getModifiers (); // If we have a static String field with a non-null initial // value, we know it points to a Utf8Const. if (f->getClass () == &java::lang::String::class$ && java::lang::reflect::Modifier::isStatic (mod)) { jstring *strp = (jstring *) f->u.addr; if (*strp) *strp = _Jv_NewStringUtf8Const ((_Jv_Utf8Const *) *strp); } f = f->getNextField (); }#ifdef INTERPRETER }#endif /* INTERPRETER */ if (klass->vtable == NULL) _Jv_MakeVTable(klass); if (klass->otable != NULL && klass->otable->state == 0) _Jv_LinkOffsetTable(klass); klass->notifyAll (); _Jv_PushClass (klass);}//// A single class can have many "initiating" class loaders,// and a single "defining" class loader. The Defining// class loader is what is returned from Class.getClassLoader()// and is used when loading dependent classes during resolution.// The set of initiating class loaders are used to ensure// safety of linking, and is maintained in the hash table// "initiated_classes". A defining classloader is by definition also// initiating, so we only store classes in this table, if they have more// than one class loader associated.//// Size of local hash table.#define HASH_LEN 1013// Hash function for Utf8Consts.#define HASH_UTF(Utf) (((Utf)->hash) % HASH_LEN)struct _Jv_LoaderInfo{ _Jv_LoaderInfo *next; java::lang::Class *klass; java::lang::ClassLoader *loader;};static _Jv_LoaderInfo *initiated_classes[HASH_LEN];static jclass loaded_classes[HASH_LEN];// This is the root of a linked list of classesjclass_Jv_FindClassInCache (_Jv_Utf8Const *name, java::lang::ClassLoader *loader){ JvSynchronize sync (&java::lang::Class::class$); jint hash = HASH_UTF (name); if (loader && loader == java::lang::ClassLoader::getSystemClassLoader()) loader = NULL; // first, if LOADER is a defining loader, then it is also initiating jclass klass; for (klass = loaded_classes[hash]; klass; klass = klass->next) { if (loader == klass->loader && _Jv_equalUtf8Consts (name, klass->name)) break; } // otherwise, it may be that the class in question was defined // by some other loader, but that the loading was initiated by // the loader in question. if (!klass) { _Jv_LoaderInfo *info; for (info = initiated_classes[hash]; info; info = info->next) { if (loader == info->loader && _Jv_equalUtf8Consts (name, info->klass->name)) { klass = info->klass; break; } } } return klass;}void_Jv_UnregisterClass (jclass the_class){ JvSynchronize sync (&java::lang::Class::class$); jint hash = HASH_UTF(the_class->name); jclass *klass = &(loaded_classes[hash]); for ( ; *klass; klass = &((*klass)->next)) { if (*klass == the_class) { *klass = (*klass)->next; break; } } _Jv_LoaderInfo **info = &(initiated_classes[hash]); for ( ; ; info = &((*info)->next)) {
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -