natclassloader.cc

来自「gcc3.2.1源代码」· CC 代码 · 共 718 行 · 第 1/2 页

CC
718
字号
// 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>// FIXME: remove these.#define CloneableClass java::lang::Cloneable::class$#define ObjectClass java::lang::Object::class$#define ClassClass java::lang::Class::class$#define VMClassLoaderClass gnu::gcj::runtime::VMClassLoader::class$#define ClassLoaderClass java::lang::ClassLoader::class$#define SerializableClass java::io::Serializable::class$/////////// 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 (&ClassClass, sizeof (_Jv_InterpClass));  _Jv_InitNewClassFields (klass);  // synchronize on the class, so that it is not  // attempted initialized until we're done loading.  _Jv_MonitorEnter (klass);  // record which is the defining loader  klass->loader = this;  // register that we are the initiating loader...  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);      _Jv_MonitorExit (klass);      // FIXME: Here we may want to test that EX does      // indeed represent a valid exception.  That is,      // anything but ClassNotFoundException,       // or some kind of Error.      // FIXME: Rewrite this as a cleanup instead of      // as a catch handler.      throw ex;    }      klass->protectionDomain = pd;  // if everything proceeded sucessfully, we're loaded.  JvAssert (klass->state == JV_STATE_LOADED);  // if an exception is generated, this is initially missed.  // however, we come back here in handleException0 below...  _Jv_MonitorExit (klass);  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::getPrimitiveClass (jchar type){  char sig[2];  sig[0] = (char) type;  sig[1] = '\0';  return _Jv_FindClassFromSignature (sig, NULL);}// This is the findClass() implementation for the System classloader. It is // the only native method in VMClassLoader, so we define it here.jclassgnu::gcj::runtime::VMClassLoader::findClass (jstring name){  _Jv_Utf8Const *name_u = _Jv_makeUtf8Const (name);  jclass klass = _Jv_FindClassInCache (name_u, 0);  if (! klass)    {      // Turn `gnu.pkg.quux' into `lib-gnu-pkg-quux'.  Then search for      // a module named (eg, on Linux) `lib-gnu-pkg-quux.so', followed      // by `lib-gnu-pkg.so' and `lib-gnu.so'.  If loading one of      // these causes the class to appear in the cache, then use it.      java::lang::StringBuffer *sb = new java::lang::StringBuffer (JvNewStringLatin1("lib-"));      jstring so_base_name = (sb->append (name)->toString ())->replace ('.', '-');      // Compare against `3' because that is the length of "lib".      while (! klass && so_base_name && so_base_name->length() > 3)	{	  using namespace ::java::lang;	  Runtime *rt = Runtime::getRuntime();	  jboolean loaded = rt->loadLibraryInternal (so_base_name);	  jint nd = so_base_name->lastIndexOf ('-');	  if (nd == -1)	    so_base_name = NULL;	  else	    so_base_name = so_base_name->substring (0, nd);	  if (loaded)	    klass = _Jv_FindClassInCache (name_u, 0);	}    }  // Now try loading using the interpreter.  if (! klass)    {      klass = java::net::URLClassLoader::findClass (name);    }  return klass;}jclassjava::lang::ClassLoader::findLoadedClass (jstring name){  return _Jv_FindClassInCache (_Jv_makeUtf8Const (name), this);}/** 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::ClassNotFoundException (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 () == &StringClass	      && 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 ();}////  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 classesjclass_Jv_FindClassInCache (_Jv_Utf8Const *name, java::lang::ClassLoader *loader){  _Jv_MonitorEnter (&ClassClass);  jint hash = HASH_UTF (name);  // first, if LOADER is a defining loader, then it is also initiating

⌨️ 快捷键说明

复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?