/* src/vm/initialize.cpp - static class initializer functions Copyright (C) 1996-2005, 2006, 2007, 2008 CACAOVM - Verein zur Foerderung der freien virtuellen Maschine CACAO This file is part of CACAO. This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ #include "config.h" #include #include "vm/types.h" #include "threads/lock.hpp" #include "vm/jit/builtin.hpp" #include "vm/class.h" #include "vm/exceptions.hpp" #include "vm/global.h" #include "vm/globals.hpp" #include "vm/initialize.hpp" #include "vm/loader.hpp" #include "vm/options.h" #include "vm/vm.hpp" #if defined(ENABLE_STATISTICS) # include "vm/statistics.h" #endif #include "vm/jit/asmpart.h" /* private functions **********************************************************/ static bool initialize_class_intern(classinfo *c); /* initialize_init ************************************************************* Initialize important system classes. *******************************************************************************/ void initialize_init(void) { TRACESUBSYSTEMINITIALIZATION("initialize_init"); #if defined(ENABLE_JAVASE) # if defined(WITH_JAVA_RUNTIME_LIBRARY_GNU_CLASSPATH) /* Nothing. */ # elif defined(WITH_JAVA_RUNTIME_LIBRARY_OPENJDK) if (!initialize_class(class_java_lang_String)) vm_abort("initialize_init: Initialization failed: java.lang.String"); if (!initialize_class(class_java_lang_System)) vm_abort("initialize_init: Initialization failed: java.lang.System"); if (!initialize_class(class_java_lang_ThreadGroup)) vm_abort("initialize_init: Initialization failed: java.lang.ThreadGroup"); if (!initialize_class(class_java_lang_Thread)) vm_abort("initialize_init: Initialization failed: java.lang.Thread"); # else # error unknown classpath configuration # endif #elif defined(ENABLE_JAVAME_CLDC1_1) /* Nothing. */ #else # error unknown Java configuration #endif } /* initialize_class ************************************************************ In Java, every class can have a static initialization function. This function has to be called BEFORE calling other methods or accessing static variables. *******************************************************************************/ bool initialize_class(classinfo *c) { bool r; if (!makeinitializations) return true; LOCK_MONITOR_ENTER(c); /* maybe the class is already initalized or the current thread, which can pass the monitor, is currently initalizing this class */ if (CLASS_IS_OR_ALMOST_INITIALIZED(c)) { LOCK_MONITOR_EXIT(c); return true; } /* if throw an Error before, the class was marked with an error and we have to throw a NoClassDefFoundError */ if (c->state & CLASS_ERROR) { exceptions_throw_noclassdeffounderror(c->name); LOCK_MONITOR_EXIT(c); /* ...but return true, this is ok (mauve test) */ return true; } /* this initalizing run begins NOW */ c->state |= CLASS_INITIALIZING; /* call the internal function */ r = initialize_class_intern(c); /* if return value is not NULL everything was ok and the class is initialized */ if (r) c->state |= CLASS_INITIALIZED; /* this initalizing run is done */ c->state &= ~CLASS_INITIALIZING; LOCK_MONITOR_EXIT(c); return r; } /* initialize_class_intern ***************************************************** This function MUST NOT be called directly, because of thread race conditions. *******************************************************************************/ static bool initialize_class_intern(classinfo *c) { methodinfo *m; java_handle_t *cause; classinfo *clazz; /* maybe the class is not already linked */ if (!(c->state & CLASS_LINKED)) if (!link_class(c)) return false; #if defined(ENABLE_STATISTICS) if (opt_stat) count_class_inits++; #endif /* Initialize super class. */ if (c->super != NULL) { if (!(c->super->state & CLASS_INITIALIZED)) { #if !defined(NDEBUG) if (initverbose) log_message_class_message_class("Initialize super class ", c->super, " from ", c); #endif if (!initialize_class(c->super)) return false; } } /* interfaces implemented need not to be initialized (VM Spec 2.17.4) */ m = class_findmethod(c, utf_clinit, utf_void__void); if (m == NULL) { #if !defined(NDEBUG) if (initverbose) log_message_class("Class has no static class initializer: ", c); #endif return true; } /* Sun's and IBM's JVM don't care about the static flag */ /* if (!(m->flags & ACC_STATIC)) { */ /* log_text("Class initializer is not static!"); */ #if !defined(NDEBUG) if (initverbose) log_message_class("Starting static class initializer for class: ", c); #endif /* now call the initializer */ (void) vm_call_method(m, NULL); /* we have an exception or error */ cause = exceptions_get_exception(); if (cause != NULL) { /* class is NOT initialized and is marked with error */ c->state |= CLASS_ERROR; /* Load java/lang/Exception for the instanceof check. */ clazz = load_class_bootstrap(utf_java_lang_Exception); if (clazz == NULL) return false; /* Is this an exception? Yes, than wrap it. */ if (builtin_instanceof(cause, clazz)) { /* clear exception, because we are calling jit code again */ exceptions_clear_exception(); /* wrap the exception */ exceptions_throw_exceptionininitializererror(cause); } return false; } #if !defined(NDEBUG) if (initverbose) log_message_class("Finished static class initializer for class: ", c); #endif return true; } /* * These are local overrides for various environment variables in Emacs. * Please do not remove this and leave it at the end of the file, where * Emacs will automagically detect them. * --------------------------------------------------------------------- * Local variables: * mode: c * indent-tabs-mode: t * c-basic-offset: 4 * tab-width: 4 * End: */