diff options
Diffstat (limited to 'libjava')
42 files changed, 8335 insertions, 592 deletions
diff --git a/libjava/ChangeLog b/libjava/ChangeLog index 19547951ae8..24051fcdc48 100644 --- a/libjava/ChangeLog +++ b/libjava/ChangeLog @@ -1,3 +1,132 @@ +1999-08-09 Anthony Green <green@cygnus.com> + + * gij.cc: New file. + + * include/config.h.in: Rebuilt. + * acconfig.h: Add INTERPRETER. + + * Makefile.in: Rebuilt. + * Makefile.am (libffi_files): Identify the libffi object files for + inclusion in libgcj. + (LIBFFIINCS): Define. + + * interpret.cc (gnu::gcj::runtime::MethodInvocation::continue1): + Dummy definition for configurations without an interpreter. + + * java/net/natPlainSocketImpl.cc (getOption): Disamiguate call to + java::lang::Boolean constructor. + + * include/java-interp.h: Always include java-cpool.h. + + * java/lang/natClassLoader.cc (getVMClassLoader0): Always return 0 + when INTERPRETER not defined. + + * java/lang/Class.h (finalize): Define. + + * gnu/gcj/util/path/DirectoryPathEntry.java (getURL): Catch + IOException from File.getCanonicalPath. + (getStream): Likewise. + + * NEWS: More news. + * THANKS: More thanks. + +1999-08-09 Kresten Krab Thorup <krab@gnu.org> + + * resolve.cc (get_ffi_type_from_signature): Generate uint16 for + jchar type. + (_Jv_PrepareClass): Allow non-abstract classes to + have abstract subclasses. + (_Jv_ResolvePoolEntry): Revert subclass check for protected + fields and methods. + * interpret.cc (continue1/perform_invoke): Don't sign extend + uint16 return val. + (continue1/lshl,lshr): Push long, not int. + (continue1/ulshr): Use UINT64, not long long. + * defineclass.cc (handleFieldsEnd): Handle case when all fields + are static. + * java/lang/natClass.cc (forName): Add call to _Jv_InitClass. + * java/lang/FirstThread.java (run): Add top-level exception + handler. + (run0): Renamed from run. + +1999-08-08 Kresten Krab Thorup <krab@gnu.org> + + * configure.in (--with-interpreter): Added. + * include/config.h.in (INTERPRETER): Added. + + * java/lang/ClassLoader.java: File replaced. + * java/lang/VMClassLoader.java: New file. + * java/lang/natClassLoader.cc: New file. + * gnu/gcj/runtime/MethodInvocation.java: New file. + * gnu/gcj/util/path/SearchPath.java: New file. + * gnu/gcj/util/path/PathEntry.java: New file. + * gnu/gcj/util/path/DirectoryPathEntry.java: New file. + * gnu/gcj/util/path/ZipPathEntry.java: New file. + * gnu/gcj/util/path/URLPathEntry.java: New file. + * gnu/gcj/util/path/CacheEntry.java: New file. + * include/java-interp.h: New file. + * include/java-cpool.h: New file. + * include/java-insns.h: New file. + * defineclass.cc: New file. + * interpret.cc: New file. + * resolve.cc: New file. + + * java/lang/natClass.cc (loaded_classes, _Jv_RegisterClass, + _Jv_RegisterClasses, _Jv_FindClassInCache, _Jv_FindClass, + _Jv_NewClass, _Jv_FindArrayClass): Moved to natClassLoader.cc. + (finalize): New. + (STATE_NOTHING, STATE_RESOLVED, STATE_IN_PROGRESS, STATE_DONE, + STATE_ERROR): Moved to java/lang/Class.h and renamed with JV_ + prefix. + (initializeClass): Use new JV_ prefixed names. Also, call + ClassLoader::resolveClass instead of _Jv_ResolveClass. + + * java/lang/Class.h (JV_STATE_PRELOADING, JV_STATE_LOADING, + JV_STATE_LOADED, JV_STATE_COMPILED, JV_STATE_PREPARED, + JV_STATE_LINKED): New. + (_Jv_WaitForState, _Jv_RegisterInitiatingLoader, + _Jv_UnregisterClass, _Jv_InternClassStrings): New friends. + (_Jv_IsInterpretedClass, _Jv_InitField, _Jv_LookupDeclaredMethod, + _Jv_DetermineVTableIndex, _Jv_ResolvePoolEntry, _Jv_PrepareClass, + _Jv_ClassReader, _Jv_InterpClass, _Jv_InterpMethod, + _Jv_InterpMethodInvocation): New friends for interpreter. + (finalize): New. + (CONSTANT_Class, CONSTANT_String, etc.): Moved to + include/java-cpool.h and renamed with JV_ prefix. + + * include/jvm.h (_Jv_makeUtf8Const, _Jv_makeUtf8TypeConst): New + decls. + (_Jv_UnregisterClass): New decl. + + * java/lang/natClassLoader.cc (_Jv_FindArrayClass): Added + class loader argument. + (_Jv_FindClass): Use class loader. + + * prims.cc (_Jv_makeUtf8Const): New function. + (_Jv_NewObjectArray): Change use of _Jv_FindArrayClass. + (_Jv_NewPrimArray): Ditto. + (_Jv_FindClassFromSignature): Ditto. + * java/lang/reflect/natArray.cc (newInstance): Ditto. + * java/lang/reflect/natMethod.cc (getType): Ditto. + + * include/java-field.h (_Jv_Field::isRef): Make robust for + non-resolved contexts. + + * boehm.cc (_Jv_MarkObj): Mark interpreter-related fields. + Also, don't mark class->next field. + + * java/lang/VirtualMachineError.java: Added FIXME note. + + * configure.in (INTERPSPEC): New spec. + * libgcj.spec.in: Added INTERPSPEC. + * Makefile.am: Added gcjh friends for java/lang/VMClassLoader and + gnu/gcj/runtime/MethodInvocation. + (libgcj_la_SOURCES): Added resolve.cc defineclass.cc interpret.cc. + (ordinary_java_source_files): Added above mentioned java classes. + + * configure: Rebuilt. + * Makefile.in: Rebuilt. + 1999-08-06 Tom Tromey <tromey@cygnus.com> * configure: Rebuilt. diff --git a/libjava/Makefile.am b/libjava/Makefile.am index c9cc9b9979c..83217c89492 100644 --- a/libjava/Makefile.am +++ b/libjava/Makefile.am @@ -27,7 +27,7 @@ data_DATA = libgcj.zip ## For now, only on native systems. if NATIVE -bin_PROGRAMS = jv-convert +bin_PROGRAMS = jv-convert gij endif ## ################################################################ @@ -77,8 +77,10 @@ endif JCFLAGS = -g JC1FLAGS = -g @LIBGCJ_JAVAFLAGS@ +LIBFFIINCS = -I$(top_srcdir)/../libffi/include -I../libffi/include + INCLUDES = -Iinclude -I$(top_srcdir)/include $(GCINCS) $(THREADINCS) \ - $(EH_COMMON_INCLUDE) $(ZINCS) + $(EH_COMMON_INCLUDE) $(ZINCS) $(LIBFFIINCS) DIVIDESPEC = @DIVIDESPEC@ @@ -97,13 +99,17 @@ c_files = $(c_source_files:.c=.lo) javao_files = $(java_source_files:.java=.lo) \ $(built_java_source_files:.java=.lo) -libgcj_la_SOURCES = prims.cc jni.cc exception.cc +## Extract the libffi object file names. +libffi_files = `$(AR) t ../libffi/.libs/libffi.a 2>/dev/null | sed 's/\.o/\.lo/g' | sed 's/^/..\/libffi\//g'` + +libgcj_la_SOURCES = prims.cc jni.cc exception.cc \ + resolve.cc defineclass.cc interpret.cc EXTRA_libgcj_la_SOURCES = boehm.cc nogc.cc posix-threads.cc no-threads.cc \ $(c_source_files) $(java_source_files) $(built_java_source_files) libgcj_la_DEPENDENCIES = libgcj.zip $(javao_files) $(nat_files) \ $(c_files) $(GCOBJS) $(THREADOBJS) libgcj_la_LIBADD = $(javao_files) $(nat_files) $(c_files) $(GCOBJS) \ - $(THREADOBJS) + $(THREADOBJS) $(libffi_files) libgcj_la_LDFLAGS = -rpath $(toolexeclibdir) \ ## The mysterious backslash is consumed by make. -version-info `grep -v '^\#' $(srcdir)/libtool-version` @@ -230,6 +236,16 @@ java/lang/reflect/Method.h: java/lang/reflect/Method.class libgcj.zip -friend 'java::lang::Class;' \ $(basename $<) +java/lang/VMClassLoader.h: java/lang/VMClassLoader.class libgcj.zip + $(GCJH) -classpath $(top_builddir) \ + -friend 'java::lang::ClassLoader;' \ + $(basename $<) + +gnu/gcj/runtime/MethodInvocation.h: gnu/gcj/runtime/MethodInvocation.class libgcj.zip + $(GCJH) -classpath $(top_builddir) \ + -friend 'class _Jv_InterpMethod;' \ + $(basename $<) + ## ################################################################ @@ -299,6 +315,26 @@ jv_convert_LDADD = $(convert_source_files:.java=.lo) libgcj.la \ jv_convert_DEPENDENCIES = $(convert_source_files:.java=.lo) \ $(GCDEPS) $(THREADDEPS) $(ZDEPS) libgcj.la libgcj.spec +gij_SOURCES = +EXTRA_gij_SOURCES = gij.cc +## We need -nodefaultlibs because we want to avoid gcj's `-lgcj'. We +## need this because we are explicitly using libtool to link using the +## `.la' file. +gij_LDFLAGS = -rpath $(toolexeclibdir) +gij_LINK = $(LIBTOOL) --mode=link $(GCJ) $(JC1FLAGS) $(LDFLAGS) \ + -o gij +## We explicitly link in the libraries we need. This way we don't +## need -nodefaultlibs, so we can still rely on gcj picking up the +## system libraries we need (via the specs file). +## We need the -L so that gcj can find libgcj with `-lgcj'. +## FIXME: should be _libs on some systems. +gij_LDADD = gij.lo libgcj.la \ + $(GCLIBS) $(THREADLIBS) $(ZLIBS) -L$(here)/.libs +## Depend on the spec file to make sure it is up to date before +## linking this program. +gij_DEPENDENCIES = gij.lo \ + $(GCDEPS) $(THREADDEPS) $(ZDEPS) libgcj.la libgcj.spec + # The Unicode consortium does not permit re-distributing the file JIS0201.TXT. # You can get it from ftp://ftp.unicode.org/Public/MAPPINGS/EASTASIA/JIS/. @@ -444,6 +480,13 @@ built_java_source_files = java/lang/ConcreteProcess.java ## convert_source_files. If the .java file has a hand-maintained ## header, please list it in special_java_source_files. ordinary_java_source_files = $(convert_source_files) \ +gnu/gcj/runtime/MethodInvocation.java \ +gnu/gcj/util/path/SearchPath.java \ +gnu/gcj/util/path/PathEntry.java \ +gnu/gcj/util/path/DirectoryPathEntry.java \ +gnu/gcj/util/path/ZipPathEntry.java \ +gnu/gcj/util/path/URLPathEntry.java \ +gnu/gcj/util/path/CacheEntry.java \ gnu/gcj/text/BaseBreakIterator.java \ gnu/gcj/text/CharacterBreakIterator.java \ gnu/gcj/text/LineBreakIterator.java \ @@ -522,6 +565,7 @@ java/lang/ClassCastException.java \ java/lang/ClassCircularityError.java \ java/lang/ClassFormatError.java \ java/lang/ClassLoader.java \ +java/lang/VMClassLoader.java \ java/lang/ClassNotFoundException.java \ java/lang/CloneNotSupportedException.java \ java/lang/Cloneable.java \ @@ -719,6 +763,7 @@ java/io/natFile.cc \ java/io/natFileDescriptor.cc \ java/lang/natCharacter.cc \ java/lang/natClass.cc \ +java/lang/natClassLoader.cc \ java/lang/natConcreteProcess.cc \ java/lang/natDouble.cc \ java/lang/natFirstThread.cc \ diff --git a/libjava/Makefile.in b/libjava/Makefile.in index 94e41891a7d..c03c3223311 100644 --- a/libjava/Makefile.in +++ b/libjava/Makefile.in @@ -78,6 +78,7 @@ GCINCS = @GCINCS@ GCLIBS = @GCLIBS@ GCOBJS = @GCOBJS@ GCSPEC = @GCSPEC@ +INTERPSPEC = @INTERPSPEC@ LD = @LD@ LIBGCJ_CFLAGS = @LIBGCJ_CFLAGS@ LIBGCJ_CXXFLAGS = @LIBGCJ_CXXFLAGS@ @@ -120,7 +121,7 @@ toolexeclib_DATA = libgcj.spec data_DATA = libgcj.zip @NATIVE_TRUE@bin_PROGRAMS = \ -@NATIVE_TRUE@jv-convert +@NATIVE_TRUE@jv-convert gij @CANADIAN_TRUE@@NULL_TARGET_TRUE@GCJ = \ @CANADIAN_TRUE@@NULL_TARGET_TRUE@gcj @CANADIAN_TRUE@@NULL_TARGET_FALSE@GCJ = \ @@ -156,8 +157,10 @@ AM_CXXFLAGS = -fno-rtti -fvtable-thunks @LIBGCJ_CXXFLAGS@ $(WARNINGS) JCFLAGS = -g JC1FLAGS = -g @LIBGCJ_JAVAFLAGS@ +LIBFFIINCS = -I$(top_srcdir)/../libffi/include -I../libffi/include + INCLUDES = -Iinclude -I$(top_srcdir)/include $(GCINCS) $(THREADINCS) \ - $(EH_COMMON_INCLUDE) $(ZINCS) + $(EH_COMMON_INCLUDE) $(ZINCS) $(LIBFFIINCS) DIVIDESPEC = @DIVIDESPEC@ @@ -168,7 +171,11 @@ javao_files = $(java_source_files:.java=.lo) \ $(built_java_source_files:.java=.lo) -libgcj_la_SOURCES = prims.cc jni.cc exception.cc +libffi_files = `$(AR) t ../libffi/.libs/libffi.a 2>/dev/null | sed 's/\.o/\.lo/g' | sed 's/^/..\/libffi\//g'` + +libgcj_la_SOURCES = prims.cc jni.cc exception.cc \ + resolve.cc defineclass.cc interpret.cc + EXTRA_libgcj_la_SOURCES = boehm.cc nogc.cc posix-threads.cc no-threads.cc \ $(c_source_files) $(java_source_files) $(built_java_source_files) @@ -176,7 +183,7 @@ libgcj_la_DEPENDENCIES = libgcj.zip $(javao_files) $(nat_files) \ $(c_files) $(GCOBJS) $(THREADOBJS) libgcj_la_LIBADD = $(javao_files) $(nat_files) $(c_files) $(GCOBJS) \ - $(THREADOBJS) + $(THREADOBJS) $(libffi_files) libgcj_la_LDFLAGS = -rpath $(toolexeclibdir) \ -version-info `grep -v '^\#' $(srcdir)/libtool-version` @@ -217,6 +224,19 @@ jv_convert_DEPENDENCIES = $(convert_source_files:.java=.lo) \ $(GCDEPS) $(THREADDEPS) $(ZDEPS) libgcj.la libgcj.spec +gij_SOURCES = +EXTRA_gij_SOURCES = gij.cc +gij_LDFLAGS = -rpath $(toolexeclibdir) +gij_LINK = $(LIBTOOL) --mode=link $(GCJ) $(JC1FLAGS) $(LDFLAGS) \ + -o gij + +gij_LDADD = gij.lo libgcj.la \ + $(GCLIBS) $(THREADLIBS) $(ZLIBS) -L$(here)/.libs + +gij_DEPENDENCIES = gij.lo \ + $(GCDEPS) $(THREADDEPS) $(ZDEPS) libgcj.la libgcj.spec + + gen_from_JIS_SOURCES = EXTRA_gen_from_JIS_SOURCES = $(srcdir)/$(CONVERT_DIR)/gen-from-JIS.c \ $(srcdir)/$(CONVERT_DIR)/make-trie.c \ @@ -294,6 +314,13 @@ java/awt/peer/WindowPeer.java built_java_source_files = java/lang/ConcreteProcess.java ordinary_java_source_files = $(convert_source_files) \ +gnu/gcj/runtime/MethodInvocation.java \ +gnu/gcj/util/path/SearchPath.java \ +gnu/gcj/util/path/PathEntry.java \ +gnu/gcj/util/path/DirectoryPathEntry.java \ +gnu/gcj/util/path/ZipPathEntry.java \ +gnu/gcj/util/path/URLPathEntry.java \ +gnu/gcj/util/path/CacheEntry.java \ gnu/gcj/text/BaseBreakIterator.java \ gnu/gcj/text/CharacterBreakIterator.java \ gnu/gcj/text/LineBreakIterator.java \ @@ -372,6 +399,7 @@ java/lang/ClassCastException.java \ java/lang/ClassCircularityError.java \ java/lang/ClassFormatError.java \ java/lang/ClassLoader.java \ +java/lang/VMClassLoader.java \ java/lang/ClassNotFoundException.java \ java/lang/CloneNotSupportedException.java \ java/lang/Cloneable.java \ @@ -569,6 +597,7 @@ java/io/natFile.cc \ java/io/natFileDescriptor.cc \ java/lang/natCharacter.cc \ java/lang/natClass.cc \ +java/lang/natClassLoader.cc \ java/lang/natConcreteProcess.cc \ java/lang/natDouble.cc \ java/lang/natFirstThread.cc \ @@ -655,13 +684,15 @@ DEFS = @DEFS@ -I. -I$(srcdir) -I./include CPPFLAGS = @CPPFLAGS@ LDFLAGS = @LDFLAGS@ LIBS = @LIBS@ -libgcj_la_OBJECTS = prims.lo jni.lo exception.lo -@NATIVE_TRUE@bin_PROGRAMS = jv-convert$(EXEEXT) +libgcj_la_OBJECTS = prims.lo jni.lo exception.lo resolve.lo \ +defineclass.lo interpret.lo +@NATIVE_TRUE@bin_PROGRAMS = jv-convert$(EXEEXT) gij$(EXEEXT) @NATIVE_TRUE@@MAINTAINER_MODE_TRUE@noinst_PROGRAMS = \ @NATIVE_TRUE@@MAINTAINER_MODE_TRUE@gen-from-JIS$(EXEEXT) PROGRAMS = $(bin_PROGRAMS) $(noinst_PROGRAMS) jv_convert_OBJECTS = +gij_OBJECTS = gen_from_JIS_OBJECTS = gen_from_JIS_LDFLAGS = CXXFLAGS = @CXXFLAGS@ @@ -688,9 +719,9 @@ GZIP_ENV = --best DIST_SUBDIRS = testsuite DEP_FILES = .deps/$(srcdir)/$(CONVERT_DIR)/gen-from-JIS.P \ .deps/$(srcdir)/$(CONVERT_DIR)/make-trie.P .deps/boehm.P \ -.deps/exception.P .deps/gnu/gcj/RawData.P \ -.deps/gnu/gcj/convert/BytesToUnicode.P .deps/gnu/gcj/convert/Convert.P \ -.deps/gnu/gcj/convert/Input_8859_1.P \ +.deps/defineclass.P .deps/exception.P .deps/gij.P \ +.deps/gnu/gcj/RawData.P .deps/gnu/gcj/convert/BytesToUnicode.P \ +.deps/gnu/gcj/convert/Convert.P .deps/gnu/gcj/convert/Input_8859_1.P \ .deps/gnu/gcj/convert/Input_EUCJIS.P \ .deps/gnu/gcj/convert/Input_JavaSrc.P \ .deps/gnu/gcj/convert/Input_SJIS.P .deps/gnu/gcj/convert/Input_UTF8.P \ @@ -703,6 +734,7 @@ DEP_FILES = .deps/$(srcdir)/$(CONVERT_DIR)/gen-from-JIS.P \ .deps/gnu/gcj/protocol/file/Handler.P \ .deps/gnu/gcj/protocol/http/Connection.P \ .deps/gnu/gcj/protocol/http/Handler.P \ +.deps/gnu/gcj/runtime/MethodInvocation.P \ .deps/gnu/gcj/text/BaseBreakIterator.P \ .deps/gnu/gcj/text/CharacterBreakIterator.P \ .deps/gnu/gcj/text/LineBreakIterator.P \ @@ -711,6 +743,12 @@ DEP_FILES = .deps/$(srcdir)/$(CONVERT_DIR)/gen-from-JIS.P \ .deps/gnu/gcj/text/SentenceBreakIterator.P \ .deps/gnu/gcj/text/WordBreakIterator.P \ .deps/gnu/gcj/util/EnumerationChain.P \ +.deps/gnu/gcj/util/path/CacheEntry.P \ +.deps/gnu/gcj/util/path/DirectoryPathEntry.P \ +.deps/gnu/gcj/util/path/PathEntry.P \ +.deps/gnu/gcj/util/path/SearchPath.P \ +.deps/gnu/gcj/util/path/URLPathEntry.P \ +.deps/gnu/gcj/util/path/ZipPathEntry.P .deps/interpret.P \ .deps/java/io/BufferedInputStream.P \ .deps/java/io/BufferedOutputStream.P .deps/java/io/BufferedReader.P \ .deps/java/io/BufferedWriter.P .deps/java/io/ByteArrayInputStream.P \ @@ -788,8 +826,9 @@ DEP_FILES = .deps/$(srcdir)/$(CONVERT_DIR)/gen-from-JIS.P \ .deps/java/lang/Throwable.P .deps/java/lang/UnknownError.P \ .deps/java/lang/UnsatisfiedLinkError.P \ .deps/java/lang/UnsupportedOperationException.P \ -.deps/java/lang/VerifyError.P .deps/java/lang/VirtualMachineError.P \ -.deps/java/lang/Void.P .deps/java/lang/dtoa.P .deps/java/lang/e_acos.P \ +.deps/java/lang/VMClassLoader.P .deps/java/lang/VerifyError.P \ +.deps/java/lang/VirtualMachineError.P .deps/java/lang/Void.P \ +.deps/java/lang/dtoa.P .deps/java/lang/e_acos.P \ .deps/java/lang/e_asin.P .deps/java/lang/e_atan2.P \ .deps/java/lang/e_exp.P .deps/java/lang/e_fmod.P \ .deps/java/lang/e_log.P .deps/java/lang/e_pow.P \ @@ -872,9 +911,9 @@ DEP_FILES = .deps/$(srcdir)/$(CONVERT_DIR)/gen-from-JIS.P \ .deps/java/util/zip/ZipException.P .deps/java/util/zip/ZipFile.P \ .deps/java/util/zip/ZipInputStream.P \ .deps/java/util/zip/ZipOutputStream.P .deps/jni.P .deps/no-threads.P \ -.deps/nogc.P .deps/posix-threads.P .deps/prims.P -SOURCES = $(libgcj_la_SOURCES) $(EXTRA_libgcj_la_SOURCES) $(jv_convert_SOURCES) $(EXTRA_jv_convert_SOURCES) $(gen_from_JIS_SOURCES) $(EXTRA_gen_from_JIS_SOURCES) -OBJECTS = $(libgcj_la_OBJECTS) $(jv_convert_OBJECTS) $(gen_from_JIS_OBJECTS) +.deps/nogc.P .deps/posix-threads.P .deps/prims.P .deps/resolve.P +SOURCES = $(libgcj_la_SOURCES) $(EXTRA_libgcj_la_SOURCES) $(jv_convert_SOURCES) $(EXTRA_jv_convert_SOURCES) $(gij_SOURCES) $(EXTRA_gij_SOURCES) $(gen_from_JIS_SOURCES) $(EXTRA_gen_from_JIS_SOURCES) +OBJECTS = $(libgcj_la_OBJECTS) $(jv_convert_OBJECTS) $(gij_OBJECTS) $(gen_from_JIS_OBJECTS) all: all-redirect .SUFFIXES: @@ -1022,6 +1061,10 @@ jv-convert$(EXEEXT): $(jv_convert_OBJECTS) $(jv_convert_DEPENDENCIES) @rm -f jv-convert$(EXEEXT) $(jv_convert_LINK) $(jv_convert_LDFLAGS) $(jv_convert_OBJECTS) $(jv_convert_LDADD) $(LIBS) +gij$(EXEEXT): $(gij_OBJECTS) $(gij_DEPENDENCIES) + @rm -f gij$(EXEEXT) + $(gij_LINK) $(gij_LDFLAGS) $(gij_OBJECTS) $(gij_LDADD) $(LIBS) + gen-from-JIS$(EXEEXT): $(gen_from_JIS_OBJECTS) $(gen_from_JIS_DEPENDENCIES) @rm -f gen-from-JIS$(EXEEXT) $(LINK) $(gen_from_JIS_LDFLAGS) $(gen_from_JIS_OBJECTS) $(gen_from_JIS_LDADD) $(LIBS) @@ -1451,6 +1494,16 @@ java/lang/reflect/Method.h: java/lang/reflect/Method.class libgcj.zip -friend 'java::lang::Class;' \ $(basename $<) +java/lang/VMClassLoader.h: java/lang/VMClassLoader.class libgcj.zip + $(GCJH) -classpath $(top_builddir) \ + -friend 'java::lang::ClassLoader;' \ + $(basename $<) + +gnu/gcj/runtime/MethodInvocation.h: gnu/gcj/runtime/MethodInvocation.class libgcj.zip + $(GCJH) -classpath $(top_builddir) \ + -friend 'class _Jv_InterpMethod;' \ + $(basename $<) + maintainer-check: libgcj.la $(NM) .libs/libgcj.a | grep ' T ' \ | grep -v '4java' \ diff --git a/libjava/NEWS b/libjava/NEWS index 114e48ac999..e83836db437 100644 --- a/libjava/NEWS +++ b/libjava/NEWS @@ -1,3 +1,7 @@ +New in libgcj X.XX: + +* libgcj now includes a bytecode interpreter. + New in libgcj 2.95: * First public release diff --git a/libjava/THANKS b/libjava/THANKS index 93b24a8b91c..5b9b9f8cf07 100644 --- a/libjava/THANKS +++ b/libjava/THANKS @@ -10,9 +10,10 @@ Eric Christopher echristo@cygnus.com Franz Sirl Franz.Sirl-kernel@lauterbach.com Geoff Berry gcb@gnu.org Gilles Zunino Gilles.Zunino@hei.fr -Per Bothner per@bother.com +Kresten Krab Thorup krab@gnu.org +Per Bothner per@bothner.com Rainer Orth ro@TechFak.Uni-Bielefeld.DE -Stu Grossman grossman@cygnus.com +Stu Grossman grossman@juniper.com Tom Tromey tromey@cygnus.com Urban Widmark urban@svenskatest.se Warren Levy warrenl@cygnus.com diff --git a/libjava/acconfig.h b/libjava/acconfig.h index daddbb14102..1bd5025ef32 100644 --- a/libjava/acconfig.h +++ b/libjava/acconfig.h @@ -93,3 +93,6 @@ #undef HAVE_READDIR_R #undef HAVE_GETHOSTBYNAME_R #undef HAVE_GETHOSTBYADDR_R + +/* Define if you want a bytecode interpreter. */ +#undef INTERPRETER diff --git a/libjava/boehm.cc b/libjava/boehm.cc index f9e72bade8e..2e1b5f23cf7 100644 --- a/libjava/boehm.cc +++ b/libjava/boehm.cc @@ -16,6 +16,7 @@ details. */ #include <java/lang/Class.h> #include <jvm.h> #include <java-field.h> +#include <java-interp.h> // We need to include gc_priv.h. However, it tries to include // config.h if it hasn't already been included. So we force the @@ -97,8 +98,14 @@ _Jv_MarkObj (void *addr, void *msp, void *msl, void * /*env*/) { jclass c = (jclass) addr; +#if 0 + // The next field should probably not be marked, since this is + // only used in the class hash table. Marking this field + // basically prohibits class unloading. --Kresten w = (word) c->next; MAYBE_MARK (w, mark_stack_ptr, mark_stack_limit, c, c2label); +#endif + w = (word) c->name; MAYBE_MARK (w, mark_stack_ptr, mark_stack_limit, c, c3label); w = (word) c->superclass; @@ -109,12 +116,23 @@ _Jv_MarkObj (void *addr, void *msp, void *msl, void * /*env*/) MAYBE_MARK (w, mark_stack_ptr, mark_stack_limit, c, c5label); } +#ifdef INTERPRETER + if (_Jv_IsInterpretedClass (c)) + { + w = (word) c->constants.tags; + MAYBE_MARK (w, mark_stack_ptr, mark_stack_limit, c, c5alabel); + w = (word) c->constants.data; + MAYBE_MARK (w, mark_stack_ptr, mark_stack_limit, c, c5blabel); + } +#endif + // If the class is an array, then the methods field holds a // pointer to the element class. If the class is primitive, // then the methods field holds a pointer to the array class. w = (word) c->methods; MAYBE_MARK (w, mark_stack_ptr, mark_stack_limit, c, c6label); + if (! c->isArray() && ! c->isPrimitive()) { // Scan each method in the cases where `methods' really @@ -127,7 +145,19 @@ _Jv_MarkObj (void *addr, void *msp, void *msl, void * /*env*/) w = (word) c->methods[i].signature; MAYBE_MARK (w, mark_stack_ptr, mark_stack_limit, c, cm2label); + // FIXME: `ncode' entry? + +#ifdef INTERPRETER + // The interpreter installs a heap-allocated + // trampoline here, so we'll mark it. + if (_Jv_IsInterpretedClass (c)) + { + w = (word) c->methods[i].ncode; + MAYBE_MARK (w, mark_stack_ptr, mark_stack_limit, c, + cm3label); + } +#endif } } @@ -136,12 +166,34 @@ _Jv_MarkObj (void *addr, void *msp, void *msl, void * /*env*/) MAYBE_MARK (w, mark_stack_ptr, mark_stack_limit, c, c8label); for (int i = 0; i < c->field_count; ++i) { + _Jv_Field* field = &c->fields[i]; + #ifndef COMPACT_FIELDS - w = (word) c->fields[i].name; + w = (word) field->name; MAYBE_MARK (w, mark_stack_ptr, mark_stack_limit, c, c8alabel); #endif - w = (word) c->fields[i].type; + w = (word) field->type; MAYBE_MARK (w, mark_stack_ptr, mark_stack_limit, c, c8blabel); + + // For the interpreter, we also need to mark the memory + // containing static members + if (field->flags & 0x0008) + { + w = (word) field->u.addr; + MAYBE_MARK (w, mark_stack_ptr, mark_stack_limit, c, c8clabel); + + // also, if the static member is a reference, + // mark also the value pointed to. We check for isResolved + // since marking can happen before memory is allocated for + // static members. + if (JvFieldIsRef (field) && field->isResolved()) + { + jobject val = *(jobject*) field->u.addr; + w = (word) val; + MAYBE_MARK (w, mark_stack_ptr, mark_stack_limit, + c, c8elabel); + } + } } w = (word) c->vtable; @@ -155,6 +207,28 @@ _Jv_MarkObj (void *addr, void *msp, void *msl, void * /*env*/) } w = (word) c->loader; MAYBE_MARK (w, mark_stack_ptr, mark_stack_limit, c, cBlabel); + +#ifdef INTERPRETER + if (_Jv_IsInterpretedClass (c)) + { + _Jv_InterpClass* ic = (_Jv_InterpClass*)c; + + w = (word) ic->interpreted_methods; + MAYBE_MARK (w, mark_stack_ptr, mark_stack_limit, ic, cElabel); + + for (int i = 0; i < c->method_count; i++) + { + w = (word) ic->interpreted_methods[i]; + MAYBE_MARK (w, mark_stack_ptr, mark_stack_limit, ic, \ + cFlabel); + } + + w = (word) ic->field_initializers; + MAYBE_MARK (w, mark_stack_ptr, mark_stack_limit, ic, cGlabel); + + } +#endif + } else { diff --git a/libjava/configure b/libjava/configure index a08e3884909..eac0e543a4b 100755 --- a/libjava/configure +++ b/libjava/configure @@ -35,6 +35,8 @@ ac_help="$ac_help ac_help="$ac_help --enable-libgcj-debug Enable runtime debugging code" ac_help="$ac_help + --enable-interpreter Enable interpreter" +ac_help="$ac_help --with-ecos Enable runtime eCos target support." ac_help="$ac_help --with-system-zlib Use installed libz" @@ -59,6 +61,7 @@ program_suffix=NONE program_transform_name=s,x,x, silent= site= +sitefile= srcdir= target=NONE verbose= @@ -173,6 +176,7 @@ Configuration: --help print this message --no-create do not create output files --quiet, --silent do not print \`checking...' messages + --site-file=FILE use FILE as the site file --version print the version of autoconf that created configure Directory and file names: --prefix=PREFIX install architecture-independent files in PREFIX @@ -343,6 +347,11 @@ EOF -site=* | --site=* | --sit=*) site="$ac_optarg" ;; + -site-file | --site-file | --site-fil | --site-fi | --site-f) + ac_prev=sitefile ;; + -site-file=* | --site-file=* | --site-fil=* | --site-fi=* | --site-f=*) + sitefile="$ac_optarg" ;; + -srcdir | --srcdir | --srcdi | --srcd | --src | --sr) ac_prev=srcdir ;; -srcdir=* | --srcdir=* | --srcdi=* | --srcd=* | --src=* | --sr=*) @@ -508,12 +517,16 @@ fi srcdir=`echo "${srcdir}" | sed 's%\([^/]\)/*$%\1%'` # Prefer explicitly selected file to automatically selected ones. -if test -z "$CONFIG_SITE"; then - if test "x$prefix" != xNONE; then - CONFIG_SITE="$prefix/share/config.site $prefix/etc/config.site" - else - CONFIG_SITE="$ac_default_prefix/share/config.site $ac_default_prefix/etc/config.site" +if test -z "$sitefile"; then + if test -z "$CONFIG_SITE"; then + if test "x$prefix" != xNONE; then + CONFIG_SITE="$prefix/share/config.site $prefix/etc/config.site" + else + CONFIG_SITE="$ac_default_prefix/share/config.site $ac_default_prefix/etc/config.site" + fi fi +else + CONFIG_SITE="$sitefile" fi for ac_site_file in $CONFIG_SITE; do if test -r "$ac_site_file"; then @@ -601,7 +614,7 @@ else { echo "configure: error: can not run $ac_config_sub" 1>&2; exit 1; } fi echo $ac_n "checking host system type""... $ac_c" 1>&6 -echo "configure:605: checking host system type" >&5 +echo "configure:618: checking host system type" >&5 host_alias=$host case "$host_alias" in @@ -622,7 +635,7 @@ host_os=`echo $host | sed 's/^\([^-]*\)-\([^-]*\)-\(.*\)$/\3/'` echo "$ac_t""$host" 1>&6 echo $ac_n "checking target system type""... $ac_c" 1>&6 -echo "configure:626: checking target system type" >&5 +echo "configure:639: checking target system type" >&5 target_alias=$target case "$target_alias" in @@ -640,7 +653,7 @@ target_os=`echo $target | sed 's/^\([^-]*\)-\([^-]*\)-\(.*\)$/\3/'` echo "$ac_t""$target" 1>&6 echo $ac_n "checking build system type""... $ac_c" 1>&6 -echo "configure:644: checking build system type" >&5 +echo "configure:657: checking build system type" >&5 build_alias=$build case "$build_alias" in @@ -688,7 +701,7 @@ fi # SVR4 /usr/ucb/install, which tries to use the nonexistent group "staff" # ./install, which can be erroneously created by make from ./install.sh. echo $ac_n "checking for a BSD compatible install""... $ac_c" 1>&6 -echo "configure:692: checking for a BSD compatible install" >&5 +echo "configure:705: checking for a BSD compatible install" >&5 if test -z "$INSTALL"; then if eval "test \"`echo '$''{'ac_cv_path_install'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 @@ -741,7 +754,7 @@ test -z "$INSTALL_SCRIPT" && INSTALL_SCRIPT='${INSTALL_PROGRAM}' test -z "$INSTALL_DATA" && INSTALL_DATA='${INSTALL} -m 644' echo $ac_n "checking whether build environment is sane""... $ac_c" 1>&6 -echo "configure:745: checking whether build environment is sane" >&5 +echo "configure:758: checking whether build environment is sane" >&5 # Just in case sleep 1 echo timestamp > conftestfile @@ -798,7 +811,7 @@ test "$program_suffix" != NONE && test "$program_transform_name" = "" && program_transform_name="s,x,x," echo $ac_n "checking whether ${MAKE-make} sets \${MAKE}""... $ac_c" 1>&6 -echo "configure:802: checking whether ${MAKE-make} sets \${MAKE}" >&5 +echo "configure:815: checking whether ${MAKE-make} sets \${MAKE}" >&5 set dummy ${MAKE-make}; ac_make=`echo "$2" | sed 'y%./+-%__p_%'` if eval "test \"`echo '$''{'ac_cv_prog_make_${ac_make}_set'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 @@ -831,12 +844,12 @@ else fi echo $ac_n "checking for Cygwin environment""... $ac_c" 1>&6 -echo "configure:835: checking for Cygwin environment" >&5 +echo "configure:848: checking for Cygwin environment" >&5 if eval "test \"`echo '$''{'ac_cv_cygwin'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 else cat > conftest.$ac_ext <<EOF -#line 840 "configure" +#line 853 "configure" #include "confdefs.h" int main() { @@ -847,7 +860,7 @@ int main() { return __CYGWIN__; ; return 0; } EOF -if { (eval echo configure:851: \"$ac_compile\") 1>&5; (eval $ac_compile) 2>&5; }; then +if { (eval echo configure:864: \"$ac_compile\") 1>&5; (eval $ac_compile) 2>&5; }; then rm -rf conftest* ac_cv_cygwin=yes else @@ -864,19 +877,19 @@ echo "$ac_t""$ac_cv_cygwin" 1>&6 CYGWIN= test "$ac_cv_cygwin" = yes && CYGWIN=yes echo $ac_n "checking for mingw32 environment""... $ac_c" 1>&6 -echo "configure:868: checking for mingw32 environment" >&5 +echo "configure:881: checking for mingw32 environment" >&5 if eval "test \"`echo '$''{'ac_cv_mingw32'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 else cat > conftest.$ac_ext <<EOF -#line 873 "configure" +#line 886 "configure" #include "confdefs.h" int main() { return __MINGW32__; ; return 0; } EOF -if { (eval echo configure:880: \"$ac_compile\") 1>&5; (eval $ac_compile) 2>&5; }; then +if { (eval echo configure:893: \"$ac_compile\") 1>&5; (eval $ac_compile) 2>&5; }; then rm -rf conftest* ac_cv_mingw32=yes else @@ -924,7 +937,7 @@ else { echo "configure: error: can not run $ac_config_sub" 1>&2; exit 1; } fi echo $ac_n "checking host system type""... $ac_c" 1>&6 -echo "configure:928: checking host system type" >&5 +echo "configure:941: checking host system type" >&5 host_alias=$host case "$host_alias" in @@ -966,7 +979,7 @@ EOF missing_dir=`cd $ac_aux_dir && pwd` echo $ac_n "checking for working aclocal""... $ac_c" 1>&6 -echo "configure:970: checking for working aclocal" >&5 +echo "configure:983: checking for working aclocal" >&5 # Run test in a subshell; some versions of sh will print an error if # an executable is not found, even if stderr is redirected. # Redirect stdin to placate older versions of autoconf. Sigh. @@ -979,7 +992,7 @@ else fi echo $ac_n "checking for working autoconf""... $ac_c" 1>&6 -echo "configure:983: checking for working autoconf" >&5 +echo "configure:996: checking for working autoconf" >&5 # Run test in a subshell; some versions of sh will print an error if # an executable is not found, even if stderr is redirected. # Redirect stdin to placate older versions of autoconf. Sigh. @@ -992,7 +1005,7 @@ else fi echo $ac_n "checking for working automake""... $ac_c" 1>&6 -echo "configure:996: checking for working automake" >&5 +echo "configure:1009: checking for working automake" >&5 # Run test in a subshell; some versions of sh will print an error if # an executable is not found, even if stderr is redirected. # Redirect stdin to placate older versions of autoconf. Sigh. @@ -1005,7 +1018,7 @@ else fi echo $ac_n "checking for working autoheader""... $ac_c" 1>&6 -echo "configure:1009: checking for working autoheader" >&5 +echo "configure:1022: checking for working autoheader" >&5 # Run test in a subshell; some versions of sh will print an error if # an executable is not found, even if stderr is redirected. # Redirect stdin to placate older versions of autoconf. Sigh. @@ -1018,7 +1031,7 @@ else fi echo $ac_n "checking for working makeinfo""... $ac_c" 1>&6 -echo "configure:1022: checking for working makeinfo" >&5 +echo "configure:1035: checking for working makeinfo" >&5 # Run test in a subshell; some versions of sh will print an error if # an executable is not found, even if stderr is redirected. # Redirect stdin to placate older versions of autoconf. Sigh. @@ -1043,7 +1056,7 @@ fi # Extract the first word of "gcc", so it can be a program name with args. set dummy gcc; ac_word=$2 echo $ac_n "checking for $ac_word""... $ac_c" 1>&6 -echo "configure:1047: checking for $ac_word" >&5 +echo "configure:1060: checking for $ac_word" >&5 if eval "test \"`echo '$''{'ac_cv_prog_CC'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 else @@ -1073,7 +1086,7 @@ if test -z "$CC"; then # Extract the first word of "cc", so it can be a program name with args. set dummy cc; ac_word=$2 echo $ac_n "checking for $ac_word""... $ac_c" 1>&6 -echo "configure:1077: checking for $ac_word" >&5 +echo "configure:1090: checking for $ac_word" >&5 if eval "test \"`echo '$''{'ac_cv_prog_CC'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 else @@ -1122,7 +1135,7 @@ fi fi echo $ac_n "checking whether we are using GNU C""... $ac_c" 1>&6 -echo "configure:1126: checking whether we are using GNU C" >&5 +echo "configure:1139: checking whether we are using GNU C" >&5 if eval "test \"`echo '$''{'ac_cv_prog_gcc'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 else @@ -1131,7 +1144,7 @@ else yes; #endif EOF -if { ac_try='${CC-cc} -E conftest.c'; { (eval echo configure:1135: \"$ac_try\") 1>&5; (eval $ac_try) 2>&5; }; } | egrep yes >/dev/null 2>&1; then +if { ac_try='${CC-cc} -E conftest.c'; { (eval echo configure:1148: \"$ac_try\") 1>&5; (eval $ac_try) 2>&5; }; } | egrep yes >/dev/null 2>&1; then ac_cv_prog_gcc=yes else ac_cv_prog_gcc=no @@ -1146,7 +1159,7 @@ if test $ac_cv_prog_gcc = yes; then ac_save_CFLAGS="$CFLAGS" CFLAGS= echo $ac_n "checking whether ${CC-cc} accepts -g""... $ac_c" 1>&6 -echo "configure:1150: checking whether ${CC-cc} accepts -g" >&5 +echo "configure:1163: checking whether ${CC-cc} accepts -g" >&5 if eval "test \"`echo '$''{'ac_cv_prog_cc_g'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 else @@ -1182,7 +1195,7 @@ do # Extract the first word of "$ac_prog", so it can be a program name with args. set dummy $ac_prog; ac_word=$2 echo $ac_n "checking for $ac_word""... $ac_c" 1>&6 -echo "configure:1186: checking for $ac_word" >&5 +echo "configure:1199: checking for $ac_word" >&5 if eval "test \"`echo '$''{'ac_cv_prog_CXX'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 else @@ -1215,7 +1228,7 @@ test -n "$CXX" || CXX="gcc" test -z "$CXX" && { echo "configure: error: no acceptable c++ found in \$PATH" 1>&2; exit 1; } echo $ac_n "checking whether we are using GNU C++""... $ac_c" 1>&6 -echo "configure:1219: checking whether we are using GNU C++" >&5 +echo "configure:1232: checking whether we are using GNU C++" >&5 if eval "test \"`echo '$''{'ac_cv_prog_gxx'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 else @@ -1224,7 +1237,7 @@ else yes; #endif EOF -if { ac_try='${CXX-g++} -E conftest.C'; { (eval echo configure:1228: \"$ac_try\") 1>&5; (eval $ac_try) 2>&5; }; } | egrep yes >/dev/null 2>&1; then +if { ac_try='${CXX-g++} -E conftest.C'; { (eval echo configure:1241: \"$ac_try\") 1>&5; (eval $ac_try) 2>&5; }; } | egrep yes >/dev/null 2>&1; then ac_cv_prog_gxx=yes else ac_cv_prog_gxx=no @@ -1239,7 +1252,7 @@ if test $ac_cv_prog_gxx = yes; then ac_save_CXXFLAGS="$CXXFLAGS" CXXFLAGS= echo $ac_n "checking whether ${CXX-g++} accepts -g""... $ac_c" 1>&6 -echo "configure:1243: checking whether ${CXX-g++} accepts -g" >&5 +echo "configure:1256: checking whether ${CXX-g++} accepts -g" >&5 if eval "test \"`echo '$''{'ac_cv_prog_cxx_g'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 else @@ -1272,7 +1285,7 @@ fi # LIBGCJ_CONFIGURE, which doesn't work because that means that it will # be run before AC_CANONICAL_HOST. echo $ac_n "checking build system type""... $ac_c" 1>&6 -echo "configure:1276: checking build system type" >&5 +echo "configure:1289: checking build system type" >&5 build_alias=$build case "$build_alias" in @@ -1293,7 +1306,7 @@ echo "$ac_t""$build" 1>&6 # Extract the first word of "${ac_tool_prefix}as", so it can be a program name with args. set dummy ${ac_tool_prefix}as; ac_word=$2 echo $ac_n "checking for $ac_word""... $ac_c" 1>&6 -echo "configure:1297: checking for $ac_word" >&5 +echo "configure:1310: checking for $ac_word" >&5 if eval "test \"`echo '$''{'ac_cv_prog_AS'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 else @@ -1325,7 +1338,7 @@ fi # Extract the first word of "${ac_tool_prefix}ar", so it can be a program name with args. set dummy ${ac_tool_prefix}ar; ac_word=$2 echo $ac_n "checking for $ac_word""... $ac_c" 1>&6 -echo "configure:1329: checking for $ac_word" >&5 +echo "configure:1342: checking for $ac_word" >&5 if eval "test \"`echo '$''{'ac_cv_prog_AR'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 else @@ -1357,7 +1370,7 @@ fi # Extract the first word of "${ac_tool_prefix}ranlib", so it can be a program name with args. set dummy ${ac_tool_prefix}ranlib; ac_word=$2 echo $ac_n "checking for $ac_word""... $ac_c" 1>&6 -echo "configure:1361: checking for $ac_word" >&5 +echo "configure:1374: checking for $ac_word" >&5 if eval "test \"`echo '$''{'ac_cv_prog_RANLIB'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 else @@ -1389,7 +1402,7 @@ if test -n "$ac_tool_prefix"; then # Extract the first word of "ranlib", so it can be a program name with args. set dummy ranlib; ac_word=$2 echo $ac_n "checking for $ac_word""... $ac_c" 1>&6 -echo "configure:1393: checking for $ac_word" >&5 +echo "configure:1406: checking for $ac_word" >&5 if eval "test \"`echo '$''{'ac_cv_prog_RANLIB'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 else @@ -1434,7 +1447,7 @@ fi # SVR4 /usr/ucb/install, which tries to use the nonexistent group "staff" # ./install, which can be erroneously created by make from ./install.sh. echo $ac_n "checking for a BSD compatible install""... $ac_c" 1>&6 -echo "configure:1438: checking for a BSD compatible install" >&5 +echo "configure:1451: checking for a BSD compatible install" >&5 if test -z "$INSTALL"; then if eval "test \"`echo '$''{'ac_cv_path_install'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 @@ -1488,7 +1501,7 @@ test -z "$INSTALL_DATA" && INSTALL_DATA='${INSTALL} -m 644' echo $ac_n "checking whether to enable maintainer-specific portions of Makefiles""... $ac_c" 1>&6 -echo "configure:1492: checking whether to enable maintainer-specific portions of Makefiles" >&5 +echo "configure:1505: checking whether to enable maintainer-specific portions of Makefiles" >&5 # Check whether --enable-maintainer-mode or --disable-maintainer-mode was given. if test "${enable_maintainer_mode+set}" = set; then enableval="$enable_maintainer_mode" @@ -1522,7 +1535,7 @@ if false; then echo $ac_n "checking for executable suffix""... $ac_c" 1>&6 -echo "configure:1526: checking for executable suffix" >&5 +echo "configure:1539: checking for executable suffix" >&5 if eval "test \"`echo '$''{'ac_cv_exeext'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 else @@ -1532,7 +1545,7 @@ else rm -f conftest* echo 'int main () { return 0; }' > conftest.$ac_ext ac_cv_exeext= - if { (eval echo configure:1536: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; }; then + if { (eval echo configure:1549: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; }; then for file in conftest.*; do case $file in *.c | *.o | *.obj | *.ilk | *.pdb) ;; @@ -1654,7 +1667,7 @@ fi # Extract the first word of "ranlib", so it can be a program name with args. set dummy ranlib; ac_word=$2 echo $ac_n "checking for $ac_word""... $ac_c" 1>&6 -echo "configure:1658: checking for $ac_word" >&5 +echo "configure:1671: checking for $ac_word" >&5 if eval "test \"`echo '$''{'ac_cv_prog_RANLIB'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 else @@ -1684,7 +1697,7 @@ fi # Extract the first word of "gcc", so it can be a program name with args. set dummy gcc; ac_word=$2 echo $ac_n "checking for $ac_word""... $ac_c" 1>&6 -echo "configure:1688: checking for $ac_word" >&5 +echo "configure:1701: checking for $ac_word" >&5 if eval "test \"`echo '$''{'ac_cv_prog_CC'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 else @@ -1714,7 +1727,7 @@ if test -z "$CC"; then # Extract the first word of "cc", so it can be a program name with args. set dummy cc; ac_word=$2 echo $ac_n "checking for $ac_word""... $ac_c" 1>&6 -echo "configure:1718: checking for $ac_word" >&5 +echo "configure:1731: checking for $ac_word" >&5 if eval "test \"`echo '$''{'ac_cv_prog_CC'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 else @@ -1765,7 +1778,7 @@ fi # Extract the first word of "cl", so it can be a program name with args. set dummy cl; ac_word=$2 echo $ac_n "checking for $ac_word""... $ac_c" 1>&6 -echo "configure:1769: checking for $ac_word" >&5 +echo "configure:1782: checking for $ac_word" >&5 if eval "test \"`echo '$''{'ac_cv_prog_CC'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 else @@ -1797,7 +1810,7 @@ fi fi echo $ac_n "checking whether the C compiler ($CC $CFLAGS $LDFLAGS) works""... $ac_c" 1>&6 -echo "configure:1801: checking whether the C compiler ($CC $CFLAGS $LDFLAGS) works" >&5 +echo "configure:1814: checking whether the C compiler ($CC $CFLAGS $LDFLAGS) works" >&5 ac_ext=c # CFLAGS is not in ac_cpp because -g, -O, etc. are not valid cpp options. @@ -1808,12 +1821,12 @@ cross_compiling=$ac_cv_prog_cc_cross cat > conftest.$ac_ext << EOF -#line 1812 "configure" +#line 1825 "configure" #include "confdefs.h" main(){return(0);} EOF -if { (eval echo configure:1817: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext}; then +if { (eval echo configure:1830: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext}; then ac_cv_prog_cc_works=yes # If we can't run a trivial program, we are probably using a cross compiler. if (./conftest; exit) 2>/dev/null; then @@ -1839,12 +1852,12 @@ if test $ac_cv_prog_cc_works = no; then { echo "configure: error: installation or configuration problem: C compiler cannot create executables." 1>&2; exit 1; } fi echo $ac_n "checking whether the C compiler ($CC $CFLAGS $LDFLAGS) is a cross-compiler""... $ac_c" 1>&6 -echo "configure:1843: checking whether the C compiler ($CC $CFLAGS $LDFLAGS) is a cross-compiler" >&5 +echo "configure:1856: checking whether the C compiler ($CC $CFLAGS $LDFLAGS) is a cross-compiler" >&5 echo "$ac_t""$ac_cv_prog_cc_cross" 1>&6 cross_compiling=$ac_cv_prog_cc_cross echo $ac_n "checking whether we are using GNU C""... $ac_c" 1>&6 -echo "configure:1848: checking whether we are using GNU C" >&5 +echo "configure:1861: checking whether we are using GNU C" >&5 if eval "test \"`echo '$''{'ac_cv_prog_gcc'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 else @@ -1853,7 +1866,7 @@ else yes; #endif EOF -if { ac_try='${CC-cc} -E conftest.c'; { (eval echo configure:1857: \"$ac_try\") 1>&5; (eval $ac_try) 2>&5; }; } | egrep yes >/dev/null 2>&1; then +if { ac_try='${CC-cc} -E conftest.c'; { (eval echo configure:1870: \"$ac_try\") 1>&5; (eval $ac_try) 2>&5; }; } | egrep yes >/dev/null 2>&1; then ac_cv_prog_gcc=yes else ac_cv_prog_gcc=no @@ -1872,7 +1885,7 @@ ac_test_CFLAGS="${CFLAGS+set}" ac_save_CFLAGS="$CFLAGS" CFLAGS= echo $ac_n "checking whether ${CC-cc} accepts -g""... $ac_c" 1>&6 -echo "configure:1876: checking whether ${CC-cc} accepts -g" >&5 +echo "configure:1889: checking whether ${CC-cc} accepts -g" >&5 if eval "test \"`echo '$''{'ac_cv_prog_cc_g'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 else @@ -1915,7 +1928,7 @@ ac_prog=ld if test "$ac_cv_prog_gcc" = yes; then # Check if gcc -print-prog-name=ld gives a path. echo $ac_n "checking for ld used by GCC""... $ac_c" 1>&6 -echo "configure:1919: checking for ld used by GCC" >&5 +echo "configure:1932: checking for ld used by GCC" >&5 ac_prog=`($CC -print-prog-name=ld) 2>&5` case "$ac_prog" in # Accept absolute paths. @@ -1939,10 +1952,10 @@ echo "configure:1919: checking for ld used by GCC" >&5 esac elif test "$with_gnu_ld" = yes; then echo $ac_n "checking for GNU ld""... $ac_c" 1>&6 -echo "configure:1943: checking for GNU ld" >&5 +echo "configure:1956: checking for GNU ld" >&5 else echo $ac_n "checking for non-GNU ld""... $ac_c" 1>&6 -echo "configure:1946: checking for non-GNU ld" >&5 +echo "configure:1959: checking for non-GNU ld" >&5 fi if eval "test \"`echo '$''{'ac_cv_path_LD'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 @@ -1978,7 +1991,7 @@ fi test -z "$LD" && { echo "configure: error: no acceptable ld found in \$PATH" 1>&2; exit 1; } echo $ac_n "checking if the linker ($LD) is GNU ld""... $ac_c" 1>&6 -echo "configure:1982: checking if the linker ($LD) is GNU ld" >&5 +echo "configure:1995: checking if the linker ($LD) is GNU ld" >&5 if eval "test \"`echo '$''{'ac_cv_prog_gnu_ld'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 else @@ -1994,7 +2007,7 @@ echo "$ac_t""$ac_cv_prog_gnu_ld" 1>&6 echo $ac_n "checking for BSD-compatible nm""... $ac_c" 1>&6 -echo "configure:1998: checking for BSD-compatible nm" >&5 +echo "configure:2011: checking for BSD-compatible nm" >&5 if eval "test \"`echo '$''{'ac_cv_path_NM'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 else @@ -2031,7 +2044,7 @@ echo "$ac_t""$NM" 1>&6 echo $ac_n "checking whether ln -s works""... $ac_c" 1>&6 -echo "configure:2035: checking whether ln -s works" >&5 +echo "configure:2048: checking whether ln -s works" >&5 if eval "test \"`echo '$''{'ac_cv_prog_LN_S'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 else @@ -2075,8 +2088,8 @@ test x"$silent" = xyes && libtool_flags="$libtool_flags --silent" case "$host" in *-*-irix6*) # Find out which ABI we are using. - echo '#line 2079 "configure"' > conftest.$ac_ext - if { (eval echo configure:2080: \"$ac_compile\") 1>&5; (eval $ac_compile) 2>&5; }; then + echo '#line 2092 "configure"' > conftest.$ac_ext + if { (eval echo configure:2093: \"$ac_compile\") 1>&5; (eval $ac_compile) 2>&5; }; then case "`/usr/bin/file conftest.o`" in *32-bit*) LD="${LD-ld} -32" @@ -2097,19 +2110,19 @@ case "$host" in SAVE_CFLAGS="$CFLAGS" CFLAGS="$CFLAGS -belf" echo $ac_n "checking whether the C compiler needs -belf""... $ac_c" 1>&6 -echo "configure:2101: checking whether the C compiler needs -belf" >&5 +echo "configure:2114: checking whether the C compiler needs -belf" >&5 if eval "test \"`echo '$''{'lt_cv_cc_needs_belf'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 else cat > conftest.$ac_ext <<EOF -#line 2106 "configure" +#line 2119 "configure" #include "confdefs.h" int main() { ; return 0; } EOF -if { (eval echo configure:2113: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext}; then +if { (eval echo configure:2126: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext}; then rm -rf conftest* lt_cv_cc_needs_belf=yes else @@ -2253,6 +2266,21 @@ EOF fi +# Check whether --enable-interpreter or --disable-interpreter was given. +if test "${enable_interpreter+set}" = set; then + enableval="$enable_interpreter" + if test "$enable_interpreter" = yes; then + cat >> confdefs.h <<\EOF +#define INTERPRETER 1 +EOF + + fi +fi + + +INTERPSPEC= + + TARGET_ECOS="no" # Check whether --with-ecos or --without-ecos was given. if test "${with_ecos+set}" = set; then @@ -2278,7 +2306,7 @@ EOF esac echo $ac_n "checking how to run the C preprocessor""... $ac_c" 1>&6 -echo "configure:2282: checking how to run the C preprocessor" >&5 +echo "configure:2310: checking how to run the C preprocessor" >&5 # On Suns, sometimes $CPP names a directory. if test -n "$CPP" && test -d "$CPP"; then CPP= @@ -2293,13 +2321,13 @@ else # On the NeXT, cc -E runs the code through the compiler's parser, # not just through cpp. cat > conftest.$ac_ext <<EOF -#line 2297 "configure" +#line 2325 "configure" #include "confdefs.h" #include <assert.h> Syntax Error EOF ac_try="$ac_cpp conftest.$ac_ext >/dev/null 2>conftest.out" -{ (eval echo configure:2303: \"$ac_try\") 1>&5; (eval $ac_try) 2>&5; } +{ (eval echo configure:2331: \"$ac_try\") 1>&5; (eval $ac_try) 2>&5; } ac_err=`grep -v '^ *+' conftest.out | grep -v "^conftest.${ac_ext}\$"` if test -z "$ac_err"; then : @@ -2310,13 +2338,13 @@ else rm -rf conftest* CPP="${CC-cc} -E -traditional-cpp" cat > conftest.$ac_ext <<EOF -#line 2314 "configure" +#line 2342 "configure" #include "confdefs.h" #include <assert.h> Syntax Error EOF ac_try="$ac_cpp conftest.$ac_ext >/dev/null 2>conftest.out" -{ (eval echo configure:2320: \"$ac_try\") 1>&5; (eval $ac_try) 2>&5; } +{ (eval echo configure:2348: \"$ac_try\") 1>&5; (eval $ac_try) 2>&5; } ac_err=`grep -v '^ *+' conftest.out | grep -v "^conftest.${ac_ext}\$"` if test -z "$ac_err"; then : @@ -2327,13 +2355,13 @@ else rm -rf conftest* CPP="${CC-cc} -nologo -E" cat > conftest.$ac_ext <<EOF -#line 2331 "configure" +#line 2359 "configure" #include "confdefs.h" #include <assert.h> Syntax Error EOF ac_try="$ac_cpp conftest.$ac_ext >/dev/null 2>conftest.out" -{ (eval echo configure:2337: \"$ac_try\") 1>&5; (eval $ac_try) 2>&5; } +{ (eval echo configure:2365: \"$ac_try\") 1>&5; (eval $ac_try) 2>&5; } ac_err=`grep -v '^ *+' conftest.out | grep -v "^conftest.${ac_ext}\$"` if test -z "$ac_err"; then : @@ -2358,7 +2386,7 @@ fi echo "$ac_t""$CPP" 1>&6 cat > conftest.$ac_ext <<EOF -#line 2362 "configure" +#line 2390 "configure" #include "confdefs.h" #include <stdint.h> EOF @@ -2373,7 +2401,7 @@ fi rm -f conftest* cat > conftest.$ac_ext <<EOF -#line 2377 "configure" +#line 2405 "configure" #include "confdefs.h" #include <inttypes.h> EOF @@ -2388,7 +2416,7 @@ fi rm -f conftest* cat > conftest.$ac_ext <<EOF -#line 2392 "configure" +#line 2420 "configure" #include "confdefs.h" #include <sys/types.h> EOF @@ -2403,7 +2431,7 @@ fi rm -f conftest* cat > conftest.$ac_ext <<EOF -#line 2407 "configure" +#line 2435 "configure" #include "confdefs.h" #include <sys/config.h> EOF @@ -2420,7 +2448,7 @@ rm -f conftest* cat > conftest.$ac_ext <<EOF -#line 2424 "configure" +#line 2452 "configure" #include "confdefs.h" #include <time.h> EOF @@ -2435,7 +2463,7 @@ fi rm -f conftest* cat > conftest.$ac_ext <<EOF -#line 2439 "configure" +#line 2467 "configure" #include "confdefs.h" #include <time.h> EOF @@ -2473,7 +2501,7 @@ ZLIBSPEC= libsubdir=.libs echo $ac_n "checking for garbage collector to use""... $ac_c" 1>&6 -echo "configure:2477: checking for garbage collector to use" >&5 +echo "configure:2505: checking for garbage collector to use" >&5 # Check whether --enable-java-gc or --disable-java-gc was given. if test "${enable_java_gc+set}" = set; then enableval="$enable_java_gc" @@ -2523,7 +2551,7 @@ esac echo $ac_n "checking for threads package to use""... $ac_c" 1>&6 -echo "configure:2527: checking for threads package to use" >&5 +echo "configure:2555: checking for threads package to use" >&5 # Check whether --enable-threads or --disable-threads was given. if test "${enable_threads+set}" = set; then enableval="$enable_threads" @@ -2715,12 +2743,12 @@ else for ac_func in strerror ioctl select open fsync sleep do echo $ac_n "checking for $ac_func""... $ac_c" 1>&6 -echo "configure:2719: checking for $ac_func" >&5 +echo "configure:2747: checking for $ac_func" >&5 if eval "test \"`echo '$''{'ac_cv_func_$ac_func'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 else cat > conftest.$ac_ext <<EOF -#line 2724 "configure" +#line 2752 "configure" #include "confdefs.h" /* System header to define __stub macros and hopefully few prototypes, which can conflict with char $ac_func(); below. */ @@ -2743,7 +2771,7 @@ $ac_func(); ; return 0; } EOF -if { (eval echo configure:2747: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext}; then +if { (eval echo configure:2775: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext}; then rm -rf conftest* eval "ac_cv_func_$ac_func=yes" else @@ -2770,12 +2798,12 @@ done for ac_func in ctime_r ctime do echo $ac_n "checking for $ac_func""... $ac_c" 1>&6 -echo "configure:2774: checking for $ac_func" >&5 +echo "configure:2802: checking for $ac_func" >&5 if eval "test \"`echo '$''{'ac_cv_func_$ac_func'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 else cat > conftest.$ac_ext <<EOF -#line 2779 "configure" +#line 2807 "configure" #include "confdefs.h" /* System header to define __stub macros and hopefully few prototypes, which can conflict with char $ac_func(); below. */ @@ -2798,7 +2826,7 @@ $ac_func(); ; return 0; } EOF -if { (eval echo configure:2802: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext}; then +if { (eval echo configure:2830: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext}; then rm -rf conftest* eval "ac_cv_func_$ac_func=yes" else @@ -2825,12 +2853,12 @@ done for ac_func in gmtime_r localtime_r readdir_r getpwuid_r do echo $ac_n "checking for $ac_func""... $ac_c" 1>&6 -echo "configure:2829: checking for $ac_func" >&5 +echo "configure:2857: checking for $ac_func" >&5 if eval "test \"`echo '$''{'ac_cv_func_$ac_func'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 else cat > conftest.$ac_ext <<EOF -#line 2834 "configure" +#line 2862 "configure" #include "confdefs.h" /* System header to define __stub macros and hopefully few prototypes, which can conflict with char $ac_func(); below. */ @@ -2853,7 +2881,7 @@ $ac_func(); ; return 0; } EOF -if { (eval echo configure:2857: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext}; then +if { (eval echo configure:2885: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext}; then rm -rf conftest* eval "ac_cv_func_$ac_func=yes" else @@ -2880,12 +2908,12 @@ done for ac_func in access stat mkdir rename rmdir unlink realpath do echo $ac_n "checking for $ac_func""... $ac_c" 1>&6 -echo "configure:2884: checking for $ac_func" >&5 +echo "configure:2912: checking for $ac_func" >&5 if eval "test \"`echo '$''{'ac_cv_func_$ac_func'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 else cat > conftest.$ac_ext <<EOF -#line 2889 "configure" +#line 2917 "configure" #include "confdefs.h" /* System header to define __stub macros and hopefully few prototypes, which can conflict with char $ac_func(); below. */ @@ -2908,7 +2936,7 @@ $ac_func(); ; return 0; } EOF -if { (eval echo configure:2912: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext}; then +if { (eval echo configure:2940: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext}; then rm -rf conftest* eval "ac_cv_func_$ac_func=yes" else @@ -2935,12 +2963,12 @@ done for ac_func in inet_aton inet_addr do echo $ac_n "checking for $ac_func""... $ac_c" 1>&6 -echo "configure:2939: checking for $ac_func" >&5 +echo "configure:2967: checking for $ac_func" >&5 if eval "test \"`echo '$''{'ac_cv_func_$ac_func'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 else cat > conftest.$ac_ext <<EOF -#line 2944 "configure" +#line 2972 "configure" #include "confdefs.h" /* System header to define __stub macros and hopefully few prototypes, which can conflict with char $ac_func(); below. */ @@ -2963,7 +2991,7 @@ $ac_func(); ; return 0; } EOF -if { (eval echo configure:2967: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext}; then +if { (eval echo configure:2995: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext}; then rm -rf conftest* eval "ac_cv_func_$ac_func=yes" else @@ -2990,12 +3018,12 @@ done for ac_func in inet_pton uname inet_ntoa do echo $ac_n "checking for $ac_func""... $ac_c" 1>&6 -echo "configure:2994: checking for $ac_func" >&5 +echo "configure:3022: checking for $ac_func" >&5 if eval "test \"`echo '$''{'ac_cv_func_$ac_func'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 else cat > conftest.$ac_ext <<EOF -#line 2999 "configure" +#line 3027 "configure" #include "confdefs.h" /* System header to define __stub macros and hopefully few prototypes, which can conflict with char $ac_func(); below. */ @@ -3018,7 +3046,7 @@ $ac_func(); ; return 0; } EOF -if { (eval echo configure:3022: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext}; then +if { (eval echo configure:3050: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext}; then rm -rf conftest* eval "ac_cv_func_$ac_func=yes" else @@ -3046,12 +3074,12 @@ done for ac_func in gethostbyname_r do echo $ac_n "checking for $ac_func""... $ac_c" 1>&6 -echo "configure:3050: checking for $ac_func" >&5 +echo "configure:3078: checking for $ac_func" >&5 if eval "test \"`echo '$''{'ac_cv_func_$ac_func'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 else cat > conftest.$ac_ext <<EOF -#line 3055 "configure" +#line 3083 "configure" #include "confdefs.h" /* System header to define __stub macros and hopefully few prototypes, which can conflict with char $ac_func(); below. */ @@ -3074,7 +3102,7 @@ $ac_func(); ; return 0; } EOF -if { (eval echo configure:3078: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext}; then +if { (eval echo configure:3106: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext}; then rm -rf conftest* eval "ac_cv_func_$ac_func=yes" else @@ -3101,7 +3129,7 @@ EOF # We look for the one that returns `int'. # Hopefully this check is robust enough. cat > conftest.$ac_ext <<EOF -#line 3105 "configure" +#line 3133 "configure" #include "confdefs.h" #include <netdb.h> EOF @@ -3121,7 +3149,7 @@ rm -f conftest* *" -D_REENTRANT "*) ;; *) echo $ac_n "checking whether gethostbyname_r declaration requires -D_REENTRANT""... $ac_c" 1>&6 -echo "configure:3125: checking whether gethostbyname_r declaration requires -D_REENTRANT" >&5 +echo "configure:3153: checking whether gethostbyname_r declaration requires -D_REENTRANT" >&5 if eval "test \"`echo '$''{'libjava_cv_gethostbyname_r_needs_reentrant'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 else @@ -3134,14 +3162,14 @@ ac_link='${CXX-g++} -o conftest${ac_exeext} $CXXFLAGS $CPPFLAGS $LDFLAGS conftes cross_compiling=$ac_cv_prog_cxx_cross cat > conftest.$ac_ext <<EOF -#line 3138 "configure" +#line 3166 "configure" #include "confdefs.h" #include <netdb.h> int main() { gethostbyname_r("", 0, 0); ; return 0; } EOF -if { (eval echo configure:3145: \"$ac_compile\") 1>&5; (eval $ac_compile) 2>&5; }; then +if { (eval echo configure:3173: \"$ac_compile\") 1>&5; (eval $ac_compile) 2>&5; }; then rm -rf conftest* libjava_cv_gethostbyname_r_needs_reentrant=no else @@ -3151,14 +3179,14 @@ else CPPFLAGS_SAVE="$CPPFLAGS" CPPFLAGS="$CPPFLAGS -D_REENTRANT" cat > conftest.$ac_ext <<EOF -#line 3155 "configure" +#line 3183 "configure" #include "confdefs.h" #include <netdb.h> int main() { gethostbyname_r("", 0, 0); ; return 0; } EOF -if { (eval echo configure:3162: \"$ac_compile\") 1>&5; (eval $ac_compile) 2>&5; }; then +if { (eval echo configure:3190: \"$ac_compile\") 1>&5; (eval $ac_compile) 2>&5; }; then rm -rf conftest* libjava_cv_gethostbyname_r_needs_reentrant=yes else @@ -3193,12 +3221,12 @@ EOF esac echo $ac_n "checking for struct hostent_data""... $ac_c" 1>&6 -echo "configure:3197: checking for struct hostent_data" >&5 +echo "configure:3225: checking for struct hostent_data" >&5 if eval "test \"`echo '$''{'libjava_cv_struct_hostent_data'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 else cat > conftest.$ac_ext <<EOF -#line 3202 "configure" +#line 3230 "configure" #include "confdefs.h" #if GETHOSTBYNAME_R_NEEDS_REENTRANT && !defined(_REENTRANT) @@ -3209,7 +3237,7 @@ int main() { struct hostent_data data; ; return 0; } EOF -if { (eval echo configure:3213: \"$ac_compile\") 1>&5; (eval $ac_compile) 2>&5; }; then +if { (eval echo configure:3241: \"$ac_compile\") 1>&5; (eval $ac_compile) 2>&5; }; then rm -rf conftest* libjava_cv_struct_hostent_data=yes else @@ -3238,12 +3266,12 @@ done for ac_func in gethostbyaddr_r do echo $ac_n "checking for $ac_func""... $ac_c" 1>&6 -echo "configure:3242: checking for $ac_func" >&5 +echo "configure:3270: checking for $ac_func" >&5 if eval "test \"`echo '$''{'ac_cv_func_$ac_func'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 else cat > conftest.$ac_ext <<EOF -#line 3247 "configure" +#line 3275 "configure" #include "confdefs.h" /* System header to define __stub macros and hopefully few prototypes, which can conflict with char $ac_func(); below. */ @@ -3266,7 +3294,7 @@ $ac_func(); ; return 0; } EOF -if { (eval echo configure:3270: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext}; then +if { (eval echo configure:3298: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext}; then rm -rf conftest* eval "ac_cv_func_$ac_func=yes" else @@ -3293,7 +3321,7 @@ EOF # We look for the one that returns `int'. # Hopefully this check is robust enough. cat > conftest.$ac_ext <<EOF -#line 3297 "configure" +#line 3325 "configure" #include "confdefs.h" #include <netdb.h> EOF @@ -3317,12 +3345,12 @@ done for ac_func in gethostname do echo $ac_n "checking for $ac_func""... $ac_c" 1>&6 -echo "configure:3321: checking for $ac_func" >&5 +echo "configure:3349: checking for $ac_func" >&5 if eval "test \"`echo '$''{'ac_cv_func_$ac_func'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 else cat > conftest.$ac_ext <<EOF -#line 3326 "configure" +#line 3354 "configure" #include "confdefs.h" /* System header to define __stub macros and hopefully few prototypes, which can conflict with char $ac_func(); below. */ @@ -3345,7 +3373,7 @@ $ac_func(); ; return 0; } EOF -if { (eval echo configure:3349: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext}; then +if { (eval echo configure:3377: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext}; then rm -rf conftest* eval "ac_cv_func_$ac_func=yes" else @@ -3369,7 +3397,7 @@ EOF EOF cat > conftest.$ac_ext <<EOF -#line 3373 "configure" +#line 3401 "configure" #include "confdefs.h" #include <unistd.h> EOF @@ -3396,12 +3424,12 @@ done for ac_func in pthread_mutexattr_settype pthread_mutexattr_setkind_np do echo $ac_n "checking for $ac_func""... $ac_c" 1>&6 -echo "configure:3400: checking for $ac_func" >&5 +echo "configure:3428: checking for $ac_func" >&5 if eval "test \"`echo '$''{'ac_cv_func_$ac_func'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 else cat > conftest.$ac_ext <<EOF -#line 3405 "configure" +#line 3433 "configure" #include "confdefs.h" /* System header to define __stub macros and hopefully few prototypes, which can conflict with char $ac_func(); below. */ @@ -3424,7 +3452,7 @@ $ac_func(); ; return 0; } EOF -if { (eval echo configure:3428: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext}; then +if { (eval echo configure:3456: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext}; then rm -rf conftest* eval "ac_cv_func_$ac_func=yes" else @@ -3454,12 +3482,12 @@ done for ac_func in sched_yield do echo $ac_n "checking for $ac_func""... $ac_c" 1>&6 -echo "configure:3458: checking for $ac_func" >&5 +echo "configure:3486: checking for $ac_func" >&5 if eval "test \"`echo '$''{'ac_cv_func_$ac_func'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 else cat > conftest.$ac_ext <<EOF -#line 3463 "configure" +#line 3491 "configure" #include "confdefs.h" /* System header to define __stub macros and hopefully few prototypes, which can conflict with char $ac_func(); below. */ @@ -3482,7 +3510,7 @@ $ac_func(); ; return 0; } EOF -if { (eval echo configure:3486: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext}; then +if { (eval echo configure:3514: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext}; then rm -rf conftest* eval "ac_cv_func_$ac_func=yes" else @@ -3504,7 +3532,7 @@ EOF else echo "$ac_t""no" 1>&6 echo $ac_n "checking for sched_yield in -lposix4""... $ac_c" 1>&6 -echo "configure:3508: checking for sched_yield in -lposix4" >&5 +echo "configure:3536: checking for sched_yield in -lposix4" >&5 ac_lib_var=`echo posix4'_'sched_yield | sed 'y%./+-%__p_%'` if eval "test \"`echo '$''{'ac_cv_lib_$ac_lib_var'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 @@ -3512,7 +3540,7 @@ else ac_save_LIBS="$LIBS" LIBS="-lposix4 $LIBS" cat > conftest.$ac_ext <<EOF -#line 3516 "configure" +#line 3544 "configure" #include "confdefs.h" /* Override any gcc2 internal prototype to avoid an error. */ /* We use char because int might match the return type of a gcc2 @@ -3523,7 +3551,7 @@ int main() { sched_yield() ; return 0; } EOF -if { (eval echo configure:3527: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext}; then +if { (eval echo configure:3555: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext}; then rm -rf conftest* eval "ac_cv_lib_$ac_lib_var=yes" else @@ -3548,7 +3576,7 @@ else echo "$ac_t""no" 1>&6 echo $ac_n "checking for sched_yield in -lrt""... $ac_c" 1>&6 -echo "configure:3552: checking for sched_yield in -lrt" >&5 +echo "configure:3580: checking for sched_yield in -lrt" >&5 ac_lib_var=`echo rt'_'sched_yield | sed 'y%./+-%__p_%'` if eval "test \"`echo '$''{'ac_cv_lib_$ac_lib_var'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 @@ -3556,7 +3584,7 @@ else ac_save_LIBS="$LIBS" LIBS="-lrt $LIBS" cat > conftest.$ac_ext <<EOF -#line 3560 "configure" +#line 3588 "configure" #include "confdefs.h" /* Override any gcc2 internal prototype to avoid an error. */ /* We use char because int might match the return type of a gcc2 @@ -3567,7 +3595,7 @@ int main() { sched_yield() ; return 0; } EOF -if { (eval echo configure:3571: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext}; then +if { (eval echo configure:3599: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext}; then rm -rf conftest* eval "ac_cv_lib_$ac_lib_var=yes" else @@ -3604,12 +3632,12 @@ done for ac_func in gettimeofday time ftime do echo $ac_n "checking for $ac_func""... $ac_c" 1>&6 -echo "configure:3608: checking for $ac_func" >&5 +echo "configure:3636: checking for $ac_func" >&5 if eval "test \"`echo '$''{'ac_cv_func_$ac_func'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 else cat > conftest.$ac_ext <<EOF -#line 3613 "configure" +#line 3641 "configure" #include "confdefs.h" /* System header to define __stub macros and hopefully few prototypes, which can conflict with char $ac_func(); below. */ @@ -3632,7 +3660,7 @@ $ac_func(); ; return 0; } EOF -if { (eval echo configure:3636: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext}; then +if { (eval echo configure:3664: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext}; then rm -rf conftest* eval "ac_cv_func_$ac_func=yes" else @@ -3663,12 +3691,12 @@ done for ac_func in memmove do echo $ac_n "checking for $ac_func""... $ac_c" 1>&6 -echo "configure:3667: checking for $ac_func" >&5 +echo "configure:3695: checking for $ac_func" >&5 if eval "test \"`echo '$''{'ac_cv_func_$ac_func'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 else cat > conftest.$ac_ext <<EOF -#line 3672 "configure" +#line 3700 "configure" #include "confdefs.h" /* System header to define __stub macros and hopefully few prototypes, which can conflict with char $ac_func(); below. */ @@ -3691,7 +3719,7 @@ $ac_func(); ; return 0; } EOF -if { (eval echo configure:3695: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext}; then +if { (eval echo configure:3723: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext}; then rm -rf conftest* eval "ac_cv_func_$ac_func=yes" else @@ -3721,12 +3749,12 @@ done for ac_func in memcpy do echo $ac_n "checking for $ac_func""... $ac_c" 1>&6 -echo "configure:3725: checking for $ac_func" >&5 +echo "configure:3753: checking for $ac_func" >&5 if eval "test \"`echo '$''{'ac_cv_func_$ac_func'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 else cat > conftest.$ac_ext <<EOF -#line 3730 "configure" +#line 3758 "configure" #include "confdefs.h" /* System header to define __stub macros and hopefully few prototypes, which can conflict with char $ac_func(); below. */ @@ -3749,7 +3777,7 @@ $ac_func(); ; return 0; } EOF -if { (eval echo configure:3753: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext}; then +if { (eval echo configure:3781: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext}; then rm -rf conftest* eval "ac_cv_func_$ac_func=yes" else @@ -3797,7 +3825,7 @@ done #-------------------------------------------------------------------- echo $ac_n "checking for socket libraries""... $ac_c" 1>&6 -echo "configure:3801: checking for socket libraries" >&5 +echo "configure:3829: checking for socket libraries" >&5 if eval "test \"`echo '$''{'gcj_cv_lib_sockets'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 else @@ -3805,12 +3833,12 @@ else gcj_checkBoth=0 unset ac_cv_func_connect echo $ac_n "checking for connect""... $ac_c" 1>&6 -echo "configure:3809: checking for connect" >&5 +echo "configure:3837: checking for connect" >&5 if eval "test \"`echo '$''{'ac_cv_func_connect'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 else cat > conftest.$ac_ext <<EOF -#line 3814 "configure" +#line 3842 "configure" #include "confdefs.h" /* System header to define __stub macros and hopefully few prototypes, which can conflict with char connect(); below. */ @@ -3833,7 +3861,7 @@ connect(); ; return 0; } EOF -if { (eval echo configure:3837: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext}; then +if { (eval echo configure:3865: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext}; then rm -rf conftest* eval "ac_cv_func_connect=yes" else @@ -3856,7 +3884,7 @@ fi if test "$gcj_checkSocket" = 1; then unset ac_cv_func_connect echo $ac_n "checking for main in -lsocket""... $ac_c" 1>&6 -echo "configure:3860: checking for main in -lsocket" >&5 +echo "configure:3888: checking for main in -lsocket" >&5 ac_lib_var=`echo socket'_'main | sed 'y%./+-%__p_%'` if eval "test \"`echo '$''{'ac_cv_lib_$ac_lib_var'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 @@ -3864,14 +3892,14 @@ else ac_save_LIBS="$LIBS" LIBS="-lsocket $LIBS" cat > conftest.$ac_ext <<EOF -#line 3868 "configure" +#line 3896 "configure" #include "confdefs.h" int main() { main() ; return 0; } EOF -if { (eval echo configure:3875: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext}; then +if { (eval echo configure:3903: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext}; then rm -rf conftest* eval "ac_cv_lib_$ac_lib_var=yes" else @@ -3898,12 +3926,12 @@ fi LIBS="$LIBS -lsocket -lnsl" unset ac_cv_func_accept echo $ac_n "checking for accept""... $ac_c" 1>&6 -echo "configure:3902: checking for accept" >&5 +echo "configure:3930: checking for accept" >&5 if eval "test \"`echo '$''{'ac_cv_func_accept'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 else cat > conftest.$ac_ext <<EOF -#line 3907 "configure" +#line 3935 "configure" #include "confdefs.h" /* System header to define __stub macros and hopefully few prototypes, which can conflict with char accept(); below. */ @@ -3926,7 +3954,7 @@ accept(); ; return 0; } EOF -if { (eval echo configure:3930: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext}; then +if { (eval echo configure:3958: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext}; then rm -rf conftest* eval "ac_cv_func_accept=yes" else @@ -3953,12 +3981,12 @@ fi gcj_oldLibs=$LIBS LIBS="$LIBS $gcj_cv_lib_sockets" echo $ac_n "checking for gethostbyname""... $ac_c" 1>&6 -echo "configure:3957: checking for gethostbyname" >&5 +echo "configure:3985: checking for gethostbyname" >&5 if eval "test \"`echo '$''{'ac_cv_func_gethostbyname'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 else cat > conftest.$ac_ext <<EOF -#line 3962 "configure" +#line 3990 "configure" #include "confdefs.h" /* System header to define __stub macros and hopefully few prototypes, which can conflict with char gethostbyname(); below. */ @@ -3981,7 +4009,7 @@ gethostbyname(); ; return 0; } EOF -if { (eval echo configure:3985: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext}; then +if { (eval echo configure:4013: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext}; then rm -rf conftest* eval "ac_cv_func_gethostbyname=yes" else @@ -3999,7 +4027,7 @@ if eval "test \"`echo '$ac_cv_func_'gethostbyname`\" = yes"; then else echo "$ac_t""no" 1>&6 echo $ac_n "checking for main in -lnsl""... $ac_c" 1>&6 -echo "configure:4003: checking for main in -lnsl" >&5 +echo "configure:4031: checking for main in -lnsl" >&5 ac_lib_var=`echo nsl'_'main | sed 'y%./+-%__p_%'` if eval "test \"`echo '$''{'ac_cv_lib_$ac_lib_var'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 @@ -4007,14 +4035,14 @@ else ac_save_LIBS="$LIBS" LIBS="-lnsl $LIBS" cat > conftest.$ac_ext <<EOF -#line 4011 "configure" +#line 4039 "configure" #include "confdefs.h" int main() { main() ; return 0; } EOF -if { (eval echo configure:4018: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext}; then +if { (eval echo configure:4046: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext}; then rm -rf conftest* eval "ac_cv_lib_$ac_lib_var=yes" else @@ -4044,9 +4072,13 @@ fi echo "$ac_t""$gcj_cv_lib_sockets" 1>&6 SYSTEMSPEC="$SYSTEMSPEC $gcj_cv_lib_sockets" + if test "$enable_interpreter" = yes; then + INTERPSPEC= + fi + if test "$with_system_zlib" = yes; then echo $ac_n "checking for deflate in -lz""... $ac_c" 1>&6 -echo "configure:4050: checking for deflate in -lz" >&5 +echo "configure:4082: checking for deflate in -lz" >&5 ac_lib_var=`echo z'_'deflate | sed 'y%./+-%__p_%'` if eval "test \"`echo '$''{'ac_cv_lib_$ac_lib_var'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 @@ -4054,7 +4086,7 @@ else ac_save_LIBS="$LIBS" LIBS="-lz $LIBS" cat > conftest.$ac_ext <<EOF -#line 4058 "configure" +#line 4090 "configure" #include "confdefs.h" /* Override any gcc2 internal prototype to avoid an error. */ /* We use char because int might match the return type of a gcc2 @@ -4065,7 +4097,7 @@ int main() { deflate() ; return 0; } EOF -if { (eval echo configure:4069: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext}; then +if { (eval echo configure:4101: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext}; then rm -rf conftest* eval "ac_cv_lib_$ac_lib_var=yes" else @@ -4094,7 +4126,7 @@ fi # requires -ldl. if test "$GC" = boehm; then echo $ac_n "checking for main in -ldl""... $ac_c" 1>&6 -echo "configure:4098: checking for main in -ldl" >&5 +echo "configure:4130: checking for main in -ldl" >&5 ac_lib_var=`echo dl'_'main | sed 'y%./+-%__p_%'` if eval "test \"`echo '$''{'ac_cv_lib_$ac_lib_var'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 @@ -4102,14 +4134,14 @@ else ac_save_LIBS="$LIBS" LIBS="-ldl $LIBS" cat > conftest.$ac_ext <<EOF -#line 4106 "configure" +#line 4138 "configure" #include "confdefs.h" int main() { main() ; return 0; } EOF -if { (eval echo configure:4113: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext}; then +if { (eval echo configure:4145: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext}; then rm -rf conftest* eval "ac_cv_lib_$ac_lib_var=yes" else @@ -4219,17 +4251,17 @@ for ac_hdr in unistd.h bstring.h sys/time.h sys/types.h fcntl.h sys/ioctl.h sys/ do ac_safe=`echo "$ac_hdr" | sed 'y%./+-%__p_%'` echo $ac_n "checking for $ac_hdr""... $ac_c" 1>&6 -echo "configure:4223: checking for $ac_hdr" >&5 +echo "configure:4255: checking for $ac_hdr" >&5 if eval "test \"`echo '$''{'ac_cv_header_$ac_safe'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 else cat > conftest.$ac_ext <<EOF -#line 4228 "configure" +#line 4260 "configure" #include "confdefs.h" #include <$ac_hdr> EOF ac_try="$ac_cpp conftest.$ac_ext >/dev/null 2>conftest.out" -{ (eval echo configure:4233: \"$ac_try\") 1>&5; (eval $ac_try) 2>&5; } +{ (eval echo configure:4265: \"$ac_try\") 1>&5; (eval $ac_try) 2>&5; } ac_err=`grep -v '^ *+' conftest.out | grep -v "^conftest.${ac_ext}\$"` if test -z "$ac_err"; then rm -rf conftest* @@ -4259,17 +4291,17 @@ for ac_hdr in dirent.h do ac_safe=`echo "$ac_hdr" | sed 'y%./+-%__p_%'` echo $ac_n "checking for $ac_hdr""... $ac_c" 1>&6 -echo "configure:4263: checking for $ac_hdr" >&5 +echo "configure:4295: checking for $ac_hdr" >&5 if eval "test \"`echo '$''{'ac_cv_header_$ac_safe'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 else cat > conftest.$ac_ext <<EOF -#line 4268 "configure" +#line 4300 "configure" #include "confdefs.h" #include <$ac_hdr> EOF ac_try="$ac_cpp conftest.$ac_ext >/dev/null 2>conftest.out" -{ (eval echo configure:4273: \"$ac_try\") 1>&5; (eval $ac_try) 2>&5; } +{ (eval echo configure:4305: \"$ac_try\") 1>&5; (eval $ac_try) 2>&5; } ac_err=`grep -v '^ *+' conftest.out | grep -v "^conftest.${ac_ext}\$"` if test -z "$ac_err"; then rm -rf conftest* @@ -4297,16 +4329,16 @@ done echo $ac_n "checking whether struct sockaddr_in6 is in netinet/in.h""... $ac_c" 1>&6 -echo "configure:4301: checking whether struct sockaddr_in6 is in netinet/in.h" >&5 +echo "configure:4333: checking whether struct sockaddr_in6 is in netinet/in.h" >&5 cat > conftest.$ac_ext <<EOF -#line 4303 "configure" +#line 4335 "configure" #include "confdefs.h" #include <netinet/in.h> int main() { struct sockaddr_in6 addr6; ; return 0; } EOF -if { (eval echo configure:4310: \"$ac_compile\") 1>&5; (eval $ac_compile) 2>&5; }; then +if { (eval echo configure:4342: \"$ac_compile\") 1>&5; (eval $ac_compile) 2>&5; }; then rm -rf conftest* cat >> confdefs.h <<\EOF #define HAVE_INET6 1 @@ -4322,16 +4354,16 @@ fi rm -f conftest* echo $ac_n "checking for socklen_t in sys/socket.h""... $ac_c" 1>&6 -echo "configure:4326: checking for socklen_t in sys/socket.h" >&5 +echo "configure:4358: checking for socklen_t in sys/socket.h" >&5 cat > conftest.$ac_ext <<EOF -#line 4328 "configure" +#line 4360 "configure" #include "confdefs.h" #include <sys/socket.h> int main() { socklen_t x = 5; ; return 0; } EOF -if { (eval echo configure:4335: \"$ac_compile\") 1>&5; (eval $ac_compile) 2>&5; }; then +if { (eval echo configure:4367: \"$ac_compile\") 1>&5; (eval $ac_compile) 2>&5; }; then rm -rf conftest* cat >> confdefs.h <<\EOF #define HAVE_SOCKLEN_T 1 @@ -4347,16 +4379,16 @@ fi rm -f conftest* echo $ac_n "checking for tm_gmtoff in struct tm""... $ac_c" 1>&6 -echo "configure:4351: checking for tm_gmtoff in struct tm" >&5 +echo "configure:4383: checking for tm_gmtoff in struct tm" >&5 cat > conftest.$ac_ext <<EOF -#line 4353 "configure" +#line 4385 "configure" #include "confdefs.h" #include <time.h> int main() { struct tm tim; tim.tm_gmtoff = 0; ; return 0; } EOF -if { (eval echo configure:4360: \"$ac_compile\") 1>&5; (eval $ac_compile) 2>&5; }; then +if { (eval echo configure:4392: \"$ac_compile\") 1>&5; (eval $ac_compile) 2>&5; }; then rm -rf conftest* cat >> confdefs.h <<\EOF #define STRUCT_TM_HAS_GMTOFF 1 @@ -4369,16 +4401,16 @@ else rm -rf conftest* echo "$ac_t""no" 1>&6 echo $ac_n "checking for global timezone variable""... $ac_c" 1>&6 -echo "configure:4373: checking for global timezone variable" >&5 +echo "configure:4405: checking for global timezone variable" >&5 cat > conftest.$ac_ext <<EOF -#line 4375 "configure" +#line 4407 "configure" #include "confdefs.h" #include <time.h> int main() { long z2 = timezone; ; return 0; } EOF -if { (eval echo configure:4382: \"$ac_compile\") 1>&5; (eval $ac_compile) 2>&5; }; then +if { (eval echo configure:4414: \"$ac_compile\") 1>&5; (eval $ac_compile) 2>&5; }; then rm -rf conftest* cat >> confdefs.h <<\EOF #define HAVE_TIMEZONE 1 @@ -4398,19 +4430,19 @@ rm -f conftest* # The Ultrix 4.2 mips builtin alloca declared by alloca.h only works # for constant arguments. Useless! echo $ac_n "checking for working alloca.h""... $ac_c" 1>&6 -echo "configure:4402: checking for working alloca.h" >&5 +echo "configure:4434: checking for working alloca.h" >&5 if eval "test \"`echo '$''{'ac_cv_header_alloca_h'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 else cat > conftest.$ac_ext <<EOF -#line 4407 "configure" +#line 4439 "configure" #include "confdefs.h" #include <alloca.h> int main() { char *p = alloca(2 * sizeof(int)); ; return 0; } EOF -if { (eval echo configure:4414: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext}; then +if { (eval echo configure:4446: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext}; then rm -rf conftest* ac_cv_header_alloca_h=yes else @@ -4431,12 +4463,12 @@ EOF fi echo $ac_n "checking for alloca""... $ac_c" 1>&6 -echo "configure:4435: checking for alloca" >&5 +echo "configure:4467: checking for alloca" >&5 if eval "test \"`echo '$''{'ac_cv_func_alloca_works'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 else cat > conftest.$ac_ext <<EOF -#line 4440 "configure" +#line 4472 "configure" #include "confdefs.h" #ifdef __GNUC__ @@ -4464,7 +4496,7 @@ int main() { char *p = (char *) alloca(1); ; return 0; } EOF -if { (eval echo configure:4468: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext}; then +if { (eval echo configure:4500: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext}; then rm -rf conftest* ac_cv_func_alloca_works=yes else @@ -4496,12 +4528,12 @@ EOF echo $ac_n "checking whether alloca needs Cray hooks""... $ac_c" 1>&6 -echo "configure:4500: checking whether alloca needs Cray hooks" >&5 +echo "configure:4532: checking whether alloca needs Cray hooks" >&5 if eval "test \"`echo '$''{'ac_cv_os_cray'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 else cat > conftest.$ac_ext <<EOF -#line 4505 "configure" +#line 4537 "configure" #include "confdefs.h" #if defined(CRAY) && ! defined(CRAY2) webecray @@ -4526,12 +4558,12 @@ echo "$ac_t""$ac_cv_os_cray" 1>&6 if test $ac_cv_os_cray = yes; then for ac_func in _getb67 GETB67 getb67; do echo $ac_n "checking for $ac_func""... $ac_c" 1>&6 -echo "configure:4530: checking for $ac_func" >&5 +echo "configure:4562: checking for $ac_func" >&5 if eval "test \"`echo '$''{'ac_cv_func_$ac_func'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 else cat > conftest.$ac_ext <<EOF -#line 4535 "configure" +#line 4567 "configure" #include "confdefs.h" /* System header to define __stub macros and hopefully few prototypes, which can conflict with char $ac_func(); below. */ @@ -4554,7 +4586,7 @@ $ac_func(); ; return 0; } EOF -if { (eval echo configure:4558: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext}; then +if { (eval echo configure:4590: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext}; then rm -rf conftest* eval "ac_cv_func_$ac_func=yes" else @@ -4581,7 +4613,7 @@ done fi echo $ac_n "checking stack direction for C alloca""... $ac_c" 1>&6 -echo "configure:4585: checking stack direction for C alloca" >&5 +echo "configure:4617: checking stack direction for C alloca" >&5 if eval "test \"`echo '$''{'ac_cv_c_stack_direction'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 else @@ -4589,7 +4621,7 @@ else ac_cv_c_stack_direction=0 else cat > conftest.$ac_ext <<EOF -#line 4593 "configure" +#line 4625 "configure" #include "confdefs.h" find_stack_direction () { @@ -4608,7 +4640,7 @@ main () exit (find_stack_direction() < 0); } EOF -if { (eval echo configure:4612: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext} && (./conftest; exit) 2>/dev/null +if { (eval echo configure:4644: \"$ac_link\") 1>&5; (eval $ac_link) 2>&5; } && test -s conftest${ac_exeext} && (./conftest; exit) 2>/dev/null then ac_cv_c_stack_direction=1 else @@ -4635,7 +4667,7 @@ do # Extract the first word of "$ac_prog", so it can be a program name with args. set dummy $ac_prog; ac_word=$2 echo $ac_n "checking for $ac_word""... $ac_c" 1>&6 -echo "configure:4639: checking for $ac_word" >&5 +echo "configure:4671: checking for $ac_word" >&5 if eval "test \"`echo '$''{'ac_cv_prog_PERL'+set}'`\" = set"; then echo $ac_n "(cached) $ac_c" 1>&6 else @@ -4868,6 +4900,7 @@ s%@LIBTOOL@%$LIBTOOL%g s%@COMPPATH@%$COMPPATH%g s%@TESTSUBDIR_TRUE@%$TESTSUBDIR_TRUE%g s%@TESTSUBDIR_FALSE@%$TESTSUBDIR_FALSE%g +s%@INTERPSPEC@%$INTERPSPEC%g s%@CPP@%$CPP%g s%@SYSTEMSPEC@%$SYSTEMSPEC%g s%@ZLIBSPEC@%$ZLIBSPEC%g diff --git a/libjava/configure.in b/libjava/configure.in index 4549889467d..861530a2e49 100644 --- a/libjava/configure.in +++ b/libjava/configure.in @@ -42,6 +42,17 @@ AC_ARG_ENABLE(libgcj-debug, AC_DEFINE(DEBUG) fi) +dnl See if the user has the enterpreter included. +AC_ARG_ENABLE(interpreter, +[ --enable-interpreter Enable interpreter], + if test "$enable_interpreter" = yes; then + AC_DEFINE(INTERPRETER) + fi) + +dnl This becomes -lffi if the interpreter is enables +INTERPSPEC= +AC_SUBST(INTERPSPEC) + dnl If the target is an eCos system, use the appropriate eCos dnl I/O routines. dnl FIXME: this should not be a local option but a global target @@ -447,6 +458,10 @@ else ]) SYSTEMSPEC="$SYSTEMSPEC $gcj_cv_lib_sockets" + if test "$enable_interpreter" = yes; then + INTERPSPEC= + fi + if test "$with_system_zlib" = yes; then AC_CHECK_LIB(z, deflate, ZLIBSPEC=-lz, ZLIBSPEC=-lzgcj) else diff --git a/libjava/defineclass.cc b/libjava/defineclass.cc new file mode 100644 index 00000000000..09f8f47fc84 --- /dev/null +++ b/libjava/defineclass.cc @@ -0,0 +1,1556 @@ +// defineclass.cc - defining a class from .class format. + +/* Copyright (C) 1999 Cygnus Solutions + + This file is part of libgcj. + +This software is copyrighted work licensed under the terms of the +Libgcj License. Please consult the file "LIBGCJ_LICENSE" for +details. */ + +/* + Author: Kresten Krab Thorup <krab@gnu.org> + + Written using the online versions of Java Language Specification (1st + ed.) and The Java Virtual Machine Specification (2nd ed.). + + Future work may include reading (and handling) attributes which are + currently being ignored ("InnerClasses", "LineNumber", etc...). +*/ + +#include <java-interp.h> + +#ifdef INTERPRETER + +#include <java-cpool.h> +#include <cni.h> + +#include <java/lang/Class.h> +#include <java/lang/Float.h> +#include <java/lang/Double.h> +#include <java/lang/Character.h> +#include <java/lang/LinkageError.h> +#include <java/lang/InternalError.h> +#include <java/lang/ClassFormatError.h> +#include <java/lang/NoClassDefFoundError.h> +#include <java/lang/ClassCircularityError.h> +#include <java/lang/ClassNotFoundException.h> +#include <java/lang/IncompatibleClassChangeError.h> + +#define ClassClass _CL_Q34java4lang5Class +extern java::lang::Class ClassClass; +#define StringClass _CL_Q34java4lang6String +extern java::lang::Class StringClass; +#define ClassObject _CL_Q34java4lang6Object +extern java::lang::Class ClassObject; + +// we don't verify method names that match these. +static _Jv_Utf8Const *clinit_name = _Jv_makeUtf8Const ("<clinit>", 8); +static _Jv_Utf8Const *init_name = _Jv_makeUtf8Const ("<init>", 6); + + +// these go in some seperate functions, to avoid having _Jv_InitClass +// inserted all over the place. +static void throw_internal_error (char *msg) + __attribute__ ((__noreturn__)); +static void throw_no_class_def_found_error (jstring msg) + __attribute__ ((__noreturn__)); +static void throw_no_class_def_found_error (char *msg) + __attribute__ ((__noreturn__)); +static void throw_class_format_error (jstring msg) + __attribute__ ((__noreturn__)); +static void throw_class_format_error (char *msg) + __attribute__ ((__noreturn__)); +static void throw_incompatible_class_change_error (jstring msg) + __attribute__ ((__noreturn__)); +static void throw_class_circularity_error (jstring msg) + __attribute__ ((__noreturn__)); + +static jdouble long_bits_to_double (jlong); +static jfloat int_bits_to_float (jint); + +/** + * We define class reading using a class. It is practical, since then + * the entire class-reader can be a friend of class Class (it needs to + * write all it's different structures); but also because this makes it + * easy to make class definition reentrant, and thus two threads can be + * defining classes at the same time. This class (_Jv_ClassReader) is + * never exposed outside this file, so we don't have to worry about + * public or private members here. + */ + +struct _Jv_ClassReader { + + // do verification? Currently, there is no option to disable this. + // This flag just controls the verificaiton done by the class loader; + // i.e., checking the integrity of the constant pool; and it is + // allways on. You always want this as far as I can see, but it also + // controls weither identifiers and type descriptors/signatures are + // verified as legal. This could be somewhat more expensive since it + // will call Characher.isJavaIdentifier{Start,Part} for each character + // in any identifier (field name or method name) it comes by. Thus, + // it might be useful to turn off this verification for classes that + // come from a trusted source. However, for GCJ, trusted classes are + // most likely to be linked in. + + bool verify; + + // input data. + unsigned char *bytes; + int len; + + // current input position + int pos; + + // the constant pool data + int pool_count; + unsigned char *tags; + unsigned int *offsets; + + // the class to define (see java-interp.h) + _Jv_InterpClass *def; + + /* check that the given number of input bytes are available */ + inline void check (int num) + { + if (pos + num > len) + throw_class_format_error ("Premature end of data"); + } + + /* skip a given number of bytes in input */ + inline void skip (int num) + { + check (num); + pos += num; + } + + /* read an unsignend 1-byte unit */ + inline static jint get1u (unsigned char* bytes) + { + return bytes[0]; + } + + /* read an unsigned 1-byte unit */ + inline jint read1u () + { + skip (1); + return get1u (bytes+pos-1); + } + + /* read an unsigned 2-byte unit */ + inline static jint get2u (unsigned char *bytes) + { + return (((jint)bytes[0]) << 8) | ((jint)bytes[1]); + } + + /* read an unsigned 2-byte unit */ + inline jint read2u () + { + skip (2); + return get2u (bytes+pos-2); + } + + /* read a 4-byte unit */ + static jint get4 (unsigned char *bytes) + { + return (((jint)bytes[0]) << 24) + | (((jint)bytes[1]) << 16) + | (((jint)bytes[2]) << 8) + | (((jint)bytes[3]) << 0); + } + + /* read a 4-byte unit, (we don't do that quite so often) */ + inline jint read4 () + { + skip (4); + return get4 (bytes+pos-4); + } + + /* read a 8-byte unit */ + static jlong get8 (unsigned char* bytes) + { + return (((jlong)bytes[0]) << 56) + | (((jlong)bytes[1]) << 48) + | (((jlong)bytes[2]) << 40) + | (((jlong)bytes[3]) << 32) + | (((jlong)bytes[4]) << 24) + | (((jlong)bytes[5]) << 16) + | (((jlong)bytes[6]) << 8) + | (((jlong)bytes[7]) << 0); + } + + /* read a 8-byte unit */ + inline jlong read8 () + { + skip (8); + return get8 (bytes+pos-8); + } + + inline void check_tag (int index, char expected_tag) + { + if (index < 0 + || index > pool_count + || tags[index] != expected_tag) + throw_class_format_error ("erroneous constant pool tag"); + } + + _Jv_ClassReader (jclass klass, jbyteArray data, jint offset, jint length) + { + if (klass == 0 || length < 0 || offset+length > data->length) + throw_internal_error ("arguments to _Jv_DefineClass"); + + verify = true; + bytes = (unsigned char*) (elements (data)+offset); + len = length; + pos = 0; + def = (_Jv_InterpClass*) klass; + } + + /** and here goes the parser members defined out-of-line */ + void parse (); + void read_constpool (); + void prepare_pool_entry (int index, unsigned char tag); + void read_fields (); + void read_methods (); + void read_one_class_attribute (); + void read_one_method_attribute (int method); + void read_one_code_attribute (int method); + void read_one_field_attribute (int field); + + /** check an utf8 entry, without creating a Utf8Const object */ + bool is_attribute_name (int index, char *name); + + /** here goes the class-loader members defined out-of-line */ + void handleConstantPool (); + void handleClassBegin (int, int, int); + void handleInterfacesBegin (int); + void handleInterface (int, int); + void handleFieldsBegin (int); + void handleField (int, int, int, int); + void handleFieldsEnd (); + void handleConstantValueAttribute (int,int); + void handleMethodsBegin (int); + void handleMethod (int, int, int, int); + void handleMethodsEnd (); + void handleCodeAttribute (int, int, int, int, int, int); + void handleExceptionTableEntry (int, int, int, int, int, int); + + void checkExtends (jclass sub, jclass super); + void checkImplements (jclass sub, jclass super); + + /* + * FIXME: we should keep a hash table of utf8-strings, since many will + * be the same. It's a little tricky, however, because the hash table + * needs to interact gracefully with the garbage collector. Much + * memory is to be saved by this, however! perhaps the improvement + * could be implemented in prims.cc (_Jv_makeUtf8Const), since it + * computes the hash value anyway. + */ + + static const int PUBLIC = 0x001; + static const int PRIVATE = 0x002; + static const int PROTECTED = 0x004; + static const int STATIC = 0x008; + static const int FINAL = 0x010; + static const int SYNCHRONIZED = 0x020; + static const int VOLATILE = 0x040; + static const int TRANSIENT = 0x080; + static const int NATIVE = 0x100; + static const int INTERFACE = 0x200; + static const int ABSTRACT = 0x400; + static const int ALL_FLAGS = 0x7FF; + +}; + +/* This is used for the isJavaIdentifierStart & isJavaIdentifierPart + methods, so we avoid doing _Jv_InitClass all the time */ + +static const java::lang::Character *character = 0; +static void prepare_character (); + +void +_Jv_DefineClass (jclass klass, jbyteArray data, jint offset, jint length) +{ + if (character == 0) + prepare_character (); + + _Jv_ClassReader reader (klass, data, offset, length); + reader.parse(); + + /* that's it! */ +} + +/** put it after _Jv_DefineClass, so it doesn't get inlined */ +static void prepare_character () +{ + character = new java::lang::Character ('!'); +} + + +/** This section defines the parsing/scanning of the class data */ + +void +_Jv_ClassReader::parse () +{ + int magic = read4 (); + + /* FIXME: Decide which range of version numbers to allow */ + + /* int minor_version = */ read2u (); + /* int major_verson = */ read2u (); + + if (magic != (int) 0xCAFEBABE) + throw_class_format_error ("bad magic number"); + + pool_count = read2u (); + + read_constpool (); + + int access_flags = read2u (); + int this_class = read2u (); + int super_class = read2u (); + + check_tag (this_class, JV_CONSTANT_Class); + if (super_class != 0) + check_tag (super_class, JV_CONSTANT_Class); + + handleClassBegin (access_flags, this_class, super_class); + + int interfaces_count = read2u (); + + handleInterfacesBegin (interfaces_count); + + for (int i = 0; i < interfaces_count; i++) + { + int iface = read2u (); + check_tag (iface, JV_CONSTANT_Class); + handleInterface (i, iface); + } + + read_fields (); + read_methods (); + + int attributes_count = read2u (); + + for (int i = 0; i < attributes_count; i++) + { + read_one_class_attribute (); + } + + if (pos != len) + throw_class_format_error ("unused data before end of file"); + + // tell everyone we're done. + def->state = JV_STATE_LOADED; + def->notifyAll (); + +} + +void _Jv_ClassReader::read_constpool () +{ + tags = (unsigned char*) _Jv_AllocBytesChecked (pool_count); + offsets = (unsigned int *) _Jv_AllocBytesChecked (sizeof (int) + * pool_count) ; + + /** first, we scan the constant pool, collecting tags and offsets */ + tags[0] = JV_CONSTANT_Undefined; + offsets[0] = pos; + for (int c = 1; c < pool_count; c++) + { + tags[c] = read1u (); + offsets[c] = pos; + + switch (tags[c]) + { + case JV_CONSTANT_String: + case JV_CONSTANT_Class: + skip (2); + break; + + case JV_CONSTANT_Fieldref: + case JV_CONSTANT_Methodref: + case JV_CONSTANT_InterfaceMethodref: + case JV_CONSTANT_NameAndType: + case JV_CONSTANT_Integer: + case JV_CONSTANT_Float: + skip (4); + break; + + case JV_CONSTANT_Double: + case JV_CONSTANT_Long: + skip (8); + tags[++c] = JV_CONSTANT_Undefined; + break; + + case JV_CONSTANT_Utf8: + { + int len = read2u (); + skip (len); + } + break; + + case JV_CONSTANT_Unicode: + throw_class_format_error ("unicode not supported"); + break; + + default: + throw_class_format_error ("erroneous constant pool tag"); + } + } + + handleConstantPool (); +} + + +void _Jv_ClassReader::read_fields () +{ + int fields_count = read2u (); + handleFieldsBegin (fields_count); + + for (int i = 0; i < fields_count; i++) + { + int access_flags = read2u (); + int name_index = read2u (); + int descriptor_index = read2u (); + int attributes_count = read2u (); + + check_tag (name_index, JV_CONSTANT_Utf8); + prepare_pool_entry (name_index, JV_CONSTANT_Utf8); + + check_tag (descriptor_index, JV_CONSTANT_Utf8); + prepare_pool_entry (descriptor_index, JV_CONSTANT_Utf8); + + handleField (i, access_flags, name_index, descriptor_index); + + for (int j = 0; j < attributes_count; j++) + { + read_one_field_attribute (i); + } + } + + handleFieldsEnd (); +} + +bool +_Jv_ClassReader::is_attribute_name (int index, char *name) +{ + check_tag (index, JV_CONSTANT_Utf8); + int len = get2u (bytes+offsets[index]); + if (len != (int) strlen (name)) + return false; + else + return !memcmp (bytes+offsets[index]+2, name, len); +} + +void _Jv_ClassReader::read_one_field_attribute (int field_index) +{ + int name = read2u (); + int length = read4 (); + + if (is_attribute_name (name, "ConstantValue")) + { + int cv = read2u (); + + if (cv < pool_count + && cv > 0 + && (tags[cv] == JV_CONSTANT_Integer + || tags[cv] == JV_CONSTANT_Float + || tags[cv] == JV_CONSTANT_Long + || tags[cv] == JV_CONSTANT_Double + || tags[cv] == JV_CONSTANT_String)) + { + handleConstantValueAttribute (field_index, cv); + } + else + { + throw_class_format_error ("erroneous ConstantValue attribute"); + } + + if (length != 2) + throw_class_format_error ("erroneous ConstantValue attribute"); + } + + else + { + skip (length); + } +} + +void _Jv_ClassReader::read_methods () +{ + int methods_count = read2u (); + + handleMethodsBegin (methods_count); + + for (int i = 0; i < methods_count; i++) + { + int access_flags = read2u (); + int name_index = read2u (); + int descriptor_index = read2u (); + int attributes_count = read2u (); + + check_tag (name_index, JV_CONSTANT_Utf8); + prepare_pool_entry (descriptor_index, JV_CONSTANT_Utf8); + + check_tag (name_index, JV_CONSTANT_Utf8); + prepare_pool_entry (descriptor_index, JV_CONSTANT_Utf8); + + handleMethod (i, access_flags, name_index, + descriptor_index); + + for (int j = 0; j < attributes_count; j++) + { + read_one_method_attribute (i); + } + } + + handleMethodsEnd (); +} + +void _Jv_ClassReader::read_one_method_attribute (int method_index) +{ + int name = read2u (); + int length = read4 (); + + if (is_attribute_name (name, "Exceptions")) + { + /* we ignore this for now */ + skip (length); + } + + else if (is_attribute_name (name, "Code")) + { + int start_off = pos; + int max_stack = read2u (); + int max_locals = read2u (); + int code_length = read4 (); + + int code_start = pos; + skip (code_length); + int exception_table_length = read2u (); + + handleCodeAttribute (method_index, + max_stack, max_locals, + code_start, code_length, + exception_table_length); + + + for (int i = 0; i < exception_table_length; i++) + { + int start_pc = read2u (); + int end_pc = read2u (); + int handler_pc = read2u (); + int catch_type = read2u (); + + if (start_pc > end_pc + || start_pc < 0 + || end_pc >= code_length + || handler_pc >= code_length) + throw_class_format_error ("erroneous exception handler info"); + + if (! (tags[catch_type] == JV_CONSTANT_Class + || tags[catch_type] == 0)) + { + throw_class_format_error ("erroneous exception handler info"); + } + + handleExceptionTableEntry (method_index, + i, + start_pc, + end_pc, + handler_pc, + catch_type); + + } + + int attributes_count = read2u (); + + for (int i = 0; i < attributes_count; i++) + { + read_one_code_attribute (method_index); + } + + if ((pos - start_off) != length) + throw_class_format_error ("code attribute too short"); + } + + else + { + /* ignore unknown attributes */ + skip (length); + } +} + +void _Jv_ClassReader::read_one_code_attribute (int /*method*/) +{ + /* ignore for now, ... later we may want to pick up + line number information, for debugging purposes; + in fact, the whole debugger issue is open! */ + + /* int name = */ read2u (); + int length = read4 (); + skip (length); + +} + +void _Jv_ClassReader::read_one_class_attribute () +{ + /* we also ignore the class attributes, ... + some day we'll add inner-classes support. */ + + /* int name = */ read2u (); + int length = read4 (); + skip (length); +} + + + + +/* this section defines the semantic actions of the parser */ + +void _Jv_ClassReader::handleConstantPool () +{ + /** now, we actually define the class' constant pool */ + + // the pool is scanned explicitly by the collector + jbyte *pool_tags = (jbyte*) _Jv_AllocBytesChecked (pool_count); + void **pool_data = (void**) _Jv_AllocBytesChecked (pool_count * sizeof (void*)); + + def->constants.tags = pool_tags; + def->constants.data = pool_data; + def->constants.size = pool_count; + + // Here we make a pass to collect the strings! We do this, because + // internally in the GCJ runtime, classes are encoded with .'s not /'s. + // Therefore, we first collect the strings, and then translate the rest + // of the utf8-entries (thus not representing strings) from /-notation + // to .-notation. + for (int i = 1; i < pool_count; i++) + { + if (tags[i] == JV_CONSTANT_String) + { + unsigned char* str_data = bytes + offsets [i]; + int utf_index = get2u (str_data); + check_tag (utf_index, JV_CONSTANT_Utf8); + unsigned char *utf_data = bytes + offsets[utf_index]; + int len = get2u (utf_data); + pool_data[i] = (void*)_Jv_makeUtf8Const ((char*)(utf_data+2), len); + pool_tags[i] = JV_CONSTANT_String; + } + else + { + pool_tags[i] = JV_CONSTANT_Undefined; + } + } + + // and now, we scan everything else but strings & utf8-entries. This + // leaves out those utf8-entries which are not used; which will be left + // with a tag of JV_CONSTANT_Undefined in the class definition. + for (int index = 1; index < pool_count; index++) + { + switch (tags[index]) + { + case JV_CONSTANT_Undefined: + case JV_CONSTANT_String: + case JV_CONSTANT_Utf8: + continue; + + default: + prepare_pool_entry (index, tags[index]); + } + } + +} + +/* this is a recursive procedure, which will prepare pool entries as needed. + Which is how we avoid initializing those entries which go unused. */ +void +_Jv_ClassReader::prepare_pool_entry (int index, unsigned char this_tag) +{ + /* these two, pool_data and pool_tags, point into the class + structure we are currently defining */ + + unsigned char *pool_tags = (unsigned char*) def->constants.tags; + void **pool_data = (void**) def->constants.data; + + /* this entry was already prepared */ + if (pool_tags[index] == this_tag) + return; + + /* this_data points to the constant-pool information for the current + constant-pool entry */ + + unsigned char *this_data = bytes + offsets[index]; + + switch (this_tag) + { + case JV_CONSTANT_Utf8: + { + // If we came here, it is because some other tag needs this + // utf8-entry for type information! Thus, we translate /'s to .'s in + // order to accomondate gcj's internal representation. + + int len = get2u (this_data); + char *buffer = (char*) alloca (len); + char *s = ((char*) this_data)+2; + + /* FIXME: avoid using a buffer here */ + for (int i = 0; i < len; i++) + { + if (s[i] == '/') + buffer[i] = '.'; + else + buffer[i] = (char) s[i]; + } + + pool_data[index] = (void*)_Jv_makeUtf8Const (buffer, len); + pool_tags[index] = JV_CONSTANT_Utf8; + } + break; + + case JV_CONSTANT_Class: + { + int utf_index = get2u (this_data); + check_tag (utf_index, JV_CONSTANT_Utf8); + prepare_pool_entry (utf_index, JV_CONSTANT_Utf8); + + if (verify) + _Jv_VerifyClassName ((_Jv_Utf8Const*)pool_data[utf_index]); + + pool_data[index] = pool_data[utf_index]; + pool_tags[index] = JV_CONSTANT_Class; + } + break; + + case JV_CONSTANT_String: + // already handled before... + break; + + case JV_CONSTANT_Fieldref: + case JV_CONSTANT_Methodref: + case JV_CONSTANT_InterfaceMethodref: + { + int class_index = get2u (this_data); + int nat_index = get2u (this_data+2); + + check_tag (class_index, JV_CONSTANT_Class); + prepare_pool_entry (class_index, JV_CONSTANT_Class); + + check_tag (nat_index, JV_CONSTANT_NameAndType); + prepare_pool_entry (nat_index, JV_CONSTANT_NameAndType); + + // here, verify the signature and identifier name + if (verify) + { + _Jv_ushort name_index, type_index; + _Jv_loadIndexes ((const void**)&pool_data[nat_index], + name_index, type_index); + + if (this_tag == JV_CONSTANT_Fieldref) + _Jv_VerifyFieldSignature + ((_Jv_Utf8Const*)pool_data[type_index]); + else + _Jv_VerifyMethodSignature + ((_Jv_Utf8Const*)pool_data[type_index]); + + _Jv_Utf8Const* name = (_Jv_Utf8Const*)pool_data[name_index]; + + if (this_tag != JV_CONSTANT_Fieldref + && ( _Jv_equalUtf8Consts (name, clinit_name) + || _Jv_equalUtf8Consts (name, init_name))) + /* ignore */; + else + _Jv_VerifyIdentifier ((_Jv_Utf8Const*)pool_data[name_index]); + } + + _Jv_storeIndexes (&pool_data[index], class_index, nat_index); + pool_tags[index] = this_tag; + } + break; + + case JV_CONSTANT_NameAndType: + { + _Jv_ushort name_index = get2u (this_data); + _Jv_ushort type_index = get2u (this_data+2); + + check_tag (name_index, JV_CONSTANT_Utf8); + prepare_pool_entry (name_index, JV_CONSTANT_Utf8); + + check_tag (type_index, JV_CONSTANT_Utf8); + prepare_pool_entry (type_index, JV_CONSTANT_Utf8); + + _Jv_storeIndexes (&pool_data[index], name_index, type_index); + pool_tags[index] = JV_CONSTANT_NameAndType; + } + break; + + case JV_CONSTANT_Float: + { + jfloat f = int_bits_to_float ((jint) get4 (this_data)); + _Jv_storeFloat (&pool_data[index], f); + pool_tags[index] = JV_CONSTANT_Float; + } + break; + + case JV_CONSTANT_Integer: + { + int i = get4 (this_data); + _Jv_storeInt (&pool_data[index], i); + pool_tags[index] = JV_CONSTANT_Integer; + } + break; + + case JV_CONSTANT_Double: + { + jdouble d = long_bits_to_double ((jlong) get8 (this_data)); + _Jv_storeDouble (&pool_data[index], d); + pool_tags[index] = JV_CONSTANT_Double; + } + break; + + case JV_CONSTANT_Long: + { + jlong i = get8 (this_data); + _Jv_storeLong (&pool_data[index], i); + pool_tags[index] = JV_CONSTANT_Long; + } + break; + + default: + throw_class_format_error ("erroneous constant pool tag"); + } +} + + +void +_Jv_ClassReader::handleClassBegin + (int access_flags, int this_class, int super_class) +{ + unsigned char *pool_tags = (unsigned char*) def->constants.tags; + void **pool_data = (void**) def->constants.data; + + check_tag (this_class, JV_CONSTANT_Class); + _Jv_Utf8Const *loadedName = (_Jv_Utf8Const*)pool_data[this_class]; + + // was ClassLoader.defineClass called with an expected class name? + if (def->name == 0) + { + jclass orig = _Jv_FindClassInCache (loadedName, def->loader); + + if (orig == 0) + { + def->name = loadedName; + } + else + { + jstring msg = JvNewStringUTF ("anonymous " + "class data denotes " + "existing class "); + msg = msg->concat (orig->getName ()); + + throw_no_class_def_found_error (msg); + } + } + + // assert that the loaded class has the expected name, 5.3.5 + else if (! _Jv_equalUtf8Consts (loadedName, def->name)) + { + jstring msg = JvNewStringUTF ("loaded class "); + msg = msg->concat (def->getName ()); + msg = msg->concat (_Jv_NewStringUTF (" was in fact named ")); + jstring klass_name = _Jv_NewStringUTF (loadedName->data); + msg = msg->concat (klass_name); + + throw_no_class_def_found_error (msg); + } + + def->accflags = access_flags; + pool_data[this_class] = (void*)def; + pool_tags[this_class] = JV_CONSTANT_ResolvedClass; + + if (super_class == 0) + { + // interfaces have java.lang.Object as super. + if (access_flags & INTERFACE) + { + def->superclass = (jclass)&ClassObject; + } + + // FIXME: Consider this carefully! + else if (!_Jv_equalUtf8Consts (def->name, ClassObject.name)) + { + throw_no_class_def_found_error ("loading java.lang.Object"); + } + } + + // In the pre-loading state, it can be looked up in the + // cache only by this thread! This allows the super-class + // to include references to this class. + + def->state = JV_STATE_PRELOADING; + _Jv_RegisterClass (def); + + if (super_class != 0) + { + // load the super class + check_tag (super_class, JV_CONSTANT_Class); + _Jv_Utf8Const* super_name = + (_Jv_Utf8Const*)pool_data[super_class]; + + // load the super class using our defining loader + jclass the_super = _Jv_FindClass (super_name, + def->loader); + + // This will establish that we are allowed to be a subclass, + // and check for class circularity error + checkExtends (def, the_super); + + def->superclass = the_super; + pool_data[super_class] = (void*) the_super; + pool_tags[super_class] = JV_CONSTANT_ResolvedClass; + } + + // now we've come past the circularity problem, we can + // now say that we're loading... + + def->state = JV_STATE_LOADING; + def->notifyAll (); +} + +///// implements the checks described in sect. 5.3.5.3 +void +_Jv_ClassReader::checkExtends (jclass sub, jclass super) +{ + // having an interface or a final class as a superclass is no good + if ((super->accflags & (INTERFACE | FINAL)) != 0) + { + throw_incompatible_class_change_error (sub->getName ()); + } + + // if the super class is not public, we need to check some more + if ((super->accflags & PUBLIC) == 0) + { + // With package scope, the classes must have the same + // class loader. + if ( sub->loader != super->loader + || !_Jv_ClassNameSamePackage (sub->name, super->name)) + { + throw_incompatible_class_change_error (sub->getName ()); + } + } + + for (; super != 0; super = super->superclass) + { + if (super == sub) + throw_class_circularity_error (sub->getName ()); + } +} + + + +void _Jv_ClassReader::handleInterfacesBegin (int count) +{ + def->interfaces = (jclass*) _Jv_AllocBytesChecked (count*sizeof (jclass)); + def->interface_count = count; +} + +void _Jv_ClassReader::handleInterface (int if_number, int offset) +{ + void ** pool_data = def->constants.data; + unsigned char * pool_tags = (unsigned char*) def->constants.tags; + + jclass the_interface; + + if (pool_tags[offset] == JV_CONSTANT_Class) + { + _Jv_Utf8Const* name = (_Jv_Utf8Const*) pool_data[offset]; + the_interface = _Jv_FindClass (name, def->loader); + } + else if (pool_tags[offset] == JV_CONSTANT_ResolvedClass) + { + the_interface = (jclass)pool_data[offset]; + } + else + { + throw_no_class_def_found_error ("erroneous constant pool tag"); + } + + // checks the validity of the_interface, and that we are in fact + // allowed to implement that interface. + checkImplements (def, the_interface); + + pool_data[offset] = (void*)the_interface; + pool_tags[offset] = JV_CONSTANT_ResolvedClass; + + def->interfaces[if_number] = the_interface; +} + +void +_Jv_ClassReader::checkImplements (jclass sub, jclass super) +{ + // well, it *must* be an interface + if ((super->accflags & INTERFACE) == 0) + { + throw_incompatible_class_change_error (sub->getName ()); + } + + // if it has package scope, it must also be defined by the + // same loader. + if ((super->accflags & PUBLIC) == 0) + { + if ( sub->loader != super->loader + || !_Jv_ClassNameSamePackage (sub->name, super->name)) + { + throw_incompatible_class_change_error (sub->getName ()); + } + } + + // FIXME: add interface circularity check here + if (sub == super) + { + throw_class_circularity_error (sub->getName ()); + } +} + +void _Jv_ClassReader::handleFieldsBegin (int count) +{ + def->fields = (_Jv_Field*) + _Jv_AllocBytesChecked (count * sizeof (_Jv_Field)); + def->field_count = count; + def->field_initializers = (_Jv_ushort*) + _Jv_AllocBytesChecked (count * sizeof (_Jv_ushort)); + for (int i = 0; i < count; i++) + def->field_initializers[i] = (_Jv_ushort) 0; +} + +void _Jv_ClassReader::handleField (int field_no, + int flags, + int name, + int desc) +{ + void **const pool_data = def->constants.data; + + _Jv_Field *field = &def->fields[field_no]; + _Jv_Utf8Const *field_name = (_Jv_Utf8Const*) pool_data[name]; + +#ifndef COMPACT_FIELDS + field->name = field_name; +#else + field->nameIndex = name; +#endif + + if (verify) + _Jv_VerifyIdentifier (field_name); + + // ignore flags we don't know about. + field->flags = flags & ALL_FLAGS; + + if (verify) + { + if (field->flags & (SYNCHRONIZED|NATIVE|INTERFACE|ABSTRACT)) + throw_class_format_error ("erroneous field access flags"); + + if (1 < ( ((field->flags & PUBLIC) ? 1 : 0) + +((field->flags & PRIVATE) ? 1 : 0) + +((field->flags & PROTECTED) ? 1 : 0))) + throw_class_format_error ("erroneous field access flags"); + } + + _Jv_Utf8Const* sig = (_Jv_Utf8Const*) pool_data[desc]; + + if (verify) + _Jv_VerifyFieldSignature (sig); + + // field->type is really a jclass, but while it is still + // unresolved we keep an _Jv_Utf8Const* instead. + field->type = (jclass) sig; + field->flags |= _Jv_FIELD_UNRESOLVED_FLAG; + field->u.boffset = 0; +} + + +void _Jv_ClassReader::handleConstantValueAttribute (int field_index, + int value) +{ + _Jv_Field *field = &def->fields[field_index]; + + if ((field->flags & (STATIC|FINAL|PRIVATE)) == 0) + { + // Ignore, as per vmspec #4.7.2 + return; + } + + // do not allow multiple constant fields! + if (field->flags & _Jv_FIELD_CONSTANT_VALUE) + throw_class_format_error ("field has multiple ConstantValue attributes"); + + field->flags |= _Jv_FIELD_CONSTANT_VALUE; + def->field_initializers[field_index] = value; + + /* type check the initializer */ + + if (value <= 0 || value >= pool_count) + throw_class_format_error ("erroneous ConstantValue attribute"); + + /* FIXME: do the rest */ +} + +void _Jv_ClassReader::handleFieldsEnd () +{ + // We need to reorganize the fields so that the static ones are first, + // to conform to GCJ class layout. + + int low = 0; + int high = def->field_count-1; + _Jv_Field *fields = def->fields; + _Jv_ushort *inits = def->field_initializers; + + // this is kind of a raw version of quicksort. + while (low < high) + { + // go forward on low, while it's a static + while (low < high && (fields[low].flags & STATIC) != 0) + low++; + + // go backwards on high, while it's a non-static + while (low < high && (fields[high].flags & STATIC) == 0) + high--; + + if (low==high) + break; + + _Jv_Field tmp = fields[low]; + _Jv_ushort itmp = inits[low]; + + fields[low] = fields[high]; + inits[low] = inits[high]; + + fields[high] = tmp; + inits[high] = itmp; + + high -= 1; + low += 1; + } + + if ((fields[low].flags & STATIC) != 0) + low += 1; + + def->static_field_count = low; +} + + + +void _Jv_ClassReader::handleMethodsBegin (int count) +{ + def->methods = (_Jv_Method*) + _Jv_AllocBytesChecked (sizeof (_Jv_Method)*count); + + def->interpreted_methods = (_Jv_InterpMethod**) + _Jv_AllocBytesChecked (sizeof (_Jv_InterpMethod*) * count); + + for (int i = 0; i < count; i++) + def->interpreted_methods[i] = 0; + + def->method_count = count; +} + + +void _Jv_ClassReader::handleMethod + (int mth_index, int accflags, int name, int desc) +{ + void **const pool_data = def->constants.data; + _Jv_Method *method = &def->methods[mth_index]; + + check_tag (name, JV_CONSTANT_Utf8); + prepare_pool_entry (name, JV_CONSTANT_Utf8); + method->name = (_Jv_Utf8Const*)pool_data[name]; + + check_tag (desc, JV_CONSTANT_Utf8); + prepare_pool_entry (desc, JV_CONSTANT_Utf8); + method->signature = (_Jv_Utf8Const*)pool_data[desc]; + + // ignore unknown flags + method->accflags = accflags & ALL_FLAGS; + + // intialize... + method->ncode = 0; + + if (verify) + { + if (_Jv_equalUtf8Consts (method->name, clinit_name) + || _Jv_equalUtf8Consts (method->name, init_name)) + /* ignore */; + else + _Jv_VerifyIdentifier (method->name); + + _Jv_VerifyMethodSignature (method->signature); + + if (method->accflags & (VOLATILE|TRANSIENT|INTERFACE)) + throw_class_format_error ("erroneous method access flags"); + + if (1 < ( ((method->accflags & PUBLIC) ? 1 : 0) + +((method->accflags & PRIVATE) ? 1 : 0) + +((method->accflags & PROTECTED) ? 1 : 0))) + throw_class_format_error ("erroneous method access flags"); + } +} + +void _Jv_ClassReader::handleCodeAttribute + (int method_index, int max_stack, int max_locals, + int code_start, int code_length, int exc_table_length) +{ + int size = _Jv_InterpMethod::size (exc_table_length, code_length); + _Jv_InterpMethod *method = + (_Jv_InterpMethod*) (_Jv_AllocBytesChecked (size)); + + method->max_stack = max_stack; + method->max_locals = max_locals; + method->code_length = code_length; + method->exc_count = exc_table_length; + method->defining_class = def; + method->self = &def->methods[method_index]; + + // grab the byte code! + memcpy ((void*) method->bytecode (), + (void*) (bytes+code_start), + code_length); + + def->interpreted_methods[method_index] = method; + + /* that's all we do for now */ +} + +void _Jv_ClassReader::handleExceptionTableEntry + (int method_index, int exc_index, + int start_pc, int end_pc, int handler_pc, int catch_type) +{ + _Jv_InterpMethod *method = def->interpreted_methods[method_index]; + _Jv_InterpException *exc = method->exceptions (); + + exc[exc_index].start_pc = start_pc; + exc[exc_index].end_pc = end_pc; + exc[exc_index].handler_pc = handler_pc; + exc[exc_index].handler_type = catch_type; +} + +void _Jv_ClassReader::handleMethodsEnd () +{ + for (int i = 0; i < def->method_count; i++) + { + _Jv_Method *method = &def->methods[i]; + if (method->accflags & (NATIVE|ABSTRACT)) + { + if (def->interpreted_methods[i] != 0) + throw_class_format_error ("code provided " + "for abstract or native method"); + } + else + { + if (def->interpreted_methods[i] == 0) + throw_class_format_error ("abstract or native method " + "with no code"); + } + } + +} + + +/** This section takes care of verifying integrity of identifiers, + signatures, field ddescriptors, and class names */ + +#define UTF8_PEEK(PTR, LIMIT) \ + ({ unsigned char* xxkeep = (PTR); \ + int xxch = UTF8_GET(PTR,LIMIT); \ + PTR = xxkeep; xxch; }) + +/* verify one element of a type descriptor or signature */ +static unsigned char* +_Jv_VerifyOne (unsigned char* ptr, unsigned char* limit, bool void_ok) +{ + if (ptr >= limit) + return 0; + + int ch = UTF8_GET (ptr, limit); + + switch (ch) + { + case 'V': + if (! void_ok) return 0; + + case 'S': case 'B': case 'I': case 'J': + case 'Z': case 'C': case 'F': case 'D': + break; + + case 'L': + { + unsigned char *start = ptr, *end; + do { + if (ptr > limit) + return 0; + + end = ptr; + + if ((ch = UTF8_GET (ptr, limit)) == -1) + return 0; + + } while (ch != ';'); + _Jv_VerifyClassName (start, (unsigned short) (end-start)); + } + break; + + case '[': + return _Jv_VerifyOne (ptr, limit, false); + break; + + default: + return 0; + } + + return ptr; + +} + + +/** verification and loading procedures **/ + +void +_Jv_VerifyFieldSignature (_Jv_Utf8Const*sig) +{ + unsigned char* ptr = (unsigned char*) sig->data; + unsigned char* limit = ptr + sig->length; + + ptr = _Jv_VerifyOne (ptr, limit, false); + + if (ptr != limit) + throw_class_format_error ("erroneous type descriptor"); +} + +void +_Jv_VerifyMethodSignature (_Jv_Utf8Const*sig) +{ + unsigned char* ptr = (unsigned char*) sig->data; + unsigned char* limit = ptr + sig->length; + + if (ptr == limit) + throw_class_format_error ("erroneous type descriptor"); + + if (UTF8_GET(ptr,limit) != '(') + throw_class_format_error ("erroneous type descriptor"); + + while (ptr && UTF8_PEEK (ptr, limit) != ')') + ptr = _Jv_VerifyOne (ptr, limit, false); + + if (UTF8_GET (ptr, limit) != ')') + throw_class_format_error ("erroneous type descriptor"); + + // get the return type + ptr = _Jv_VerifyOne (ptr, limit, true); + + if (ptr != limit) + throw_class_format_error ("erroneous type descriptor"); + + return; + +} + +/* we try to avoid calling the Character methods all the time, + in fact, they will only be called for non-standard things */ + +static __inline__ int +is_identifier_start (int c) +{ + unsigned int ch = (unsigned)c; + + if ((ch - 0x41U) < 29U) /* A ... Z */ + return 1; + if ((ch - 0x61U) < 29U) /* a ... z */ + return 1; + if (ch == 0x5FU) /* _ */ + return 1; + + return character->isJavaIdentifierStart ((jchar) ch); +} + +static __inline__ int +is_identifier_part (int c) +{ + unsigned int ch = (unsigned)c; + + if ((ch - 0x41U) < 29U) /* A ... Z */ + return 1; + if ((ch - 0x61U) < 29U) /* a ... z */ + return 1; + if ((ch - 0x30) < 10U) /* 0 .. 9 */ + return 1; + if (ch == 0x5FU || ch == 0x24U) /* _ $ */ + return 1; + + return character->isJavaIdentifierStart ((jchar) ch); +} + +void +_Jv_VerifyIdentifier (_Jv_Utf8Const* name) +{ + unsigned char *ptr = (unsigned char*) name->data; + unsigned char *limit = ptr + name->length; + int ch; + + if ((ch = UTF8_GET (ptr, limit))==-1 + || ! is_identifier_start (ch)) + throw_class_format_error ("erroneous identifier"); + + while (ptr != limit) + { + if ((ch = UTF8_GET (ptr, limit))==-1 + || ! is_identifier_part (ch)) + throw_class_format_error ("erroneous identifier"); + } +} + + +void +_Jv_VerifyClassName (unsigned char* ptr, _Jv_ushort length) +{ + unsigned char *limit = ptr+length; + int ch; + + next_level: + do { + if ((ch = UTF8_GET (ptr, limit))==-1) + throw_class_format_error ("erroneous class name"); + if (! is_identifier_start (ch)) + throw_class_format_error ("erroneous class name"); + do { + if (ptr == limit) + return; + else if ((ch = UTF8_GET (ptr, limit))==-1) + throw_class_format_error ("erroneous class name"); + else if (ch == '.') + goto next_level; + else if (! is_identifier_part (ch)) + throw_class_format_error ("erroneous class name"); + } while (true); + } while (true); + +} + +void +_Jv_VerifyClassName (_Jv_Utf8Const *name) +{ + _Jv_VerifyClassName ((unsigned char*)&name->data[0], + (_Jv_ushort) name->length); +} + + +/** returns true, if name1 and name2 represents classes in the same + package. */ + +bool +_Jv_ClassNameSamePackage (_Jv_Utf8Const *name1, _Jv_Utf8Const *name2) +{ + unsigned char* ptr1 = (unsigned char*) name1->data; + unsigned char* limit1 = ptr1 + name1->length; + + unsigned char* last1 = ptr1; + + // scan name1, and find the last occurrence of '.' + while (ptr1 < limit1) { + int ch1 = UTF8_GET (ptr1, limit1); + + if (ch1 == '.') + last1 = ptr1; + + else if (ch1 == -1) + return false; + } + + // now the length of name1's package name is len + int len = last1 - (unsigned char*) name1->data; + + // if this is longer than name2, then we're off + if (len > name2->length) + return false; + + // then compare the first len bytes for equality + if (memcmp ((void*) name1->data, (void*) name2->data, len) == 0) + { + // check that there are no .'s after position len in name2 + + unsigned char* ptr2 = (unsigned char*) name2->data + len; + unsigned char* limit2 = + (unsigned char*) name2->data + name2->length; + + while (ptr2 < limit2) + { + int ch2 = UTF8_GET (ptr2, limit2); + if (ch2 == -1 || ch2 == '.') + return false; + } + return true; + } + return false; +} + + + +/** Here we define the exceptions that can be thrown */ + +static void +throw_no_class_def_found_error (jstring msg) +{ + if (msg == 0) + JvThrow (new java::lang::NoClassDefFoundError); + else + JvThrow (new java::lang::NoClassDefFoundError (msg)); +} + +static void +throw_no_class_def_found_error (char *msg) +{ + throw_no_class_def_found_error (JvNewStringLatin1 (msg)); +} + +static void +throw_class_format_error (jstring msg) +{ + if (msg == 0) + JvThrow (new java::lang::ClassFormatError); + else + JvThrow (new java::lang::ClassFormatError (msg)); +} + +static void +throw_class_format_error (char *msg) +{ + throw_class_format_error (JvNewStringLatin1 (msg)); +} + +static void +throw_internal_error (char *msg) +{ + JvThrow + (new java::lang::InternalError (JvNewStringLatin1 (msg))); +} + +static jfloat int_bits_to_float (jint value) +{ + return java::lang::Float::intBitsToFloat (value); +} + +static jdouble long_bits_to_double (jlong value) +{ + return java::lang::Double::longBitsToDouble (value); +} + +static void throw_incompatible_class_change_error (jstring msg) +{ + JvThrow (new java::lang::IncompatibleClassChangeError (msg)); +} + +static void throw_class_circularity_error (jstring msg) +{ + JvThrow (new java::lang::ClassCircularityError (msg)); +} + +#endif /* INTERPRETER */ + diff --git a/libjava/gij.cc b/libjava/gij.cc new file mode 100644 index 00000000000..f29e82520d1 --- /dev/null +++ b/libjava/gij.cc @@ -0,0 +1,27 @@ +/* Copyright (C) 1999 Cygnus Solutions + + This file is part of libgcj. + +This software is copyrighted work licensed under the terms of the +Libgcj License. Please consult the file "LIBGCJ_LICENSE" for +details. */ + +/* Author: Kresten Krab Thorup <krab@gnu.org> */ + +#include <jvm.h> +#include <cni.h> +#include <stdio.h> + +#include <java/lang/System.h> +#include <java/util/Properties.h> + +int main (int argc, const char **argv) +{ + if (argc < 2) + { + printf ("usage: %s <class name> args\n", argv[0]); + exit (1); + } + + JvRunMain (0, argc, argv); +} diff --git a/libjava/gnu/gcj/runtime/MethodInvocation.java b/libjava/gnu/gcj/runtime/MethodInvocation.java new file mode 100644 index 00000000000..d1664bae414 --- /dev/null +++ b/libjava/gnu/gcj/runtime/MethodInvocation.java @@ -0,0 +1,32 @@ +// MethodInvocation.java - wrapper used by the interpreter. +// (the native method is implemented in interpret.cc) + +/* Copyright (C) 1999 Cygnus Solutions + + This file is part of libgcj. + +This software is copyrighted work licensed under the terms of the +Libgcj License. Please consult the file "LIBGCJ_LICENSE" for +details. */ + +/* Author: Kresten Krab Thorup <krab@gnu.org> */ + +package gnu.gcj.runtime; + +import gnu.gcj.RawData; + +final class MethodInvocation { + + private static Throwable continue0 (RawData meth, RawData inv) + { + try { + continue1 (meth, inv); + } catch (Throwable ex) { + return ex; + } + return null; + } + + private static native void continue1 (RawData meth, RawData inv); + +} diff --git a/libjava/gnu/gcj/util/path/CacheEntry.java b/libjava/gnu/gcj/util/path/CacheEntry.java new file mode 100644 index 00000000000..949200b7a29 --- /dev/null +++ b/libjava/gnu/gcj/util/path/CacheEntry.java @@ -0,0 +1,65 @@ +// CacheEntry.java -- directory cache + +/* Copyright (C) 1999 Cygnus Solutions + + This file is part of libgcj. + +This software is copyrighted work licensed under the terms of the +Libgcj License. Please consult the file "LIBGCJ_LICENSE" for +details. */ + +/* Author: Kresten Krab Thorup <krab@gnu.org> */ + +package gnu.gcj.util.path; + +import java.util.*; +import java.util.zip.*; +import java.io.*; +import java.net.*; + + +final class CacheEntry { + String dir; + String[] files; + long time; + + CacheEntry (String d) + { + dir = d; + files = new File(dir).list(); + time = System.currentTimeMillis (); + } + + void touch () + { + time = System.currentTimeMillis (); + } + + final long EXPIRATION_TIME_MS = 1000; + + boolean is_old () { + return (System.currentTimeMillis () - time) > EXPIRATION_TIME_MS; + } + + public int hashCode () { return dir.hashCode(); } + boolean contains (String file) { + if (files == null) + return false; + + int index = file.lastIndexOf(SearchPath.file_seperator_char); + String f; + + if (index == -1) + f = file; + else + f = file.substring (index+1); + + for (int i = 0; i < files.length; i++) + { + if (f.equals (files[i])) return true; + } + + return false; + } +} + diff --git a/libjava/gnu/gcj/util/path/DirectoryPathEntry.java b/libjava/gnu/gcj/util/path/DirectoryPathEntry.java new file mode 100644 index 00000000000..a9ca602341b --- /dev/null +++ b/libjava/gnu/gcj/util/path/DirectoryPathEntry.java @@ -0,0 +1,136 @@ +// DirectoryPathEntry.java -- search path element for directories + +/* Copyright (C) 1999 Cygnus Solutions + + This file is part of libgcj. + +This software is copyrighted work licensed under the terms of the +Libgcj License. Please consult the file "LIBGCJ_LICENSE" for +details. */ + +/* Author: Kresten Krab Thorup <krab@gnu.org> */ + +package gnu.gcj.util.path; + +import java.util.*; +import java.util.zip.*; +import java.io.*; +import java.net.*; + +final class DirectoryPathEntry extends PathEntry +{ + final File dir; + final String base_canon; + + public String toString () { return base_canon; } + + DirectoryPathEntry (File f) + throws java.io.IOException + { + if (!f.isAbsolute ()) + throw new IllegalArgumentException (); + + dir = f; + base_canon = dir.getCanonicalPath (); + } + + /* + * We maintain a cache of files, so that we + * can avoid many calls to stat(), which are + * very expensive. + * + * seen_cache contains (as keys) the directories + * which we have visited so far. The values are + * instances of CacheEntry, containing a time stamp, + * and a list of files in that directory. + * + */ + + private Hashtable seen_cache = new Hashtable (); + + private boolean in_cache (File f) + { + String rel_dir = f.getParent (); + CacheEntry ent; + + if (rel_dir == null) + throw new IllegalArgumentException (); + + ent = (CacheEntry) seen_cache.get (rel_dir); + if (ent == null) + { + ent = new CacheEntry (rel_dir); + seen_cache.put (rel_dir, ent); + } + + if (ent.contains (f.getPath ())) + { + return true; + } + + if ( ent.is_old () ) + { + if (f.exists ()) + { + seen_cache.remove (rel_dir); + return true; + } + else + { + ent.touch (); + } + } + + return false; + } + + URL getURL (String file) { + try { + File f = new File((new File (dir, file).getCanonicalPath ())); + + if (! f.getCanonicalPath ().startsWith (base_canon)) + throw new IllegalArgumentException (file); + + + if (in_cache (f)) + return new URL ("file", "", f.getPath ()); + else + return null; + + } catch (IOException x) { + return null; + } + } + + InputStream getStream (String file) { + try { + File f = new File((new File (dir, file)).getCanonicalPath ()); + + if (! f.getCanonicalPath ().startsWith (base_canon)) + throw new IllegalArgumentException (file); + + if (in_cache (f)) + return new FileInputStream (f); + else + return null; + } catch (IOException x) { + return null; + } + } + + byte[] getBytes (String file) { + File f = new File (dir, file); + + try { + if (in_cache (f)) + return readbytes (new FileInputStream (f), + (int) f.length ()); + else + return null; + } catch (IOException x) { + return null; + } + } + +} + diff --git a/libjava/gnu/gcj/util/path/PathEntry.java b/libjava/gnu/gcj/util/path/PathEntry.java new file mode 100644 index 00000000000..f83fc170170 --- /dev/null +++ b/libjava/gnu/gcj/util/path/PathEntry.java @@ -0,0 +1,55 @@ +// PathEntry.java -- abstract element of search paths + +/* Copyright (C) 1999 Cygnus Solutions + + This file is part of libgcj. + +This software is copyrighted work licensed under the terms of the +Libgcj License. Please consult the file "LIBGCJ_LICENSE" for +details. */ + +/* Author: Kresten Krab Thorup <krab@gnu.org> */ + +package gnu.gcj.util.path; + +import java.util.*; +import java.util.zip.*; +import java.io.*; +import java.net.*; + +abstract class PathEntry { + abstract URL getURL (String file); + abstract InputStream getStream (String file); + abstract byte[] getBytes (String file); + + /** + * Utility routine like InputStream.read(byte[], 0, len), but will + * read fully, even if all the data is not available at once. + */ + protected static byte[] readbytes (InputStream is, int length) + { + try { + + byte[] data = new byte[length]; + int read; + int off = 0; + + while (off != length) + { + read = is.read (data, off, (int) (length-off)); + + if (read == -1) + return null; + + off += read; + } + + return data; + } catch (IOException x) { + return null; + } + } + +} + + diff --git a/libjava/gnu/gcj/util/path/SearchPath.java b/libjava/gnu/gcj/util/path/SearchPath.java new file mode 100644 index 00000000000..ffc2ca8d178 --- /dev/null +++ b/libjava/gnu/gcj/util/path/SearchPath.java @@ -0,0 +1,205 @@ +// SearchPath.java -- generic search path utility + +/* Copyright (C) 1999 Cygnus Solutions + + This file is part of libgcj. + +This software is copyrighted work licensed under the terms of the +Libgcj License. Please consult the file "LIBGCJ_LICENSE" for +details. */ + +/* Author: Kresten Krab Thorup <krab@gnu.org> */ + + +package gnu.gcj.util.path; + +import java.util.*; +import java.util.zip.*; +import java.io.*; +import java.net.*; + +final public class SearchPath { + + final static String path_seperator + = System.getProperty ("path.separator"); + final static char path_seperator_char + = path_seperator.charAt (0); + final static String file_seperator + = System.getProperty ("file.separator"); + final static char file_seperator_char + = file_seperator.charAt (0); + + private Vector path; + + /** + * Constructs a SearchPath object, given a system path. + * The system path is expected to be seperated by the string + * defined by the <code>path.seperator</code> property. + * (<code>":"</code> on unix, <code>;</code> on Windows, etc.). + * The path may contain names of directories, or names of + * .zip or .jar files. Elements that are neither of these + * are ignored. + * @param sys_path the search path + */ + + SearchPath (String sys_path) + { + StringTokenizer st = new StringTokenizer (sys_path, path_seperator); + init (st); + } + + /** + * Constructs a SearchPath object, given a Vector of + * <code>String</code>, <code>File</code> or <code>URL</code> + * objects. + * The path may contain names of directories, or names of + * .zip or .jar files. Elements that are neither of these + * are ignored. + * @param p the vector of search path elements + */ + + SearchPath (Vector p) + { + init (p.elements ()); + } + + public URL getURL (String element) + { + URL result; + + Enumeration e = path.elements (); + while (e.hasMoreElements ()) + { + PathEntry ent = (PathEntry) e.nextElement (); + + result = ent.getURL (element); + + if (result != null) + { + return result; + } + } + + return null; + } + + + public InputStream getStream (String element) + { + InputStream result; + + Enumeration e = path.elements (); + while (e.hasMoreElements ()) + { + PathEntry ent = (PathEntry) e.nextElement (); + + result = ent.getStream (element); + + if (result != null) + { + return result; + } + } + + return null; + } + + + public byte[] getBytes (String element) + { + byte[] result; + + Enumeration e = path.elements (); + while (e.hasMoreElements ()) + { + PathEntry ent = (PathEntry) e.nextElement (); + result = ent.getBytes (element); + if (result != null) + { + System.out.println ("loading " + ent + + "(" + element + ")"); + return result; + } + } + + return null; + } + + + + private void init (Enumeration st) + { + path = new Vector (); + while (st.hasMoreElements ()) + { + Object e = st.nextElement (); + + String elem; + File efile; + + if (e instanceof URL) + { + path.addElement (new URLPathEntry ((URL) e)); + continue; + } + + if (e instanceof File) + { + efile = (File) e; + elem = efile.getPath (); + } + + else if (e instanceof String) + { + elem = (String) e; + efile = new File (elem); + } + + else + throw new IllegalArgumentException (); + + // make sure it is absolute, so we won't get + // trouble if the cwd is changed... + if (! efile.isAbsolute ()) + efile = new File (efile.getAbsolutePath ()); + + if (efile.isDirectory ()) + { + try { + path.addElement(new DirectoryPathEntry (efile)); + } catch (IOException x) { + /* ignore for now */ + } + } + + else if (efile.isFile ()) + { + int ext = elem.lastIndexOf ('.'); + if (ext == -1) + continue; + + if (!elem.substring(ext+1).equalsIgnoreCase("zip")) + continue; + + ZipPathEntry zpe = null; + try { + zpe = new ZipPathEntry (efile); + } catch (ZipException zx) { + System.err.println ("SearchPath::ZipException"); + zpe = null; + } catch (MalformedURLException mx) { + System.err.println ("SearchPath::URLException"); + zpe = null; + } catch (IOException iox) { + System.err.println ("SearchPath::IOException"); + zpe = null; + } + if (zpe != null) path.addElement (zpe); + } + } + + } + + +} + diff --git a/libjava/gnu/gcj/util/path/URLPathEntry.java b/libjava/gnu/gcj/util/path/URLPathEntry.java new file mode 100644 index 00000000000..68f9200b555 --- /dev/null +++ b/libjava/gnu/gcj/util/path/URLPathEntry.java @@ -0,0 +1,67 @@ +// URLPathEntry.java -- search path element for URL's + +/* Copyright (C) 1999 Cygnus Solutions + + This file is part of libgcj. + +This software is copyrighted work licensed under the terms of the +Libgcj License. Please consult the file "LIBGCJ_LICENSE" for +details. */ + +/* Author: Kresten Krab Thorup <krab@gnu.org> */ + +package gnu.gcj.util.path; + +import java.util.*; +import java.util.zip.*; +import java.io.*; +import java.net.*; + +final class URLPathEntry extends PathEntry { + final URL base; + + URLPathEntry (URL f) { + base = f; + } + + public String toString () { return base.toString (); } + + URL getURL (String file) { + + try { + URL res = new URL (base, file); + InputStream is = res.openStream (); // exc if not found + is.close (); + return res; + } catch (java.io.IOException x) { + return null; + } + } + + InputStream getStream (String file) { + + try { + URL res = new URL (base, file); + return res.openStream (); + } catch (java.io.IOException x) { + return null; + } + + } + + byte[] getBytes (String file) { + + try { + URL res = new URL (base, file); + URLConnection conn = res.openConnection (); + int len = conn.getContentLength (); + if (len == -1) return null; + return readbytes (conn.getInputStream (), len); + } catch (java.io.IOException x) { + return null; + } + + } + +} + diff --git a/libjava/gnu/gcj/util/path/ZipPathEntry.java b/libjava/gnu/gcj/util/path/ZipPathEntry.java new file mode 100644 index 00000000000..ac0226290a5 --- /dev/null +++ b/libjava/gnu/gcj/util/path/ZipPathEntry.java @@ -0,0 +1,86 @@ +// ZipPathEntry.java -- search path element for directories + +/* Copyright (C) 1999 Cygnus Solutions + + This file is part of libgcj. + +This software is copyrighted work licensed under the terms of the +Libgcj License. Please consult the file "LIBGCJ_LICENSE" for +details. */ + +/* Author: Kresten Krab Thorup <krab@gnu.org> */ + +package gnu.gcj.util.path; + +import java.util.*; +import java.util.zip.*; +import java.io.*; +import java.net.*; + + +final class ZipPathEntry extends PathEntry { + final ZipFile zip; + final URL file; + + public String toString () { return zip.getName (); } + + ZipPathEntry (File f) + throws MalformedURLException, ZipException, IOException + { + file = new URL ("file", "", f.getPath ()); + zip = new ZipFile (f); + zip.readDirectory (); + } + + /* + The url for a zip-file resource is, + + <code>file:///path/file.zip#name</code> + + Then, it is URLConnection's problem to handle that. + */ + + URL getURL (String f) { + + ZipEntry ent = zip.getEntry (f); + + try { + if (ent != null) + return new URL (file, "#"+f); + else + return null; + } catch (IOException x) { + return null; + } + } + + InputStream getStream (String f) { + + ZipEntry ent = zip.getEntry (f); + + try { + if (ent != null) + return zip.getInputStream (ent); + else + return null; + } catch (IOException x) { + return null; + } + } + + byte[] getBytes (String f) { + ZipEntry ent = zip.getEntry (f); + + try { + if (ent != null) + return readbytes (zip.getInputStream (ent), + (int) ent.getSize ()); + else + return null; + } catch (IOException x) { + return null; + } + + } +} + diff --git a/libjava/include/config.h.in b/libjava/include/config.h.in index d9b9a0a9cf8..03f51baca78 100644 --- a/libjava/include/config.h.in +++ b/libjava/include/config.h.in @@ -106,6 +106,9 @@ #undef HAVE_GETHOSTBYNAME_R #undef HAVE_GETHOSTBYADDR_R +/* Define if you want a bytecode interpreter. */ +#undef INTERPRETER + /* Define if you have the access function. */ #undef HAVE_ACCESS diff --git a/libjava/include/java-cpool.h b/libjava/include/java-cpool.h new file mode 100644 index 00000000000..f4d7ef9910a --- /dev/null +++ b/libjava/include/java-cpool.h @@ -0,0 +1,173 @@ +// java-cpool.h - Constant pool parsing header. -*- c++ -*- + +/* Copyright (C) 1999 Cygnus Solutions + + This file is part of libgcj. + +This software is copyrighted work licensed under the terms of the +Libgcj License. Please consult the file "LIBGCJ_LICENSE" for +details. */ + +#ifndef __JAVA_CPOOL_H__ +#define __JAVA_CPOOL_H__ + +#include <javaprims.h> + +// we rename these, to avoid polluting the name space +#define JV_CONSTANT_Undefined (0L) +#define JV_CONSTANT_Utf8 (1L) +#define JV_CONSTANT_Unicode (2L) +#define JV_CONSTANT_Integer (3L) +#define JV_CONSTANT_Float (4L) +#define JV_CONSTANT_Long (5L) +#define JV_CONSTANT_Double (6L) +#define JV_CONSTANT_Class (7L) +#define JV_CONSTANT_String (8L) +#define JV_CONSTANT_Fieldref (9L) +#define JV_CONSTANT_Methodref (10L) +#define JV_CONSTANT_InterfaceMethodref (11L) +#define JV_CONSTANT_NameAndType (12L) +#define JV_CONSTANT_ResolvedFlag (16L) +#define JV_CONSTANT_ResolvedString (16L | 8L) +#define JV_CONSTANT_ResolvedClass (16L | 7L) + +/* We use the following two operations uniformly for all put/get operations + * in the runtime system (constant pool & stack), to assure that we keep + * everything in the same format. The idea is, that these should be inlined + * away, into just a simple store (for small data types, and a pair of stores + * if double or long has alignment greater than void *. On an 64-bit + * architecture, all operations should be simple stores; on a 32-bit + * architecture it depends on the alignment requirement for the specific + * type. */ + +template <class T> +static inline void _Jv_put (void *dst, T value) +{ +#if 0 + if (sizeof (T) == 8 && __alignof__ (T) > __alignof__ (void*)) + { + jint *v_dst = (jint*)(dst); + jint *v_src = (jint*)&value; + + v_dst[0] = v_src[0]; + v_dst[1] = v_src[1]; + } + else +#endif + { + *((T*) (dst)) = value; + } +} + +template <class T> +static inline T _Jv_get (void *src) +{ +#if 0 + if (sizeof (T) == 8 && __alignof__ (T) > __alignof__ (void*)) + { + T value; + jint *v_dst = (jint*)&value; + jint *v_src = (jint*)src; + + v_dst[0] = v_src[0]; + v_dst[1] = v_src[1]; + + return value; + } + else +#endif + { + return *((T*) (src)); + } +} + +/** needed to keep the CONSTANT_XXXRef & CONSTANT_NameAndType entries */ +extern inline void +_Jv_storeIndexes (void **data, + _Jv_ushort index0, + _Jv_ushort index1) +{ + // accomodate 64bit machines... + if (sizeof (void*) == (2 * sizeof (jint))) + { + ((jint*)data)[0] = index0; + ((jint*)data)[1] = index0; + } + else + { + _Jv_put<jint>(data, ((jint)index0 << 16) | (jint)index1); + } +} + +extern inline void +_Jv_loadIndexes (const void **data, + _Jv_ushort& index0, + _Jv_ushort& index1) +{ + if (sizeof (void*) == (2*sizeof (jint))) + { + index0 = ((jint*)data)[0]; + index0 = ((jint*)data)[1]; + } + else + { + jint udata = _Jv_get<jint>(data); + + _Jv_uint uindex0 = ((udata >> 16) & 0xffff); + _Jv_uint uindex1 = udata & 0xffff; + + index0 = uindex0; + index1 = uindex1; + } +} + +extern inline void +_Jv_storeFloat (void **data, jfloat f) +{ + _Jv_put<jfloat>(data, f); +} + +extern inline jfloat +_Jv_loadFloat (void **data) +{ + return _Jv_get<jfloat>(data); +} + +extern inline void +_Jv_storeInt (void **data, jint i) +{ + _Jv_put<jint>(data, i); +} + +extern inline jint +_Jv_loadInt (void **data) +{ + return _Jv_get<jint>(data); +} + +extern inline void +_Jv_storeLong (void **data, jlong l) +{ + return _Jv_put<jlong>(data, l); +} + +extern inline jlong +_Jv_loadLong (void **data) +{ + return _Jv_get<jlong>(data); +} + +extern inline void +_Jv_storeDouble (void **data, jdouble d) +{ + _Jv_put<jdouble>(data, d); +} + +extern inline jdouble +_Jv_loadDouble (void **data) +{ + return _Jv_get<jdouble> (data); +} + + +#endif /* __JAVA_CPOOL_H__ */ diff --git a/libjava/include/java-field.h b/libjava/include/java-field.h index b4529fca30a..d00d9ce7726 100644 --- a/libjava/include/java-field.h +++ b/libjava/include/java-field.h @@ -53,7 +53,18 @@ struct _Jv_Field jfieldID getNextInstanceField () { return this + 1; } - jboolean isRef () { return ! isResolved () || ! type->isPrimitive (); } + jboolean isRef () + { + if (!isResolved ()) + { + char first = ((_Jv_Utf8Const*)type)->data[0]; + return first == '[' || first == 'L'; + } + else + { + return ! type->isPrimitive (); + } + } // FIXME - may need to mask off internal flags. int getModifiers() { return flags; } diff --git a/libjava/include/java-insns.h b/libjava/include/java-insns.h new file mode 100644 index 00000000000..8b19abd58ad --- /dev/null +++ b/libjava/include/java-insns.h @@ -0,0 +1,247 @@ +// java-insns.h - Instruction encodings. This is -*- c++ -*- + +/* Copyright (C) 1999 Cygnus Solutions + + This file is part of libgcj. + +This software is copyrighted work licensed under the terms of the +Libgcj License. Please consult the file "LIBGCJ_LICENSE" for +details. */ + +static const int op_nop = 0x00; +static const int op_aconst_null = 0x01; +static const int op_iconst_m1 = 0x02; +static const int op_iconst_0 = 0x03; +static const int op_iconst_1 = 0x04; +static const int op_iconst_2 = 0x05; +static const int op_iconst_3 = 0x06; +static const int op_iconst_4 = 0x07; +static const int op_iconst_5 = 0x08; +static const int op_lconst_0 = 0x09; +static const int op_lconst_1 = 0x0a; +static const int op_fconst_0 = 0x0b; +static const int op_fconst_1 = 0x0c; +static const int op_fconst_2 = 0x0d; +static const int op_dconst_0 = 0x0e; +static const int op_dconst_1 = 0x0f; +static const int op_bipush = 0x10; +static const int op_sipush = 0x11; +static const int op_ldc = 0x12; +static const int op_ldc_w = 0x13; +static const int op_ldc2_w = 0x14; +static const int op_iload = 0x15; +static const int op_lload = 0x16; +static const int op_fload = 0x17; +static const int op_dload = 0x18; +static const int op_aload = 0x19; +static const int op_iload_0 = 0x1a; +static const int op_iload_1 = 0x1b; +static const int op_iload_2 = 0x1c; +static const int op_iload_3 = 0x1d; +static const int op_lload_0 = 0x1e; +static const int op_lload_1 = 0x1f; +static const int op_lload_2 = 0x20; +static const int op_lload_3 = 0x21; +static const int op_fload_0 = 0x22; +static const int op_fload_1 = 0x23; +static const int op_fload_2 = 0x24; +static const int op_fload_3 = 0x25; +static const int op_dload_0 = 0x26; +static const int op_dload_1 = 0x27; +static const int op_dload_2 = 0x28; +static const int op_dload_3 = 0x29; +static const int op_aload_0 = 0x2a; +static const int op_aload_1 = 0x2b; +static const int op_aload_2 = 0x2c; +static const int op_aload_3 = 0x2d; +static const int op_iaload = 0x2e; +static const int op_laload = 0x2f; +static const int op_faload = 0x30; +static const int op_daload = 0x31; +static const int op_aaload = 0x32; +static const int op_baload = 0x33; +static const int op_caload = 0x34; +static const int op_saload = 0x35; +static const int op_istore = 0x36; +static const int op_lstore = 0x37; +static const int op_fstore = 0x38; +static const int op_dstore = 0x39; +static const int op_astore = 0x3a; +static const int op_istore_0 = 0x3b; +static const int op_istore_1 = 0x3c; +static const int op_istore_2 = 0x3d; +static const int op_istore_3 = 0x3e; +static const int op_lstore_0 = 0x3f; +static const int op_lstore_1 = 0x40; +static const int op_lstore_2 = 0x41; +static const int op_lstore_3 = 0x42; +static const int op_fstore_0 = 0x43; +static const int op_fstore_1 = 0x44; +static const int op_fstore_2 = 0x45; +static const int op_fstore_3 = 0x46; +static const int op_dstore_0 = 0x47; +static const int op_dstore_1 = 0x48; +static const int op_dstore_2 = 0x49; +static const int op_dstore_3 = 0x4a; +static const int op_astore_0 = 0x4b; +static const int op_astore_1 = 0x4c; +static const int op_astore_2 = 0x4d; +static const int op_astore_3 = 0x4e; +static const int op_iastore = 0x4f; +static const int op_lastore = 0x50; +static const int op_fastore = 0x51; +static const int op_dastore = 0x52; +static const int op_aastore = 0x53; +static const int op_bastore = 0x54; +static const int op_castore = 0x55; +static const int op_sastore = 0x56; +static const int op_pop = 0x57; +static const int op_pop2 = 0x58; +static const int op_dup = 0x59; +static const int op_dup_x1 = 0x5a; +static const int op_dup_x2 = 0x5b; +static const int op_dup2 = 0x5c; +static const int op_dup2_x1 = 0x5d; +static const int op_dup2_x2 = 0x5e; +static const int op_swap = 0x5f; +static const int op_iadd = 0x60; +static const int op_ladd = 0x61; +static const int op_fadd = 0x62; +static const int op_dadd = 0x63; +static const int op_isub = 0x64; +static const int op_lsub = 0x65; +static const int op_fsub = 0x66; +static const int op_dsub = 0x67; +static const int op_imul = 0x68; +static const int op_lmul = 0x69; +static const int op_fmul = 0x6a; +static const int op_dmul = 0x6b; +static const int op_idiv = 0x6c; +static const int op_ldiv = 0x6d; +static const int op_fdiv = 0x6e; +static const int op_ddiv = 0x6f; +static const int op_irem = 0x70; +static const int op_lrem = 0x71; +static const int op_frem = 0x72; +static const int op_drem = 0x73; +static const int op_ineg = 0x74; +static const int op_lneg = 0x75; +static const int op_fneg = 0x76; +static const int op_dneg = 0x77; +static const int op_ishl = 0x78; +static const int op_lshl = 0x79; +static const int op_ishr = 0x7a; +static const int op_lshr = 0x7b; +static const int op_iushr = 0x7c; +static const int op_lushr = 0x7d; +static const int op_iand = 0x7e; +static const int op_land = 0x7f; +static const int op_ior = 0x80; +static const int op_lor = 0x81; +static const int op_ixor = 0x82; +static const int op_lxor = 0x83; +static const int op_iinc = 0x84; +static const int op_i2l = 0x85; +static const int op_i2f = 0x86; +static const int op_i2d = 0x87; +static const int op_l2i = 0x88; +static const int op_l2f = 0x89; +static const int op_l2d = 0x8a; +static const int op_f2i = 0x8b; +static const int op_f2l = 0x8c; +static const int op_f2d = 0x8d; +static const int op_d2i = 0x8e; +static const int op_d2l = 0x8f; +static const int op_d2f = 0x90; +static const int op_i2b = 0x91; +static const int op_i2c = 0x92; +static const int op_i2s = 0x93; +static const int op_lcmp = 0x94; +static const int op_fcmpl = 0x95; +static const int op_fcmpg = 0x96; +static const int op_dcmpl = 0x97; +static const int op_dcmpg = 0x98; +static const int op_ifeq = 0x99; +static const int op_ifne = 0x9a; +static const int op_iflt = 0x9b; +static const int op_ifge = 0x9c; +static const int op_ifgt = 0x9d; +static const int op_ifle = 0x9e; +static const int op_if_icmpeq = 0x9f; +static const int op_if_icmpne = 0xa0; +static const int op_if_icmplt = 0xa1; +static const int op_if_icmpge = 0xa2; +static const int op_if_icmpgt = 0xa3; +static const int op_if_icmple = 0xa4; +static const int op_if_acmpeq = 0xa5; +static const int op_if_acmpne = 0xa6; +static const int op_goto = 0xa7; +static const int op_jsr = 0xa8; +static const int op_ret = 0xa9; +static const int op_tableswitch = 0xaa; +static const int op_lookupswitch = 0xab; +static const int op_ireturn = 0xac; +static const int op_lreturn = 0xad; +static const int op_freturn = 0xae; +static const int op_dreturn = 0xaf; +static const int op_areturn = 0xb0; +static const int op_return = 0xb1; +static const int op_getstatic = 0xb2; +static const int op_putstatic = 0xb3; +static const int op_getfield = 0xb4; +static const int op_putfield = 0xb5; +static const int op_invokevirtual = 0xb6; +static const int op_invokespecial = 0xb7; +static const int op_invokestatic = 0xb8; +static const int op_invokeinterface = 0xb9; +static const int op_xxxunusedxxx1 = 0xba; +static const int op_new = 0xbb; +static const int op_newarray = 0xbc; +static const int op_anewarray = 0xbd; +static const int op_arraylength = 0xbe; +static const int op_athrow = 0xbf; +static const int op_checkcast = 0xc0; +static const int op_instanceof = 0xc1; +static const int op_monitorenter = 0xc2; +static const int op_monitorexit = 0xc3; +static const int op_wide = 0xc4; +static const int op_multianewarray = 0xc5; +static const int op_ifnull = 0xc6; +static const int op_ifnonnull = 0xc7; +static const int op_goto_w = 0xc8; +static const int op_jsr_w = 0xc9; + +// new opcodes + +static const int op_putfield_1 = 0xca; +static const int op_putfield_2 = 0xcb; +static const int op_putfield_4 = 0xcd; +static const int op_putfield_8 = 0xce; +static const int op_putfield_a = 0xcf; + +static const int op_putstatic_1 = 0xd0; +static const int op_putstatic_2 = 0xd1; +static const int op_putstatic_4 = 0xd2; +static const int op_putstatic_8 = 0xd3; +static const int op_putstatic_a = 0xd4; + +static const int op_getfield_1 = 0xd5; +static const int op_getfield_2s = 0xd6; +static const int op_getfield_2u = 0xd7; +static const int op_getfield_4 = 0xd8; +static const int op_getfield_8 = 0xd9; +static const int op_getfield_a = 0xda; + +static const int op_getstatic_1 = 0xdb; +static const int op_getstatic_2s = 0xdc; +static const int op_getstatic_2u = 0xdd; +static const int op_getstatic_4 = 0xde; +static const int op_getstatic_8 = 0xdf; +static const int op_getstatic_a = 0xe0; + +static const int op_invokefinal = 0xe1; +static const int op_invokevtable = 0xe2; + + + + diff --git a/libjava/include/java-interp.h b/libjava/include/java-interp.h new file mode 100644 index 00000000000..70feec19beb --- /dev/null +++ b/libjava/include/java-interp.h @@ -0,0 +1,172 @@ +// java-interp.h - Header file for the bytecode interpreter. -*- c++ -*- + +/* Copyright (C) 1999 Cygnus Solutions + + This file is part of libgcj. + +This software is copyrighted work licensed under the terms of the +Libgcj License. Please consult the file "LIBGCJ_LICENSE" for +details. */ + +#ifndef __JAVA_INTERP_H__ +#define __JAVA_INTERP_H__ + +#include <config.h> + +#include <jvm.h> +#include <java-cpool.h> + +#ifdef INTERPRETER + +#pragma interface + +#include <java/lang/Class.h> +#include <java/lang/ClassLoader.h> +#include <gnu/gcj/runtime/MethodInvocation.h> + +extern "C" { +#include <ffi.h> +} + +extern inline jboolean +_Jv_IsInterpretedClass (jclass c) +{ + return (c->loader != 0); +} + +struct _Jv_ResolvedMethod; + +void _Jv_VerifyFieldSignature (_Jv_Utf8Const*sig); +void _Jv_VerifyMethodSignature (_Jv_Utf8Const*sig); +void _Jv_VerifyClassName (unsigned char* ptr, _Jv_ushort length); +void _Jv_VerifyClassName (_Jv_Utf8Const *name); +void _Jv_VerifyIdentifier (_Jv_Utf8Const *); +bool _Jv_ClassNameSamePackage (_Jv_Utf8Const *name1, _Jv_Utf8Const *name2); +void _Jv_DefineClass (jclass, jbyteArray, jint, jint); +void _Jv_ResolveField (_Jv_Field *, java::lang::ClassLoader*); + +void _Jv_InitField (jobject, jclass, int); +void * _Jv_AllocMethodInvocation (jsize size); + +/* FIXME: this should really be defined in some more generic place */ +#define ROUND(V, A) (((((unsigned) (V))-1) | ((A)-1))+1) + +/* the interpreter is written in C++, primarily because it makes it easy for + * the entire thing to be "friend" with class Class. */ + +class _Jv_InterpClass; +class _Jv_InterpMethod; +class _Jv_InterpMethodInvocation; + +class _Jv_InterpException { + int start_pc; + int end_pc; + int handler_pc; + int handler_type; + + friend class _Jv_ClassReader; + friend class _Jv_InterpMethod; +}; + +class _Jv_InterpMethod { + + _Jv_ushort max_stack; + _Jv_ushort max_locals; + int code_length; + + _Jv_ushort exc_count; + _Jv_ushort args_raw_size; + + _Jv_InterpClass *defining_class; + _Jv_Method *self; + + unsigned char* bytecode () + { + return + ((unsigned char*)this) + + ROUND((sizeof (_Jv_InterpMethod) + + exc_count*sizeof (_Jv_InterpException)), 4); + } + + _Jv_InterpException * exceptions () + { + return (_Jv_InterpException*) (this+1); + } + + static size_t size (int exc_count, int code_length) + { + return + ROUND ((sizeof (_Jv_InterpMethod) + + (exc_count * sizeof (_Jv_InterpException))), 4) + + code_length; + } + + // return the method's invocation pointer (a stub). + void *ncode (); + void continue1 (_Jv_InterpMethodInvocation *inv); + + static void run_normal (ffi_cif*, void*, void**, void*); + static void run_synch_object (ffi_cif*, void*, void**, void*); + static void run_synch_class (ffi_cif*, void*, void**, void*); + + inline jobject run (ffi_cif*, void*, void**, + _Jv_InterpMethodInvocation*); + + bool find_exception (jobject ex, + _Jv_InterpMethodInvocation *inv); + + public: + static void dump_object(jobject o); + + friend class _Jv_ClassReader; + friend class _Jv_InterpMethodInvocation; + friend class gnu::gcj::runtime::MethodInvocation; + + friend void _Jv_PrepareClass(jclass); + + friend void _Jv_callInterpretedMethod (ffi_cif*, + void*, + void **, + void*); +}; + +class _Jv_InterpMethodInvocation { + _Jv_InterpMethod *running; + void **sp; + unsigned char *pc; + void* state[0]; + + void** stack_base () { return &state[0]; } + void** local_base () { return &state[running->max_stack]; } + + friend class _Jv_InterpMethod; +}; + +class _Jv_InterpClass : public java::lang::Class +{ + _Jv_InterpMethod **interpreted_methods; + _Jv_ushort *field_initializers; + + friend class _Jv_ClassReader; + friend class _Jv_InterpMethod; + friend void _Jv_PrepareClass(jclass); + friend void _Jv_InitField (jobject, jclass, int); + friend void* _Jv_MarkObj (void *, void *, void *, void *); +}; + +struct _Jv_ResolvedMethod { + jint stack_item_count; + jint vtable_index; + jclass klass; + _Jv_Method* method; + + // a resolved method holds the cif in-line, so that _Jv_MarkObj just needs + // to mark the resolved method to hold on to the cif. Some memory could be + // saved by keeping a cache of cif's, since many will be the same. + ffi_cif cif; + ffi_type * arg_types[0]; +}; + +#endif /* INTERPRETER */ + +#endif /* __JAVA_INTERP_H__ */ diff --git a/libjava/include/javaprims.h b/libjava/include/javaprims.h index 98781ad51f2..9d8c0a6eab5 100644 --- a/libjava/include/javaprims.h +++ b/libjava/include/javaprims.h @@ -162,6 +162,7 @@ extern "Java" class VerifyError; class VirtualMachineError; class Void; + class VMClassLoader; namespace reflect { class AccessibleObject; diff --git a/libjava/include/jvm.h b/libjava/include/jvm.h index 75ca8274c4d..34da0edbea9 100644 --- a/libjava/include/jvm.h +++ b/libjava/include/jvm.h @@ -40,6 +40,8 @@ extern int _Jv_strLengthUtf8(char* str, int len); typedef struct _Jv_Utf8Const Utf8Const; _Jv_Utf8Const *_Jv_makeUtf8Const (char *s, int len); +_Jv_Utf8Const *_Jv_makeUtf8TypeConst (char* s, int len); +_Jv_Utf8Const *_Jv_makeUtf8Const (jstring string); extern jboolean _Jv_equalUtf8Consts (_Jv_Utf8Const *, _Jv_Utf8Const *); extern jboolean _Jv_equal (_Jv_Utf8Const *, jstring, jint); @@ -91,6 +93,8 @@ extern "C" void *_Jv_LookupInterfaceMethod (jclass klass, Utf8Const *name, extern "C" void _Jv_CheckArrayStore (jobject array, jobject obj); extern "C" void _Jv_RegisterClass (jclass klass); extern "C" void _Jv_RegisterClasses (jclass *classes); +extern void _Jv_UnregisterClass (_Jv_Utf8Const*, java::lang::ClassLoader*); + extern jclass _Jv_FindClass (_Jv_Utf8Const *name, java::lang::ClassLoader *loader); extern jclass _Jv_FindClassFromSignature (char *, diff --git a/libjava/interpret.cc b/libjava/interpret.cc new file mode 100644 index 00000000000..1b9b8006da7 --- /dev/null +++ b/libjava/interpret.cc @@ -0,0 +1,2449 @@ +// interpret.cc - Code for the interpreter + +/* Copyright (C) 1999 Cygnus Solutions + + This file is part of libgcj. + +This software is copyrighted work licensed under the terms of the +Libgcj License. Please consult the file "LIBGCJ_LICENSE" for +details. */ + +/* Author: Kresten Krab Thorup <krab@gnu.org> */ + +/* define this to get instruction timings. */ +/* #define TIME_MAINLOOP */ + +#include <config.h> + +#pragma implementation "java-interp.h" + +#include <cni.h> +#include <jvm.h> +#include <java-field.h> +#include <java-cpool.h> +#include <java-interp.h> +#include <java/lang/fdlibm.h> +#include <java/lang/System.h> +#include <java/lang/String.h> +#include <java/lang/Integer.h> +#include <java/lang/StringBuffer.h> +#include <java/io/PrintStream.h> +#include <java/lang/Class.h> +#include <java/lang/reflect/Modifier.h> +#include <java/lang/ClassCastException.h> +#include <java/lang/VirtualMachineError.h> +#include <java/lang/InternalError.h> +#include <java/lang/NullPointerException.h> +#include <java/lang/ArithmeticException.h> +#include <java/lang/IncompatibleClassChangeError.h> +#include <java-insns.h> +#include <java-signal.h> +#ifdef TIME_MAINLOOP +#include <sys/time.h> +#include <stdio.h> +#endif + +#ifndef INTERPRETER + +#include <gnu/gcj/runtime/MethodInvocation.h> + +/* this is the exception handler hack, for the interpreter */ +void +gnu::gcj::runtime::MethodInvocation::continue1 (gnu::gcj::RawData *, + gnu::gcj::RawData *) +{ + JvFail ("no interpreter"); +} + +#else + +#define ClassError _CL_Q34java4lang5Error +extern java::lang::Class ClassError; + +static const int PUBLIC = 0x001; +static const int PRIVATE = 0x002; +static const int PROTECTED = 0x004; +static const int STATIC = 0x008; +static const int FINAL = 0x010; +static const int SYNCHRONIZED = 0x020; +static const int VOLATILE = 0x040; +static const int TRANSIENT = 0x080; +static const int NATIVE = 0x100; +static const int INTERFACE = 0x200; +static const int ABSTRACT = 0x400; +static const int ALL_FLAGS = 0x7FF; + +static _Jv_Utf8Const *init_name = _Jv_makeUtf8Const ("<init>", 6); + +static void throw_internal_error (char *msg) + __attribute__ ((__noreturn__)); +static void throw_incompatible_class_change_error (jstring msg) + __attribute__ ((__noreturn__)); +#if !HANDLE_SEGV +static void throw_null_pointer_exception () + __attribute__ ((__noreturn__)); +#endif +#if !HANDLE_FPE +static void throw_arithmetic_exception () + __attribute__ ((__noreturn__)); +#endif + + +static inline void dupx (void **&sp, int n, int x) +{ + // first "slide" n+x elements n to the right + int top = n-1; + for (int i = 0; i < n+x; i++) + { + sp[(top-i)] = sp[(top-i)-n]; + } + + // next, copy the n top elements, n+x down + for (int i = 0; i < n; i++) + { + sp[top-(n+x)-i] = sp[top-i]; + } + + // the net effect + sp += n; +}; + + +#define PUSHA(V) \ + ({ jobject __v=(V); *(jobject*)sp++ = __v; }) +#define PUSHI(V) \ + ({ jint __v=(V); *(jint*)sp++ = __v; }) +#define PUSHF(V) \ + ({ jfloat __v=(V); *(jfloat*)sp++ = __v; }) +#define PUSHL(V) \ + ({ jlong __v=(V); memcpy ((void*)sp, (void*)&__v, 8); sp+=2; }) +#define PUSHD(V) \ + ({ jdouble __v=(V); memcpy ((void*)sp, (void*)&__v, 8); sp+=2; }) + +#define POPA() (*(jobject*)--sp) +#define POPI() (*(jint*)--sp) +#define POPF() (*(jfloat*)--sp) +#define POPL() ({ jlong __r; sp-=2; memcpy ((void*)&__r, sp, 8); __r; }) +#define POPD() ({ jdouble __r; sp-=2; memcpy ((void*)&__r, sp, 8); __r; }) + +#define LOADA(I) *sp++ = locals[I] +#define LOADI(I) *sp++ = locals[I] +#define LOADF(I) *sp++ = locals[I] +#define LOADL(I) ({ memcpy (sp, locals+(I), 8); sp+=2; }) +#define LOADD(I) ({ memcpy (sp, locals+(I), 8); sp+=2; }) + +#define STOREA(I) locals[I] = *--sp +#define STOREI(I) locals[I] = *--sp +#define STOREF(I) locals[I] = *--sp +#define STOREL(I) ({ sp-=2; memcpy (locals+(I), sp, 8); }) +#define STORED(I) ({ sp-=2; memcpy (locals+(I), sp, 8); }) + +#define PEEKI(I) (*(jint*) (locals+(I))) +#define PEEKA(I) (*(jobject*) (locals+(I))) + +#define POKEI(I,V) (*(jint*) (locals+(I)) = (V)) + + +#define BINOPI(OP) { \ + jint value2 = POPI(); \ + jint value1 = POPI(); \ + PUSHI(value1 OP value2); \ +} + +#define BINOPF(OP) { \ + jfloat value2 = POPF(); \ + jfloat value1 = POPF(); \ + PUSHF(value1 OP value2); \ +} + +#define BINOPL(OP) { \ + jlong value2 = POPL(); \ + jlong value1 = POPL(); \ + PUSHL(value1 OP value2); \ +} + +#define BINOPD(OP) { \ + jdouble value2 = POPD(); \ + jdouble value1 = POPD(); \ + PUSHD(value1 OP value2); \ +} + +static inline jint get1s(unsigned char* loc) { + return *(signed char*)loc; +} + +static inline jint get1u(unsigned char* loc) { + return *loc; +} + +static inline jint get2s(unsigned char* loc) { + return (((jint)*(signed char*)loc) << 8) | ((jint)*(loc+1)); +} + +static inline jint get2u(unsigned char* loc) { + return (((jint)(*loc)) << 8) | ((jint)*(loc+1)); +} + +static jint get4(unsigned char* loc) { + return (((jint)(loc[0])) << 24) + | (((jint)(loc[1])) << 16) + | (((jint)(loc[2])) << 8) + | (((jint)(loc[3])) << 0); +} + + +#if HANDLE_SEGV +#define NULLCHECK(X) +#else +#define NULLCHECK(X) \ + do { if ((X)==NULL) throw_null_pointer_exception (); } while (0) +#endif + +#if HANDLE_FPE +#define ZEROCHECK(X) +#else +#define ZEROCHECK(X) \ + do { if ((X) == 0) throw_arithmetic_exception (); } while (0) +#endif + +// this method starts the actual running of the method. It is inlined +// in three different variants in the static methods run_normal, +// run_sync_object and run_sync_class (see below). Those static methods +// are installed directly in the stub for this method (by +// _Jv_InterpMethod::ncode, in resolve.cc). + +inline jobject +_Jv_InterpMethod::run (ffi_cif* cif, + void *retp, + void**args, + _Jv_InterpMethodInvocation *inv) +{ + inv->running = this; + inv->pc = bytecode (); + inv->sp = inv->stack_base (); + void **locals = inv->local_base (); + + /* Go straight at it! the ffi raw format matches the internal + stack representation exactly! + */ + memcpy ((void*) locals, (void*) args, args_raw_size); + + next_segment: + /* this will call the method _Jv_InterpMethod::continue0, see below */ + jobject ex = + gnu::gcj::runtime::MethodInvocation::continue0 + ((gnu::gcj::RawData *)this, (gnu::gcj::RawData *)inv); + + if (ex == 0) // no exception... + { + /* define sp locally, so the POP? macros will pick it up */ + void **sp = (void**)inv->sp; + int rtype = cif->rtype->type; + + if (rtype == FFI_TYPE_POINTER) + { + jobject r = POPA(); + *(jobject*) retp = r; + return 0; + } + else if (rtype == FFI_TYPE_SINT32) + { + jint r = POPI(); + *(jint*)retp = r; + return 0; + } + else if (rtype == FFI_TYPE_VOID) + { + return 0; + } + else switch (rtype) + { + case FFI_TYPE_FLOAT: + { + jfloat r = POPF(); + *(jfloat*)retp = r; + return 0; + } + + case FFI_TYPE_DOUBLE: + { + jdouble r = POPD(); + *(jdouble*)retp = r; + return 0; + } + + case FFI_TYPE_UINT8: + case FFI_TYPE_UINT16: + case FFI_TYPE_UINT32: + case FFI_TYPE_SINT8: + case FFI_TYPE_SINT16: + { + jint r = POPI(); + *(jint*)retp = r; + return 0; + } + + case FFI_TYPE_SINT64: + { + jlong r = POPL(); + *(jlong*)retp = r; + return 0; + } + + default: + throw_internal_error ("unknown return type"); + } + + } + + /** handle an exception */ + if ( find_exception (ex, inv) ) + goto next_segment; + + java::lang::System::out->println + (_Jv_NewStringUTF (self->name->data)); + + return ex; +} + +bool _Jv_InterpMethod::find_exception (jobject ex, + _Jv_InterpMethodInvocation *inv) +{ + int logical_pc = inv->pc - bytecode (); + _Jv_InterpException *exc = exceptions (); + jclass exc_class = ex->getClass (); + + for (int i = 0; i < exc_count; i++) + { + if (exc[i].start_pc <= logical_pc && logical_pc < exc[i].end_pc) + { + jclass handler; + + if (exc[i].handler_type != 0) + handler = (jclass) + _Jv_ResolvePoolEntry (defining_class, + exc[i].handler_type); + else + handler = NULL; + + if (handler==NULL || handler->isAssignableFrom (exc_class)) + { + inv->pc = bytecode () + exc[i].handler_pc; + inv->sp = inv->stack_base (); // reset stack + *(jobject*) (inv->sp ++) = ex; + return true; + } + } + } + return false; +} + +void _Jv_InterpMethod::run_normal (ffi_cif* cif, + void* ret, + void** args, + void* __this) +{ + _Jv_InterpMethod* _this = (_Jv_InterpMethod*)__this; + + // we do the alloca of the method invocation here, to allow the method + // "run" ro be inlined. Otherwise gcc will ignore the inline directive. + int storage_size = _this->max_stack+_this->max_locals; + _Jv_InterpMethodInvocation* inv = (_Jv_InterpMethodInvocation*) + alloca (sizeof (_Jv_InterpMethodInvocation) + + storage_size * sizeof (void*)); + + jobject ex = _this->run (cif, ret, args, inv); + if (ex != 0) _Jv_Throw (ex); +} + +void _Jv_InterpMethod::run_synch_object (ffi_cif* cif, + void* ret, + void** args, + void* __this) +{ + _Jv_InterpMethod* _this = (_Jv_InterpMethod*)__this; + jobject rcv = (jobject)args[0]; + + int storage_size = _this->max_stack+_this->max_locals; + _Jv_InterpMethodInvocation* inv = (_Jv_InterpMethodInvocation*) + alloca (sizeof (_Jv_InterpMethodInvocation) + + storage_size * sizeof (void*)); + + _Jv_MonitorEnter (rcv); + jobject ex = _this->run (cif, ret, args, inv); + _Jv_MonitorExit (rcv); + + if (ex != 0) _Jv_Throw (ex); +} + +void _Jv_InterpMethod::run_synch_class (ffi_cif* cif, + void* ret, + void** args, + void* __this) +{ + _Jv_InterpMethod* _this = (_Jv_InterpMethod*)__this; + jclass sync = _this->defining_class; + + int storage_size = _this->max_stack+_this->max_locals; + _Jv_InterpMethodInvocation* inv = (_Jv_InterpMethodInvocation*) + alloca (sizeof (_Jv_InterpMethodInvocation) + + storage_size * sizeof (void*)); + + _Jv_MonitorEnter (sync); + jobject ex = _this->run (cif, ret, args, inv); + _Jv_MonitorExit (sync); + + if (ex != 0) _Jv_Throw (ex); +} + +/* this is the exception handler hack, for the interpreter */ +void +gnu::gcj::runtime::MethodInvocation::continue1 (gnu::gcj::RawData *meth, + gnu::gcj::RawData *inv) +{ + _Jv_InterpMethod *meth0 = (_Jv_InterpMethod*)meth; + _Jv_InterpMethodInvocation *inv0 = (_Jv_InterpMethodInvocation*)inv; + meth0->continue1 (inv0); +} + +/* + This proceeds execution, as designated in "inv". If an exception + happens, then it is simply thrown, and handled in Java. Thus, the pc + needs to be stored in the invocation at all times, so we can figure + out which handler (if any) to invoke. + + One design issue, which I have not completely considered, is if it + should be possible to have interpreted classes linked in! Seldom used + (or non-critical) classes could reasonably be interpreted. +*/ + + +#ifdef TIME_MAINLOOP +static jlong insn_time [256] = { 0 }; +static jlong insn_count[256] = { 0 }; + +static void +dump_time () +{ + double total_all = 0; + for (int i = 0; i < 256; i++) + { + total_all += insn_time[i]; + } + + for (int i = 0; i < 256; i++) + { + jlong total = insn_time[i]; + jlong count = insn_count[i]; + + if (count == 0) continue; + + jlong amount = total/count; + + printf ("in 0x%02x: %7Li %7Li %7Li %2.1f%%\n", i, + (long long)count, (long long)total, (long long)amount, + (float) (100.0*(double)total/total_all) + ); + } +} +#endif + +void _Jv_InterpMethod::continue1 (_Jv_InterpMethodInvocation *inv) +{ + /* for some reason, which I do not understand, the compiler on x86 + * allocates almost 4k stack space for this function! Even though + * there are many local variables, they are all nicely contained + * within a block scope, except for the few declared right below + * here. What's going on?? It could well be, that there in fact is + * on the order of 1000 local variables, including all those inlined + * and expanded from macros... Compiling with -O0, it allocates a + * "modest" 300 bytes of stack space. Among all those options of + * gcc, why isn't there a -fpack-stack, allowing reuse of stack + * locations? */ + + void** sp = inv->sp; + unsigned char *pc = inv->pc; + void** locals = inv->local_base (); + int opcode; + + jclass defining_class = this->defining_class; + void **pool_data = defining_class->constants.data; + + /* these two are used in the invokeXXX instructions */ + void (*fun)(...); + _Jv_ResolvedMethod* rmeth; + +#ifdef TIME_MAINLOOP + struct timeval tv; + int last_opcode; + jlong last_time; + static jlong time_warp = 0; + +#define USEC(TV) \ + ((jlong) (TV).tv_sec * 1000000LL + (jlong)(TV).tv_usec) + + + if (time_warp == 0) + { + struct timeval tv2; + + gettimeofday (&tv, 0); + for (int i = 0; i < 100; i++) + gettimeofday (&tv2, 0); + + jlong then = USEC(tv); + jlong now = USEC(tv2); + time_warp = (now - then) / 100; + + if (time_warp == 0) + time_warp = 1; + } + +#define TIME_SUSPEND do { \ + gettimeofday (&tv, 0); \ + jlong now = USEC(tv); \ + insn_time[last_opcode] += (now - last_time) - time_warp; \ +} while(0) + +#define TIME_RESUME do { \ + gettimeofday (&tv, 0); \ + last_time = USEC(tv); \ +} while(0) + + last_opcode = 0; + gettimeofday (&tv, 0); + last_time = (jlong)tv.tv_sec * 1000000LL + (jlong)tv.tv_usec; + +#else + +#define TIME_SUSPEND +#define TIME_RESUME + +#endif + + next_insn: + inv->pc = pc; + +#ifdef TIME_MAINLOOP + + gettimeofday (&tv, 0); + jlong now = USEC(tv); + insn_time[last_opcode] += (now - last_time) - time_warp; + last_time = now; + last_opcode = *pc; + insn_count[last_opcode] += 1; + +#endif + opcode = *pc++; + + /* we special-case the single opcode aload_0 -- it makes + up 10% of the time spent in the main loop. */ + + switch (opcode) + { + case op_aload_0: // 0x2a + LOADA(0); + goto next_insn; + + case op_iload: // 0x15 + LOADI (get1u (pc++)); + goto next_insn; + + case op_getfield_4: // 0xd8 + { + jobject obj = POPA(); + NULLCHECK(obj); + jint field_offset = get2u (pc); pc += 2; + PUSHI (*(jint*) ((char*)obj + field_offset)); + } + goto next_insn; + + case op_iload_1: // 0x1b + LOADI (1); + goto next_insn; + + case op_getfield_a: // 0xda + { + jobject obj = POPA(); + NULLCHECK(obj); + jint field_offset = get2u (pc); pc += 2; + PUSHA(*(jobject*) ((char*)obj + field_offset)); + } + goto next_insn; + + case op_invokevirtual: // 0xb6 + { + int index = get2u (pc); pc += 2; + + /* _Jv_ResolvePoolEntry returns immediately if the value already + * is resolved. If we want to clutter up the code here to gain + * a little performance, then we can check the corresponding bit + * JV_CONSTANT_ResolvedFlag in the tag directly. For now, I + * don't think it is worth it. */ + + rmeth = (_Jv_ResolvedMethod*) + _Jv_ResolvePoolEntry (defining_class, index); + + sp -= rmeth->stack_item_count; + NULLCHECK(sp[0]); + + if (rmeth->vtable_index == -1) + { + // final methods do not appear in the vtable, + // if it does not appear in the superclass. + fun = (void (*) (...)) rmeth->method->ncode; + } + else + { + jobject rcv = (jobject)sp[0]; + _Jv_VTable *table = *(_Jv_VTable**)rcv; + fun = (void (*) (...))table->method[rmeth->vtable_index]; + } + } + goto perform_invoke; + + perform_invoke: + { + /* here goes the magic again... */ + ffi_cif *cif = &rmeth->cif; + void **raw = sp; + + jdouble rvalue; + + TIME_SUSPEND; + ffi_raw_call (cif, fun, (void*)&rvalue, (ffi_raw*) raw); + TIME_RESUME; + + int rtype = cif->rtype->type; + + /* the likelyhood of object, int, or void return is very high, + * so those are checked before the switch */ + if (rtype == FFI_TYPE_POINTER) + { + PUSHA (*(jobject*)&rvalue); + } + else if (rtype == FFI_TYPE_SINT32) + { + PUSHI (*(jint*)&rvalue); + } + else if (rtype == FFI_TYPE_VOID) + { + /* skip */ + } + else switch (rtype) + { + case FFI_TYPE_SINT8: + { + jbyte value = (*(jint*)&rvalue) & 0xff; + PUSHI (value); + } + break; + + case FFI_TYPE_SINT16: + { + jshort value = (*(jint*)&rvalue) & 0xffff; + PUSHI (value); + } + break; + + case FFI_TYPE_UINT16: + { + jint value = (*(jint*)&rvalue) & 0xffff; + PUSHI (value); + } + break; + + case FFI_TYPE_FLOAT: + PUSHF (*(jfloat*)&rvalue); + break; + + case FFI_TYPE_DOUBLE: + PUSHD (rvalue); + break; + + case FFI_TYPE_SINT64: + PUSHL (*(jlong*)&rvalue); + break; + + default: + throw_internal_error ("unknown return type in invokeXXX"); + } + + } + goto next_insn; + + + case op_nop: + goto next_insn; + + case op_aconst_null: + PUSHA (NULL); + goto next_insn; + + case op_iconst_m1: + case op_iconst_0: + case op_iconst_1: + case op_iconst_2: + case op_iconst_3: + case op_iconst_4: + case op_iconst_5: + PUSHI (opcode-op_iconst_0); + goto next_insn; + + case op_lconst_0: + case op_lconst_1: + PUSHL ((jlong) (opcode-op_lconst_0)); + goto next_insn; + + case op_fconst_0: + case op_fconst_1: + case op_fconst_2: + PUSHF ((jfloat) (opcode-op_fconst_0)); + goto next_insn; + + case op_dconst_0: + case op_dconst_1: + PUSHD ((jdouble) (opcode-op_dconst_0)); + goto next_insn; + + case op_bipush: + PUSHI (get1s(pc++)); + goto next_insn; + + case op_sipush: + PUSHI (get2s(pc)); pc += 2; + goto next_insn; + + case op_ldc: + { + int index = get1u (pc++); + PUSHA((jobject) pool_data[index]); + } + goto next_insn; + + case op_ldc_w: + { + int index = get2u (pc); pc += 2; + PUSHA((jobject) pool_data[index]); + } + goto next_insn; + + case op_ldc2_w: + { + int index = get2u (pc); pc += 2; + memcpy (sp, &pool_data[index], 8); + sp += 2; + } + goto next_insn; + + case op_lload: + LOADL (get1u (pc++)); + goto next_insn; + + case op_fload: + LOADF (get1u (pc++)); + goto next_insn; + + case op_dload: + LOADD (get1u (pc++)); + goto next_insn; + + case op_aload: + LOADA (get1u (pc++)); + goto next_insn; + + case op_iload_0: + LOADI (0); + goto next_insn; + + case op_iload_2: + LOADI (2); + goto next_insn; + + case op_iload_3: + LOADI (3); + goto next_insn; + + case op_lload_0: + case op_lload_1: + case op_lload_2: + case op_lload_3: + LOADL (opcode-op_lload_0); + goto next_insn; + + case op_fload_0: + case op_fload_1: + case op_fload_2: + case op_fload_3: + LOADF (opcode-op_fload_0); + goto next_insn; + + case op_dload_0: + case op_dload_1: + case op_dload_2: + case op_dload_3: + LOADD (opcode-op_dload_0); + goto next_insn; + + case op_aload_1: + LOADA(1); + goto next_insn; + + case op_aload_2: + LOADA(2); + goto next_insn; + + case op_aload_3: + LOADA(3); + goto next_insn; + + case op_iaload: + { + jint index = POPI(); + jintArray arr = (jintArray) POPA(); + NULLCHECK (arr); + if (index < 0 || index >= arr->length) + { + TIME_SUSPEND; + _Jv_ThrowBadArrayIndex (index); + } + PUSHI( elements(arr)[index] ); + } + goto next_insn; + + case op_laload: + { + jint index = POPI(); + jlongArray arr = (jlongArray) POPA(); + NULLCHECK (arr); + if (index < 0 || index >= arr->length) + { + TIME_SUSPEND; + _Jv_ThrowBadArrayIndex (index); + } + PUSHL( elements(arr)[index] ); + } + goto next_insn; + + case op_faload: + { + jint index = POPI(); + jfloatArray arr = (jfloatArray) POPA(); + NULLCHECK (arr); + if (index < 0 || index >= arr->length) + { + TIME_SUSPEND; + _Jv_ThrowBadArrayIndex (index); + } + PUSHF( elements(arr)[index] ); + } + goto next_insn; + + case op_daload: + { + jint index = POPI(); + jdoubleArray arr = (jdoubleArray) POPA(); + NULLCHECK (arr); + if (index < 0 || index >= arr->length) + { + TIME_SUSPEND; + _Jv_ThrowBadArrayIndex (index); + } + PUSHD( elements(arr)[index] ); + } + goto next_insn; + + case op_aaload: + { + jint index = POPI(); + jobjectArray arr = (jobjectArray) POPA(); + NULLCHECK (arr); + if (index < 0 || index >= arr->length) + { + TIME_SUSPEND; + _Jv_ThrowBadArrayIndex (index); + } + PUSHA( elements(arr)[index] ); + } + goto next_insn; + + case op_baload: + { + jint index = POPI(); + jbyteArray arr = (jbyteArray) POPA(); + NULLCHECK (arr); + if (index < 0 || index >= arr->length) + { + TIME_SUSPEND; + _Jv_ThrowBadArrayIndex (index); + } + PUSHI( elements(arr)[index] ); + } + goto next_insn; + + case op_caload: + { + jint index = POPI(); + jcharArray arr = (jcharArray) POPA(); + NULLCHECK (arr); + if (index < 0 || index >= arr->length) + { + TIME_SUSPEND; + _Jv_ThrowBadArrayIndex (index); + } + PUSHI( elements(arr)[index] ); + } + goto next_insn; + + case op_saload: + { + jint index = POPI(); + jshortArray arr = (jshortArray) POPA(); + NULLCHECK (arr); + if (index < 0 || index >= arr->length) + { + TIME_SUSPEND; + _Jv_ThrowBadArrayIndex (index); + } + PUSHI( elements(arr)[index] ); + } + goto next_insn; + + case op_istore: + STOREI (get1u (pc++)); + goto next_insn; + + case op_lstore: + STOREL (get1u (pc++)); + goto next_insn; + + case op_fstore: + STOREF (get1u (pc++)); + goto next_insn; + + case op_dstore: + STORED (get1u (pc++)); + goto next_insn; + + case op_astore: + STOREI (get1u (pc++)); + goto next_insn; + + case op_istore_0: + case op_istore_1: + case op_istore_2: + case op_istore_3: + STOREI (opcode-op_istore_0); + goto next_insn; + + case op_lstore_0: + case op_lstore_1: + case op_lstore_2: + case op_lstore_3: + STOREL (opcode-op_lstore_0); + goto next_insn; + + case op_fstore_0: + case op_fstore_1: + case op_fstore_2: + case op_fstore_3: + STOREF (opcode-op_fstore_0); + goto next_insn; + + case op_dstore_0: + case op_dstore_1: + case op_dstore_2: + case op_dstore_3: + STORED (opcode-op_dstore_0); + goto next_insn; + + case op_astore_0: + case op_astore_1: + case op_astore_2: + case op_astore_3: + STOREA (opcode-op_astore_0); + goto next_insn; + + case op_iastore: + { + jint value = POPI(); + jint index = POPI(); + jintArray arr = (jintArray) POPA(); + NULLCHECK (arr); + if (index < 0 || index >= arr->length) + { + TIME_SUSPEND; + _Jv_ThrowBadArrayIndex (index); + } + elements(arr)[index] = value; + } + goto next_insn; + + case op_lastore: + { + jlong value = POPL(); + jint index = POPI(); + jlongArray arr = (jlongArray) POPA(); + NULLCHECK (arr); + if (index < 0 || index >= arr->length) + { + TIME_SUSPEND; + _Jv_ThrowBadArrayIndex (index); + } + elements(arr)[index] = value; + } + goto next_insn; + + case op_fastore: + { + jfloat value = POPF(); + jint index = POPI(); + jfloatArray arr = (jfloatArray) POPA(); + NULLCHECK (arr); + if (index < 0 || index >= arr->length) + { + TIME_SUSPEND; + _Jv_ThrowBadArrayIndex (index); + } + elements(arr)[index] = value; + } + goto next_insn; + + case op_dastore: + { + jdouble value = POPD(); + jint index = POPI(); + jdoubleArray arr = (jdoubleArray) POPA(); + NULLCHECK (arr); + if (index < 0 || index >= arr->length) + { + TIME_SUSPEND; + _Jv_ThrowBadArrayIndex (index); + } + elements(arr)[index] = value; + } + goto next_insn; + + case op_aastore: + { + jobject value = POPA(); + jint index = POPI(); + jobjectArray arr = (jobjectArray) POPA(); + NULLCHECK (arr); + if (index < 0 || index >= arr->length) + { + TIME_SUSPEND; + _Jv_ThrowBadArrayIndex (index); + } + _Jv_CheckArrayStore (arr, value); + elements(arr)[index] = value; + } + goto next_insn; + + case op_bastore: + { + jbyte value = (jbyte) POPI(); + jint index = POPI(); + jbyteArray arr = (jbyteArray) POPA(); + NULLCHECK (arr); + if (index < 0 || index >= arr->length) + { + TIME_SUSPEND; + _Jv_ThrowBadArrayIndex (index); + } + elements(arr)[index] = value; + } + goto next_insn; + + case op_castore: + { + jchar value = (jchar) POPI(); + jint index = POPI(); + jcharArray arr = (jcharArray) POPA(); + NULLCHECK (arr); + if (index < 0 || index >= arr->length) + { + TIME_SUSPEND; + _Jv_ThrowBadArrayIndex (index); + } + elements(arr)[index] = value; + } + goto next_insn; + + case op_sastore: + { + jshort value = (jshort) POPI(); + jint index = POPI(); + jshortArray arr = (jshortArray) POPA(); + NULLCHECK (arr); + if (index < 0 || index >= arr->length) + { + TIME_SUSPEND; + _Jv_ThrowBadArrayIndex (index); + } + elements(arr)[index] = value; + } + goto next_insn; + + case op_pop: + sp -= 1; + goto next_insn; + + case op_pop2: + sp -= 2; + goto next_insn; + + case op_dup: + sp[0] = sp[-1]; + sp += 1; + goto next_insn; + + case op_dup_x1: + dupx (sp, 1, 1); + goto next_insn; + + case op_dup_x2: + dupx (sp, 1, 2); + goto next_insn; + + case op_dup2: + sp[0] = sp[-2]; + sp[1] = sp[-1]; + sp += 2; + goto next_insn; + + case op_dup2_x1: + dupx (sp, 2, 1); + goto next_insn; + + case op_dup2_x2: + dupx (sp, 2, 2); + goto next_insn; + + case op_swap: + { + jobject tmp1 = POPA(); + jobject tmp2 = POPA(); + PUSHA (tmp1); + PUSHA (tmp2); + } + goto next_insn; + + case op_iadd: + BINOPI(+); + goto next_insn; + + case op_ladd: + BINOPL(+); + goto next_insn; + + case op_fadd: + BINOPF(+); + goto next_insn; + + case op_dadd: + BINOPD(+); + goto next_insn; + + case op_isub: + BINOPI(-); + goto next_insn; + + case op_lsub: + BINOPL(-); + goto next_insn; + + case op_fsub: + BINOPF(-); + goto next_insn; + + case op_dsub: + BINOPD(-); + goto next_insn; + + case op_imul: + BINOPI(*); + goto next_insn; + + case op_lmul: + BINOPL(*); + goto next_insn; + + case op_fmul: + BINOPF(*); + goto next_insn; + + case op_dmul: + BINOPD(*); + goto next_insn; + + case op_idiv: + { + jint value2 = POPI(); + jint value1 = POPI(); + ZEROCHECK (value2); + jint res = value1 / value2; + PUSHI (res); + } + goto next_insn; + + case op_ldiv: + { + jlong value2 = POPL(); + jlong value1 = POPL(); + ZEROCHECK (value2); + jlong res = value1 / value2; + PUSHL (res); + } + goto next_insn; + + case op_fdiv: + { + jfloat value2 = POPF(); + jfloat value1 = POPF(); + ZEROCHECK (value2); + jfloat res = value1 / value2; + PUSHF (res); + } + goto next_insn; + + case op_ddiv: + { + jdouble value2 = POPD(); + jdouble value1 = POPD(); + ZEROCHECK (value2); + jdouble res = value1 / value2; + PUSHD (res); + } + goto next_insn; + + case op_irem: + { + jint value2 = POPI(); + jint value1 = POPI(); + ZEROCHECK (value2); + jint res = value1 % value2; + PUSHI (res); + } + goto next_insn; + + case op_lrem: + { + jlong value2 = POPL(); + jlong value1 = POPL(); + ZEROCHECK (value2); + jlong res = value1 % value2; + PUSHL (res); + } + goto next_insn; + + case op_frem: + { + jfloat value2 = POPF(); + jfloat value1 = POPF(); + ZEROCHECK (value2); + jfloat res = __ieee754_fmod (value1, value2); + PUSHF (res); + } + goto next_insn; + + case op_drem: + { + jdouble value2 = POPD(); + jdouble value1 = POPD(); + ZEROCHECK (value2); + jdouble res = __ieee754_fmod (value1, value2); + PUSHD (res); + } + goto next_insn; + + case op_ineg: + *(jint*) (sp-1) *= -1; + goto next_insn; + + case op_lneg: + *(jlong*) (sp-1) *= -1; + goto next_insn; + + case op_fneg: + *(jfloat*) (sp-1) *= -1; + goto next_insn; + + case op_dneg: + *(jdouble*) (sp-1) *= -1; + goto next_insn; + + case op_ishl: + { + jint shift = (POPI() & 0x1f); + jint value = POPI(); + PUSHI (value << shift); + } + goto next_insn; + + case op_lshl: + { + jint shift = (POPI() & 0x3f); + jlong value = POPL(); + PUSHL (value << shift); + } + goto next_insn; + + case op_ishr: + { + jint shift = (POPI() & 0x1f); + jint value = POPI(); + PUSHI (value >> shift); + } + goto next_insn; + + case op_lshr: + { + jint shift = (POPI() & 0x3f); + jlong value = POPL(); + PUSHL (value >> shift); + } + goto next_insn; + + case op_iushr: + { + jint shift = (POPI() & 0x1f); + unsigned long value = POPI(); + PUSHI ((jint) (value >> shift)); + } + goto next_insn; + + case op_lushr: + { + jint shift = (POPI() & 0x3f); + UINT64 value = (UINT64) POPL(); + PUSHL ((value >> shift)); + } + goto next_insn; + + case op_iand: + BINOPI (&); + goto next_insn; + + case op_land: + BINOPL (&); + goto next_insn; + + case op_ior: + BINOPI (|); + goto next_insn; + + case op_lor: + BINOPL (|); + goto next_insn; + + case op_ixor: + BINOPI (^); + goto next_insn; + + case op_lxor: + BINOPL (^); + goto next_insn; + + case op_iinc: + { + jint index = get1u (pc++); + jint amount = get1s (pc++); + *(jint*) (locals + index) += amount; + } + goto next_insn; + + case op_i2l: + PUSHL ((jlong)POPI ()); + goto next_insn; + + case op_i2f: + PUSHF ((jfloat)POPI ()); + goto next_insn; + + case op_i2d: + PUSHD ((jdouble)POPI ()); + goto next_insn; + + case op_l2i: + PUSHI ((jint)POPL ()); + goto next_insn; + + case op_l2f: + PUSHF ((jfloat)POPL ()); + goto next_insn; + + case op_l2d: + PUSHD ((jdouble)POPL ()); + goto next_insn; + + case op_f2i: + PUSHI ((jint)POPF ()); + goto next_insn; + + case op_f2l: + PUSHL ((jlong)POPF ()); + goto next_insn; + + case op_f2d: + PUSHD ((jdouble)POPF ()); + goto next_insn; + + case op_d2i: + PUSHI ((jint)POPD ()); + goto next_insn; + + case op_d2l: + PUSHL ((jlong)POPD ()); + goto next_insn; + + case op_d2f: + PUSHF ((jfloat)POPD ()); + goto next_insn; + + case op_i2b: + PUSHI ((jbyte)POPI ()); + goto next_insn; + + case op_i2c: + PUSHI ((jchar)POPI ()); + goto next_insn; + + case op_i2s: + PUSHI ((jshort)POPI ()); + goto next_insn; + + case op_lcmp: + { + jlong value2 = POPL (); + jlong value1 = POPL (); + if (value1 > value2) + { PUSHI (1); } + else if (value1 == value2) + { PUSHI (0); } + else + { PUSHI (-1); } + } + goto next_insn; + + case op_fcmpl: + case op_fcmpg: + { + jfloat value2 = POPF (); + jfloat value1 = POPF (); + if (value1 > value2) + PUSHI (1); + else if (value1 == value2) + PUSHI (0); + else if (value1 < value2) + PUSHI (-1); + else if (opcode == op_fcmpg) + PUSHI (1); + else + PUSHI (-1); + } + goto next_insn; + + case op_dcmpl: + case op_dcmpg: + { + jdouble value2 = POPD (); + jdouble value1 = POPD (); + if (value1 > value2) + PUSHI (1); + else if (value1 == value2) + PUSHI (0); + else if (value1 < value2) + PUSHI (-1); + else if (opcode == op_dcmpg) + PUSHI (1); + else + PUSHI (-1); + } + goto next_insn; + + case op_ifeq: + { + jint offset = get2s (pc); + if (POPI() == 0) + pc = pc-1+offset; + else + pc = pc+2; + } + goto next_insn; + + case op_ifne: + { + jint offset = get2s (pc); + if (POPI() != 0) + pc = pc-1+offset; + else + pc = pc+2; + } + goto next_insn; + + case op_iflt: + { + jint offset = get2s (pc); + if (POPI() < 0) + pc = pc-1+offset; + else + pc = pc+2; + } + goto next_insn; + + case op_ifge: + { + jint offset = get2s (pc); + if (POPI() >= 0) + pc = pc-1+offset; + else + pc = pc+2; + } + goto next_insn; + + case op_ifgt: + { + jint offset = get2s (pc); + if (POPI() > 0) + pc = pc-1+offset; + else + pc = pc+2; + } + goto next_insn; + + case op_ifle: + { + jint offset = get2s (pc); + if (POPI() <= 0) + pc = pc-1+offset; + else + pc = pc+2; + } + goto next_insn; + + case op_if_icmpeq: + { + jint offset = get2s (pc); + jint value2 = POPI(); + jint value1 = POPI(); + if (value1 == value2) + pc = pc-1+offset; + else + pc = pc+2; + } + goto next_insn; + + case op_if_icmpne: + { + jint offset = get2s (pc); + jint value2 = POPI(); + jint value1 = POPI(); + if (value1 != value2) + pc = pc-1+offset; + else + pc = pc+2; + } + goto next_insn; + + case op_if_icmplt: + { + jint offset = get2s (pc); + jint value2 = POPI(); + jint value1 = POPI(); + if (value1 < value2) + pc = pc-1+offset; + else + pc = pc+2; + } + goto next_insn; + + case op_if_icmpge: + { + jint offset = get2s (pc); + jint value2 = POPI(); + jint value1 = POPI(); + if (value1 >= value2) + pc = pc-1+offset; + else + pc = pc+2; + } + goto next_insn; + + case op_if_icmpgt: + { + jint offset = get2s (pc); + jint value2 = POPI(); + jint value1 = POPI(); + if (value1 > value2) + pc = pc-1+offset; + else + pc = pc+2; + } + goto next_insn; + + case op_if_icmple: + { + jint offset = get2s (pc); + jint value2 = POPI(); + jint value1 = POPI(); + if (value1 <= value2) + pc = pc-1+offset; + else + pc = pc+2; + } + goto next_insn; + + case op_if_acmpeq: + { + jint offset = get2s (pc); + jobject value2 = POPA(); + jobject value1 = POPA(); + if (value1 == value2) + pc = pc-1+offset; + else + pc = pc+2; + } + goto next_insn; + + case op_if_acmpne: + { + jint offset = get2s (pc); + jobject value2 = POPA(); + jobject value1 = POPA(); + if (value1 != value2) + pc = pc-1+offset; + else + pc = pc+2; + } + goto next_insn; + + case op_goto: + { + jint offset = get2s (pc); + pc = pc-1+offset; + } + goto next_insn; + + case op_jsr: + { + unsigned char *base_pc = pc-1; + jint offset = get2s (pc); pc += 2; + PUSHA ((jobject)pc); + pc = base_pc+offset; + } + goto next_insn; + + case op_ret: + { + jint index = get1u (pc); + pc = (unsigned char*) PEEKA (index); + } + goto next_insn; + + case op_tableswitch: + { + unsigned char *base_pc = pc-1; + int index = POPI(); + + unsigned char* base = bytecode (); + while ((pc-base) % 4 != 0) + pc++; + + jint def = get4 (pc); + jint low = get4 (pc+4); + jint high = get4 (pc+8); + + if (index < low || index > high) + pc = base_pc + def; + else + pc = base_pc + get4 (pc+4*(index-low+3)); + } + goto next_insn; + + case op_lookupswitch: + { + unsigned char *base_pc = pc-1; + int index = POPI(); + + unsigned char* base = bytecode (); + while ((pc-base) % 4 != 0) + pc++; + + jint def = get4 (pc); + jint npairs = get4 (pc+4); + + int max = npairs-1; + int min = 0; + + // simple binary search... + while (min < max) + { + int half = (min+max)/2; + int match = get4 (pc+ 4*(2 + 2*half)); + + if (index == match) + min = max = half; + + else if (index < match) + max = half-1; + + else + min = half+1; + } + + if (index == get4 (pc+ 4*(2 + 2*min))) + pc = base_pc + get4 (pc+ 4*(2 + 2*min + 1)); + else + pc = base_pc + def; + } + goto next_insn; + + /* on return, just save the sp and return to caller */ + case op_ireturn: + case op_lreturn: + case op_freturn: + case op_dreturn: + case op_areturn: + case op_return: + inv->sp = sp; + TIME_SUSPEND; + return; + + case op_getstatic: + { + unsigned char *base_pc = pc-1; + jint fieldref_index = get2u (pc); pc += 2; + _Jv_ResolvePoolEntry (defining_class, fieldref_index); + _Jv_Field *field = (_Jv_Field*) pool_data[fieldref_index]; + + if ((field->flags & STATIC) == 0) + throw_incompatible_class_change_error + (JvNewStringLatin1 ("field no longer static")); + + jclass type = field->type; + + if (type->isPrimitive ()) + { + switch (type->size_in_bytes) + { + case 1: + *base_pc = op_getstatic_1; + break; + + case 2: + if (type == JvPrimClass (char)) + *base_pc = op_getstatic_2u; + else + *base_pc = op_getstatic_2s; + break; + + case 4: + *base_pc = op_getstatic_4; + break; + + case 8: + *base_pc = op_getstatic_8; + break; + } + } + else + { + *base_pc = op_getstatic_a; + } + + pc = base_pc; + } + goto next_insn; + + case op_getfield: + { + unsigned char *base_pc = pc-1; + jint fieldref_index = get2u (pc); pc += 2; + _Jv_ResolvePoolEntry (defining_class, fieldref_index); + _Jv_Field *field = (_Jv_Field*) pool_data[fieldref_index]; + + if ((field->flags & STATIC) != 0) + throw_incompatible_class_change_error + (JvNewStringLatin1 ("field is static")); + + jclass type = field->type; + + if (type->isPrimitive ()) + { + switch (type->size_in_bytes) + { + case 1: + *base_pc = op_getfield_1; + break; + + case 2: + if (type == JvPrimClass (char)) + *base_pc = op_getfield_2u; + else + *base_pc = op_getfield_2s; + break; + + case 4: + *base_pc = op_getfield_4; + break; + + case 8: + *base_pc = op_getfield_8; + break; + } + } + else + { + *base_pc = op_getfield_a; + } + + if (field->u.boffset > 0xffff) + JvThrow (new java::lang::VirtualMachineError); + + base_pc[1] = (field->u.boffset>>8) & 0xff; + base_pc[2] = field->u.boffset & 0xff; + + pc = base_pc; + } + goto next_insn; + + case op_putstatic: + { + unsigned char* base_pc = pc-1; + jint fieldref_index = get2u (pc); pc += 2; + _Jv_ResolvePoolEntry (defining_class, fieldref_index); + _Jv_Field *field = (_Jv_Field*) pool_data[fieldref_index]; + + jclass type = field->type; + + // ResolvePoolEntry cannot check this + if ((field->flags & STATIC) == 0) + throw_incompatible_class_change_error + (JvNewStringLatin1 ("field no longer static")); + + /* if this is patented, then maybe we could install + a function in the constant pool, to do the right thing */ + + if (type->isPrimitive ()) + { + switch (type->size_in_bytes) + { + case 1: + *base_pc = op_putstatic_1; + break; + + case 2: + *base_pc = op_putstatic_2; + break; + + case 4: + *base_pc = op_putstatic_4; + break; + + case 8: + *base_pc = op_putstatic_8; + break; + } + } + else + { + *base_pc = op_putstatic_a; + } + + // do the instruction again! + pc = base_pc; + } + goto next_insn; + + + case op_putfield: + { + unsigned char* base_pc = pc-1; + jint fieldref_index = get2u (pc); pc += 2; + _Jv_ResolvePoolEntry (defining_class, fieldref_index); + _Jv_Field *field = (_Jv_Field*) pool_data[fieldref_index]; + + jclass type = field->type; + + if ((field->flags & STATIC) != 0) + throw_incompatible_class_change_error + (JvNewStringLatin1 ("field is static")); + + if (type->isPrimitive ()) + { + switch (type->size_in_bytes) + { + case 1: + *base_pc = op_putfield_1; + break; + + case 2: + *base_pc = op_putfield_2; + break; + + case 4: + *base_pc = op_putfield_4; + break; + + case 8: + *base_pc = op_putfield_8; + break; + } + } + else + { + *base_pc = op_putfield_a; + } + + if (field->u.boffset > 0xffff) + JvThrow (new java::lang::VirtualMachineError); + + base_pc[1] = (field->u.boffset>>8) & 0xff; + base_pc[2] = field->u.boffset & 0xff; + + // do the instruction again! + pc = base_pc; + } + goto next_insn; + + + case op_getfield_1: + { + jobject obj = POPA(); + NULLCHECK(obj); + jint field_offset = get2u (pc); pc += 2; + PUSHI (*(jbyte*) ((char*)obj + field_offset)); + } + goto next_insn; + + case op_getfield_2s: + { + jobject obj = POPA(); + NULLCHECK(obj); + jint field_offset = get2u (pc); pc += 2; + PUSHI (*(jshort*) ((char*)obj + field_offset)); + } + goto next_insn; + + case op_getfield_2u: + { + jobject obj = POPA(); + NULLCHECK(obj); + jint field_offset = get2u (pc); pc += 2; + PUSHI (*(jchar*) ((char*)obj + field_offset)); + } + goto next_insn; + + case op_getfield_8: + { + jobject obj = POPA(); + NULLCHECK(obj); + jint field_offset = get2u (pc); pc += 2; + PUSHL(*(jlong*) ((char*)obj + field_offset)); + } + goto next_insn; + + case op_getstatic_1: + { + jint fieldref_index = get2u (pc); pc += 2; + _Jv_Field *field = (_Jv_Field*) pool_data[fieldref_index]; + PUSHI (*(jbyte*) (field->u.addr)); + } + goto next_insn; + + case op_getstatic_2s: + { + jint fieldref_index = get2u (pc); pc += 2; + _Jv_Field *field = (_Jv_Field*) pool_data[fieldref_index]; + PUSHI(*(jshort*) (field->u.addr)); + } + goto next_insn; + + case op_getstatic_2u: + { + jint fieldref_index = get2u (pc); pc += 2; + _Jv_Field *field = (_Jv_Field*) pool_data[fieldref_index]; + PUSHI(*(jchar*) (field->u.addr)); + } + goto next_insn; + + case op_getstatic_4: + { + jint fieldref_index = get2u (pc); pc += 2; + _Jv_Field *field = (_Jv_Field*) pool_data[fieldref_index]; + PUSHI(*(jint*) (field->u.addr)); + } + goto next_insn; + + case op_getstatic_8: + { + jint fieldref_index = get2u (pc); pc += 2; + _Jv_Field *field = (_Jv_Field*) pool_data[fieldref_index]; + PUSHL(*(jlong*) (field->u.addr)); + } + goto next_insn; + + case op_getstatic_a: + { + jint fieldref_index = get2u (pc); pc += 2; + _Jv_Field *field = (_Jv_Field*) pool_data[fieldref_index]; + PUSHA(*(jobject*) (field->u.addr)); + } + goto next_insn; + + case op_putfield_1: + { + jint value = POPI(); + jobject obj = POPA(); + NULLCHECK(obj); + jint field_offset = get2u (pc); pc += 2; + *(jbyte*) ((char*)obj + field_offset) = value; + } + goto next_insn; + + case op_putfield_2: + { + jint value = POPI(); + jobject obj = POPA(); + NULLCHECK(obj); + jint field_offset = get2u (pc); pc += 2; + *(jchar*) ((char*)obj + field_offset) = value; + } + goto next_insn; + + case op_putfield_4: + { + jint value = POPI(); + jobject obj = POPA(); + NULLCHECK(obj); + jint field_offset = get2u (pc); pc += 2; + *(jint*) ((char*)obj + field_offset) = value; + } + goto next_insn; + + case op_putfield_8: + { + jlong value = POPL(); + jobject obj = POPA(); + NULLCHECK(obj); + jint field_offset = get2u (pc); pc += 2; + *(jlong*) ((char*)obj + field_offset) = value; + } + goto next_insn; + + case op_putfield_a: + { + jobject value = POPA(); + jobject obj = POPA(); + NULLCHECK(obj); + jint field_offset = get2u (pc); pc += 2; + *(jobject*) ((char*)obj + field_offset) = value; + } + goto next_insn; + + case op_putstatic_1: + { + jint value = POPI(); + jint fieldref_index = get2u (pc); pc += 2; + _Jv_Field *field = (_Jv_Field*) pool_data[fieldref_index]; + *(jbyte*) (field->u.addr) = value; + } + goto next_insn; + + case op_putstatic_2: + { + jint value = POPI(); + jint fieldref_index = get2u (pc); pc += 2; + _Jv_Field *field = (_Jv_Field*) pool_data[fieldref_index]; + *(jchar*) (field->u.addr) = value; + } + goto next_insn; + + case op_putstatic_4: + { + jint value = POPI(); + jint fieldref_index = get2u (pc); pc += 2; + _Jv_Field *field = (_Jv_Field*) pool_data[fieldref_index]; + *(jint*) (field->u.addr) = value; + } + goto next_insn; + + case op_putstatic_8: + { + jlong value = POPL(); + jint fieldref_index = get2u (pc); pc += 2; + _Jv_Field *field = (_Jv_Field*) pool_data[fieldref_index]; + *(jlong*) (field->u.addr) = value; + } + goto next_insn; + + case op_putstatic_a: + { + jobject value = POPA(); + jint fieldref_index = get2u (pc); pc += 2; + _Jv_Field *field = (_Jv_Field*) pool_data[fieldref_index]; + *(jobject*) (field->u.addr) = value; + } + goto next_insn; + + case op_invokespecial: + { + int index = get2u (pc); pc += 2; + + rmeth = (_Jv_ResolvedMethod*) + _Jv_ResolvePoolEntry (defining_class, index); + + sp -= rmeth->stack_item_count; + + NULLCHECK(sp[0]); + + fun = (void (*) (...))rmeth->method->ncode; + } + goto perform_invoke; + + case op_invokestatic: + { + int index = get2u (pc); pc += 2; + + rmeth = (_Jv_ResolvedMethod*) + _Jv_ResolvePoolEntry (defining_class, index); + + sp -= rmeth->stack_item_count; + + _Jv_InitClass (rmeth->klass); + fun = (void (*) (...))rmeth->method->ncode; + } + goto perform_invoke; + + case op_invokeinterface: + { + int index = get2u (pc); pc += 2; + + // invokeinterface has two unused bytes... + pc += 2; + + rmeth = (_Jv_ResolvedMethod*) + _Jv_ResolvePoolEntry (defining_class, index); + + sp -= rmeth->stack_item_count; + NULLCHECK(sp[0]); + + jobject rcv = (jobject)sp[0]; + + fun = (void (*) (...)) + _Jv_LookupInterfaceMethod (rcv->getClass (), + rmeth->method->name, + rmeth->method->signature); + } + goto perform_invoke; + + + case op_new: + { + int index = get2u (pc); pc += 2; + jclass klass = (jclass) _Jv_ResolvePoolEntry (defining_class, + index); + _Jv_InitClass (klass); + jobject res = _Jv_AllocObject (klass, klass->size_in_bytes); + PUSHA (res); + } + goto next_insn; + + case op_newarray: + { + int atype = get1u (pc++); + int size = POPI(); + jobject result = _Jv_NewArray (atype, size); + PUSHA (result); + } + goto next_insn; + + case op_anewarray: + { + int index = get2u (pc); pc += 2; + jclass klass = (jclass) _Jv_ResolvePoolEntry (defining_class, index); + int size = POPI(); + _Jv_InitClass (klass); + jobject result = _Jv_NewObjectArray (size, klass, 0); + PUSHA (result); + } + goto next_insn; + + case op_arraylength: + { + __JArray *arr = (__JArray*)POPA(); + PUSHI (arr->length); + } + goto next_insn; + + case op_athrow: + { + jobject value = POPA(); + TIME_SUSPEND; + JvThrow (value); + } + goto next_insn; + + case op_checkcast: + { + jobject value = POPA(); + jint index = get2u (pc); pc += 2; + jclass to = (jclass)_Jv_ResolvePoolEntry (defining_class, + index); + + if (value != NULL && ! to->isInstance (value)) + { + TIME_SUSPEND; + JvThrow (new java::lang::ClassCastException + (to->getName())); + } + + PUSHA (value); + } + goto next_insn; + + case op_instanceof: + { + jobject value = POPA(); + jint index = get2u (pc); pc += 2; + jclass to = (jclass)_Jv_ResolvePoolEntry (defining_class, + index); + PUSHI (to->isInstance (value)); + } + goto next_insn; + + case op_monitorenter: + { + jobject value = POPA(); + NULLCHECK(value); + _Jv_MonitorEnter (value); + } + goto next_insn; + + case op_monitorexit: + { + jobject value = POPA(); + NULLCHECK(value); + _Jv_MonitorExit (value); + } + goto next_insn; + + case op_ifnull: + { + unsigned char* base_pc = pc-1; + jint offset = get2s (pc); pc += 2; + jobject val = POPA(); + if (val == NULL) + pc = base_pc+offset; + } + goto next_insn; + + case op_ifnonnull: + { + unsigned char* base_pc = pc-1; + jint offset = get2s (pc); pc += 2; + jobject val = POPA(); + if (val != NULL) + pc = base_pc+offset; + } + goto next_insn; + + case op_wide: + { + jint the_mod_op = get1u (pc++); + jint wide = get2u (pc); pc += 2; + + switch (the_mod_op) + { + case op_istore: + STOREI (wide); + goto next_insn; + + case op_fstore: + STOREF (wide); + goto next_insn; + + case op_astore: + STOREA (wide); + goto next_insn; + + case op_lload: + LOADL (wide); + goto next_insn; + + case op_dload: + LOADD (wide); + goto next_insn; + + case op_iload: + LOADI (wide); + goto next_insn; + + case op_aload: + LOADA (wide); + goto next_insn; + + case op_lstore: + STOREL (wide); + goto next_insn; + + case op_dstore: + STORED (wide); + goto next_insn; + + case op_ret: + pc = (unsigned char*) PEEKA (wide); + goto next_insn; + + case op_iinc: + { + jint amount = get2s (pc); pc += 2; + jint value = PEEKI (wide); + POKEI (wide, value+amount); + } + goto next_insn; + + default: + throw_internal_error ("illegal bytecode modified by wide"); + } + + } + + case op_multianewarray: + { + int kind_index = get2u (pc); pc += 2; + int dim = get1u (pc); pc += 1; + + jclass type = (jclass) _Jv_ResolvePoolEntry (defining_class, + kind_index); + _Jv_InitClass (type); + jint *sizes = (jint*) alloca (sizeof (jint)*dim); + + for (int i = dim - 1; i >= 0; i--) + { + sizes[i] = POPI (); + } + + jobject res = _Jv_NewMultiArray (type,dim, sizes); + + PUSHA (res); + } + goto next_insn; + + case op_goto_w: + { + unsigned char* base_pc = pc-1; + int offset = get4 (pc); pc += 4; + pc = base_pc+offset; + } + goto next_insn; + + case op_jsr_w: + { + unsigned char* base_pc = pc-1; + int offset = get4 (pc); pc += 4; + PUSHA((jobject)pc); + pc = base_pc+offset; + } + goto next_insn; + + default: + throw_internal_error ("opcode not implemented"); + + } + goto next_insn; +} + + +static void +throw_internal_error (char *msg) +{ + JvThrow (new java::lang::InternalError (JvNewStringLatin1 (msg))); +} + +static void +throw_incompatible_class_change_error (jstring msg) +{ + JvThrow (new java::lang::IncompatibleClassChangeError (msg)); +} + +#if !HANDLE_SEGV +static java::lang::NullPointerException *null_pointer_exc; +static void +throw_null_pointer_exception () +{ + if (null_pointer_exc == NULL) + null_pointer_exc = new java::lang::NullPointerException; + + JvThrow (null_pointer_exc); +} +#endif + +#if !HANDLE_FPE +static java::lang::ArithmeticException *arithmetic_exc; +static void +throw_arithmetic_exception () +{ + if (arithmetic_exc == NULL) + arithmetic_exc = new java::lang::ArithmeticException + (JvNewStringLatin1 ("/ by zero")); + + JvThrow (arithmetic_exc); +} +#endif + +void +jvdump(jobject o) +{ + _Jv_InterpMethod::dump_object(o); +} + +/* FIXME: This is not finished! */ +void +_Jv_InterpMethod::dump_object(jobject o) +{ + java::io::PrintStream *out = java::lang::System::out; + + if (o == NULL) + { + out->println (JvNewStringLatin1 ("<null>")); + return; + } + + jclass klass = o->getClass (); + + out->print (klass->getName ()); + out->print (JvNewStringLatin1 ("@0x")); + out->print (java::lang::Integer::toHexString ((jint)o)); + out->print (JvNewStringLatin1 ("{")); +#if 0 + while (klass && klass != &ObjectClass) + { + _Jv_Field *fields = klass->fields; + int max = klass->field_count; + + for (int i = 0; i < max; ++i) + { + out->print (_Jv_NewStringUTF (field->name->data)); + out->print (JvNewStringLatin1 ("=")); + + if (JvFieldIsRef (field)) + { + if (field->flags & STATIC) + out->print (JvGetSt) + } + field = field->getNextInstanceField (); + + if (i+1 < max && klass->getSuperclass () != null) + out->print (JvNewStringLatin1 ("; ")); + } + + klass = klass->getSuperclass(); + } +#endif + out->print (JvNewStringLatin1 ("}\n")); + +} + +#endif // INTERPRETER diff --git a/libjava/java/lang/Class.h b/libjava/java/lang/Class.h index eb0a2f95533..df55425daee 100644 --- a/libjava/java/lang/Class.h +++ b/libjava/java/lang/Class.h @@ -21,25 +21,30 @@ details. */ extern "C" void _Jv_InitClass (jclass klass); extern "C" void _Jv_RegisterClasses (jclass *classes); +// These are the possible values for the `state' field of the class +// structure. Note that ordering is important here; in particular +// `resolved' must come between `nothing' and the other states. +// Whenever the state changes, one should notify all waiters of this +// class. +#define JV_STATE_NOTING 0 // set by compiler + +#define JV_STATE_PRELOADING 1 // can do _Jv_FindClass +#define JV_STATE_LOADING 3 // has super installed +#define JV_STATE_LOADED 5 // is complete + +#define JV_STATE_COMPILED 6 // this was a compiled class + +#define JV_STATE_PREPARED 7 // layout & static init done +#define JV_STATE_LINKED 9 // strings interned + +#define JV_STATE_IN_PROGRESS 10 // <clinit> running +#define JV_STATE_DONE 12 // + +#define JV_STATE_ERROR 14 // must be last + struct _Jv_Field; struct _Jv_VTable; -#define CONSTANT_Class 7 -#define CONSTANT_Fieldref 9 -#define CONSTANT_Methodref 10 -#define CONSTANT_InterfaceMethodref 11 -#define CONSTANT_String 8 -#define CONSTANT_Integer 3 -#define CONSTANT_Float 4 -#define CONSTANT_Long 5 -#define CONSTANT_Double 6 -#define CONSTANT_NameAndType 12 -#define CONSTANT_Utf8 1 -#define CONSTANT_Unicode 2 -#define CONSTANT_ResolvedFlag 16 -#define CONSTANT_ResolvedString (CONSTANT_String+CONSTANT_ResolvedFlag) -#define CONSTANT_ResolvedClass (CONSTANT_Class+CONSTANT_ResolvedFlag) - struct _Jv_Constants { jint size; @@ -134,9 +139,11 @@ public: return size_in_bytes; } + // finalization + void finalize (); + private: void checkMemberAccess (jint flags); - void resolveConstants (void); // Various functions to handle class initialization. java::lang::Throwable *hackTrampoline (jint, java::lang::Throwable *); @@ -147,12 +154,6 @@ private: friend _Jv_Method *_Jv_GetMethodLocal (jclass klass, _Jv_Utf8Const *name, _Jv_Utf8Const *signature); friend void _Jv_InitClass (jclass klass); - friend void _Jv_RegisterClasses (jclass *classes); - friend jclass _Jv_FindClassInCache (_Jv_Utf8Const *name, - java::lang::ClassLoader *loader); - friend jclass _Jv_FindArrayClass (jclass element); - friend jclass _Jv_NewClass (_Jv_Utf8Const *name, jclass superclass, - java::lang::ClassLoader *loader); friend jfieldID JvGetFirstInstanceField (jclass); friend jint JvNumInstanceFields (jclass); @@ -165,6 +166,41 @@ private: friend class _Jv_PrimClass; + // Friends classes and functions to implement the ClassLoader + friend class java::lang::ClassLoader; + + friend void _Jv_WaitForState (jclass, int); + friend void _Jv_RegisterClasses (jclass *classes); + friend void _Jv_RegisterInitiatingLoader (jclass,java::lang::ClassLoader*); + friend void _Jv_UnregisterClass (jclass); + friend jclass _Jv_FindClass (_Jv_Utf8Const *name, + java::lang::ClassLoader *loader); + friend jclass _Jv_FindClassInCache (_Jv_Utf8Const *name, + java::lang::ClassLoader *loader); + friend jclass _Jv_FindArrayClass (jclass element, + java::lang::ClassLoader *loader); + friend jclass _Jv_NewClass (_Jv_Utf8Const *name, jclass superclass, + java::lang::ClassLoader *loader); + + friend void _Jv_InternClassStrings (jclass); + +#ifdef INTERPRETER + friend jboolean _Jv_IsInterpretedClass (jclass); + friend void _Jv_InitField (jobject, jclass, _Jv_Field*); + friend _Jv_Method* _Jv_LookupDeclaredMethod (jclass, _Jv_Utf8Const *, + _Jv_Utf8Const*); + friend int _Jv_DetermineVTableIndex (jclass, _Jv_Utf8Const *, + _Jv_Utf8Const*); + friend void _Jv_InitField (jobject, jclass, int); + friend void* _Jv_ResolvePoolEntry (jclass, int); + friend void _Jv_PrepareClass (jclass); + + friend class _Jv_ClassReader; + friend class _Jv_InterpClass; + friend class _Jv_InterpMethod; + friend class _Jv_InterpMethodInvocation; +#endif + #ifdef JV_MARKOBJ_DECL friend JV_MARKOBJ_DECL; #endif diff --git a/libjava/java/lang/Class.java b/libjava/java/lang/Class.java index 4ffcceaf675..bc826e9029b 100644 --- a/libjava/java/lang/Class.java +++ b/libjava/java/lang/Class.java @@ -152,4 +152,7 @@ public final class Class implements Serializable // Initialize the class. private native void initializeClass (); + + // finalization + protected native void finalize (); } diff --git a/libjava/java/lang/ClassLoader.java b/libjava/java/lang/ClassLoader.java index 048cea7d883..f0b533f6041 100644 --- a/libjava/java/lang/ClassLoader.java +++ b/libjava/java/lang/ClassLoader.java @@ -9,86 +9,394 @@ Libgcj License. Please consult the file "LIBGCJ_LICENSE" for details. */ package java.lang; + import java.io.InputStream; -import java.util.Hashtable; +import java.net.URL; +import java.net.URLConnection; +import java.util.Stack; /** - * @author Tom Tromey <tromey@cygnus.com> - * @date October 28, 1998 + * The class <code>ClassLoader</code> is intended to be subclassed by + * applications in order to describe new ways of loading classes, + * such as over the network. + * + * @author Kresten Krab Thorup */ /* Written using "Java Class Libraries", 2nd edition, ISBN 0-201-31002-3 * Status: Just a stub; not useful at all. */ -public abstract class ClassLoader -{ - protected ClassLoader () - { - cache = new Hashtable (); - } +public abstract class ClassLoader { - protected final Class defineClass (String className, byte[] bytecode, - int offset, int length) - { - throw new ClassFormatError ("defineClass unimplemented"); - } + static private ClassLoader system; + + private static native ClassLoader getVMClassLoader0 (); - protected final Class defineClass (byte[] bytecodes, - int offset, int length) - { - return defineClass (null, bytecodes, offset, length); - } + static public ClassLoader getSystemClassLoader () { + if (system == null) + system = getVMClassLoader0 (); + return system; + } - protected final Class findLoadedClass (String className) - { - return (Class) cache.get(className); - } + /** + * Creates a <code>ClassLoader</code>. The only thing this + * constructor does, is to call + * <code>checkCreateClassLoader</code> on the current + * security manager. + * @exception java.lang.SecurityException if not allowed + */ + protected ClassLoader() + { + SecurityManager security = System.getSecurityManager (); + if (security != null) + security.checkCreateClassLoader (); + } - protected final Class findSystemClass (String className) - throws ClassNotFoundException - { - Class c = system.findLoadedClass(className); - system.resolveClass(c); - return c; - } + /** + * Loads and link the class by the given name. + * @param name the name of the class. + * @return the class loaded. + * @see ClassLoader#loadClass(String,boolean) + * @exception java.lang.ClassNotFoundException + */ + public Class loadClass(String name) + throws java.lang.ClassNotFoundException, java.lang.LinkageError + { + return loadClass (name, true); + } - // FIXME: Needs URL. - // public URL getResource (String resName); + /** + * Loads the class by the given name. + * As per java 1.1, this has been deprecated. Use + * <code>loadClass(String)</code> + * instead. + * @param name the name of the class. + * @param link if the class should be linked. + * @return the class loaded. + * @exception java.lang.ClassNotFoundException + * @deprecated + */ + protected abstract Class loadClass(String name, boolean link) + throws java.lang.ClassNotFoundException, java.lang.LinkageError; - public InputStream getResourceAsStream (String resName) - { - return null; - } + /** + * Defines a class, given the class-data. According to the JVM, this + * method should not be used; instead use the variant of this method + * in which the name of the class being defined is specified + * explicitly. + * <P> + * If the name of the class, as specified (implicitly) in the class + * data, denotes a class which has already been loaded by this class + * loader, an instance of + * <code>java.lang.ClassNotFoundException</code> will be thrown. + * + * @param data bytes in class file format. + * @param off offset to start interpreting data. + * @param len length of data in class file. + * @return the class defined. + * @exception java.lang.ClassNotFoundException + * @exception java.lang.LinkageError + * @see ClassLoader#defineClass(String,byte[],int,int) */ + protected final Class defineClass(byte[] data, int off, int len) + throws java.lang.ClassNotFoundException, java.lang.LinkageError + { + return defineClass (null, data, off, len); + } - // FIXME: Needs URL. - // public static final URL getSystemResource (String resName); + /** + * Defines a class, given the class-data. This is preferable + * over <code>defineClass(byte[],off,len)</code> since it is more + * secure. If the expected name does not match that of the class + * file, <code>ClassNotFoundException</code> is thrown. If + * <code>name</code> denotes the name of an already loaded class, a + * <code>LinkageError</code> is thrown. + * <p> + * + * FIXME: How do we assure that the class-file data is not being + * modified, simultaneously with the class loader running!? If this + * was done in some very clever way, it might break security. + * Right now I am thinking that defineclass should make sure never to + * read an element of this array more than once, and that that would + * assure the ``immutable'' appearance. It is still to be determined + * if this is in fact how defineClass operates. + * + * @param name the expected name. + * @param data bytes in class file format. + * @param off offset to start interpreting data. + * @param len length of data in class file. + * @return the class defined. + * @exception java.lang.ClassNotFoundException + * @exception java.lang.LinkageError + */ + protected final synchronized Class defineClass(String name, + byte[] data, + int off, + int len) + throws java.lang.ClassNotFoundException, java.lang.LinkageError + { + if (data==null || data.length < off+len || off<0 || len<0) + throw new ClassFormatError ("arguments to defineClass " + + "are meaningless"); - public static final InputStream getSystemResourceAsStream (String resName) - { - return null; - } + // as per 5.3.5.1 + if (name != null && findLoadedClass (name) != null) + throw new java.lang.LinkageError ("class " + + name + + " already loaded"); - protected abstract Class loadClass (String className, boolean resolve) - throws ClassNotFoundException; - public Class loadClass (String name) throws ClassNotFoundException - { - return loadClass (name, true); - } + try { + // Since we're calling into native code here, + // we better make sure that any generated + // exception is to spec! + + return defineClass0 (name, data, off, len); + + } catch (java.lang.LinkageError x) { + throw x; // rethrow + + } catch (java.lang.ClassNotFoundException x) { + throw x; // rethrow + + } catch (java.lang.VirtualMachineError x) { + throw x; // rethrow + + } catch (java.lang.Throwable x) { + // This should never happen, or we are beyond spec. + + throw new InternalError ("Unexpected exception " + + "while defining class " + + name + ": " + + x.toString ()); + } + } + + /** This is the entry point of defineClass into the native code */ + private native Class defineClass0 (String name, + byte[] data, + int off, + int len) + throws java.lang.ClassNotFoundException, java.lang.LinkageError; - protected final void resolveClass (Class c) - { - // Nothing for now. - } - protected final void setSigners (Class cl, Object[] signers) - { - // Nothing for now. + /** This is called by defineClass0, once the "raw" and uninitialized + * class object has been created, and handles exceptions generated + * while actually defining the class (_Jv_DefineClass). defineClass0 + * holds the lock on the new class object, so it needs to capture + * these exceptions. */ + + private static Throwable defineClass1 (Class klass, byte[] data, + int offset, int length) + { + try { + defineClass2 (klass, data, offset, length); + } catch (Throwable x) { + return x; } + return null; + } + + /** This is just a wrapper for _Jv_DefineClass */ + private static native void defineClass2 (Class klass, byte[] data, + int offset, int length) + throws Throwable; + + /** + * Link the given class. This will bring the class to a state where + * the class initializer can be run. Linking involves the following + * steps: + * <UL> + * <LI> Prepare (allocate and internalize) the constant strings that + * are used in this class. + * <LI> Allocate storage for static fields, and define the layout + * of instance fields. + * <LI> Perform static initialization of ``static final'' int, + * long, float, double and String fields for which there is a + * compile-time constant initializer. + * <LI> Create the internal representation of the ``vtable''. + * </UL> + * For <code>gcj</code>-compiled classes, only the first step is + * performed. The compiler will have done the rest already. + * <P> + * This is called by the system automatically, + * as part of class initialization; there is no reason to ever call + * this method directly. + * <P> + * For historical reasons, this method has a name which is easily + * misunderstood. Java classes are never ``resolved''. Classes are + * linked; whereas method and field references are resolved. + * <P> + * FIXME: The JDK documentation declares this method + * <code>final</code>, we declare it <code>static</code> -- any + * objections? This allows us to call it directly from native code + * with less hassle. + * + * @param clazz the class to link. + * @exception java.lang.LinkageError + */ + protected static void resolveClass(Class clazz) + throws java.lang.LinkageError + { + synchronized (clazz) + { + try { + linkClass0 (clazz); + } catch (Throwable x) { + markClassErrorState0 (clazz); + + if (x instanceof Error) + throw (Error)x; + else + throw new java.lang.InternalError + ("unexpected exception during linking: " + x); + } + } + } + + /** Internal method. Calls _Jv_PrepareClass and + * _Jv_InternClassStrings. This is only called from resolveClass. */ + private static native void linkClass0(Class clazz) + throws java.lang.LinkageError; + + /** Internal method. Marks the given clazz to be in an erroneous + * state, and calls notifyAll() on the class object. This should only + * be called when the caller has the lock on the class object. */ + private static native void markClassErrorState0(Class clazz); + + + /** + * Returns a class found in a system-specific way, typically + * via the <code>java.class.path</code> system property. + * + * @param name the class to resolve. + * @return the class loaded. + * @exception java.lang.LinkageError + * @exception java.lang.ClassNotFoundException + */ + protected native static Class findSystemClass(String name) + throws java.lang.ClassNotFoundException, java.lang.LinkageError; + + /* + * Does currently nothing. + */ + protected final void setSigners(Class claz, Object[] signers) { + /* claz.setSigners (signers); */ + } + + /* + * If a class named <code>name</code> was previously loaded using + * this <code>ClassLoader</code>, then it is returned. Otherwise + * it returns <code>null</code>. + * @param name class to find. + * @return the class loaded, or null. + */ + protected native Class findLoadedClass(String name); - // Class cache. - private Hashtable cache; + public static final InputStream getSystemResourceAsStream(String name) { + return system.getResourceAsStream (name); + } - // The system class loader. FIXME: should have an actual value - private static final ClassLoader system = null; + public static final URL getSystemResource(String name) { + return system.getResource (name); + } + + public static final byte[] getSystemResourceAsBytes(String name) { + return system.getResourceAsBytes (name); + } + + /** + * Return an InputStream representing the resource name. + * This is essentially like + * <code>getResource(name).openStream()</code>, except + * it masks out any IOException and returns null on failure. + * @param name resource to load + * @return an InputStream, or null + * @see java.lang.ClassLoader#getResource(String) + * @see java.lang.ClassLoader#getResourceAsBytes(String) + * @see java.io.InputStream + */ + public InputStream getResourceAsStream(String name) + { + try { + URL res = getResource (name); + if (res == null) return null; + return res.openStream (); + } catch (java.io.IOException x) { + return null; + } + } + + /** + * Return a byte array <code>byte[]</code> representing the + * resouce <code>name</code>. This only works for resources + * that have a known <code>content-length</code>, and + * it will block while loading the resource. Returns null + * for error conditions.<p> + * Since it is synchroneous, this is only convenient for + * resources that are "readily" available. System resources + * can conveniently be loaded this way, and the runtime + * system uses this to load class files. <p> + * To find the class data for a given class, use + * something like the following: + * <ul><code> + * String res = clazz.getName().replace ('.', '/')) + ".class";<br> + * byte[] data = getResourceAsBytes (res); + * </code></ul> + * @param name resource to load + * @return a byte array, or null + * @see java.lang.ClassLoader#getResource(String) + * @see java.lang.ClassLoader#getResourceAsStream(String) + */ + public byte[] getResourceAsBytes(String name) { + try { + URL res = getResource (name); + if (res == null) return null; + URLConnection conn = res.openConnection (); + int len = conn.getContentLength (); + if (len == -1) return null; + return readbytes (conn.getInputStream (), len); + } catch (java.io.IOException x) { + return null; + } + } + + /** + * Return an java.io.URL representing the resouce <code>name</code>. + * @param name resource to load + * @return a URL, or null if there is no such resource. + * @see java.lang.ClassLoader#getResourceAsBytes(String) + * @see java.lang.ClassLoader#getResourceAsStream(String) + * @see java.io.URL + */ + public URL getResource(String name) { + return null; + } + + /** + * Utility routine to read a resource fully, even if the given + * InputStream only provides partial results. + */ + private static byte[] readbytes (InputStream is, int length) + { + try { + + byte[] data = new byte[length]; + int read; + int off = 0; + + while (off != length) + { + read = is.read (data, off, (int) (length-off)); + + if (read == -1) + return null; + + off += read; + } + + return data; + } catch (java.io.IOException x) { + return null; + } + } } diff --git a/libjava/java/lang/FirstThread.java b/libjava/java/lang/FirstThread.java index ec0f1db33d5..0dd5c9c67ad 100644 --- a/libjava/java/lang/FirstThread.java +++ b/libjava/java/lang/FirstThread.java @@ -19,12 +19,30 @@ package java.lang; final class FirstThread extends Thread { - public native void run (); + public native void run0 (); + public void run () + { + try { + run0 (); + } catch (Throwable ex) { + System.err.println ("uncaught exception at top level"); + ex.printStackTrace (); + } + } public FirstThread (ThreadGroup g, Class k, Object o) { super (g, null, "main"); klass = k; + klass_name = null; + args = o; + } + + public FirstThread (ThreadGroup g, String class_name, Object o) + { + super (g, null, "main"); + klass = null; + klass_name = class_name; args = o; } @@ -36,5 +54,6 @@ final class FirstThread extends Thread // Private data. private Class klass; + private String klass_name; private Object args; } diff --git a/libjava/java/lang/VMClassLoader.java b/libjava/java/lang/VMClassLoader.java new file mode 100644 index 00000000000..026f6d8d1fb --- /dev/null +++ b/libjava/java/lang/VMClassLoader.java @@ -0,0 +1,117 @@ +/* Copyright (C) 1999 Cygnus Solutions + + This file is part of libgcj. + +This software is copyrighted work licensed under the terms of the +Libgcj License. Please consult the file "LIBGCJ_LICENSE" for +details. */ + +/* Author: Kresten Krab Thorup <krab@gnu.org> */ + +package java.lang; + +import java.io.*; +import java.net.URL; +import gnu.gcj.util.path.SearchPath; + +final class VMClassLoader extends java.lang.ClassLoader +{ + private SearchPath path; + private final String path_seperator; + private final String file_seperator; + private final char file_seperator_char; + + private VMClassLoader () { + path_seperator = System.getProperty ("path.separator", ":"); + file_seperator = System.getProperty ("file.separator", "/"); + + file_seperator_char = file_seperator.charAt (0); + + String class_path = System.getProperty ("java.class.path", "."); + path = new SearchPath (class_path); + } + + protected Class loadClass(String name, + boolean resolve) + throws java.lang.ClassNotFoundException, java.lang.LinkageError + { + return loadClassInternal (name, resolve, false); + } + + /** I'm a little in doubt here, if this method is + actually supposed to throw a LinkageError, or not. + The spec, 20.14.3, is a little unclear. It says: + + `` The general contract of loadClass is that, given the name + of a class, it either returns the Class object for the class + or throws a ClassNotFoundException.'' + + However, by making LinkageError a checked exception, + i.e., mention it directly in the throws clause, + we'll force caller to consider that case as well. + **/ + + protected Class loadClassInternal(String name, + boolean resolve, + boolean fromBootLoader) + throws java.lang.ClassNotFoundException, java.lang.LinkageError + { + Class clazz; + + /** TODO: call _Jv_VerifyClassName **/ + if ( (name.indexOf ('/') != -1) + || (name.charAt (0) == '.') + || (name.indexOf (file_seperator) != -1) + || (name.indexOf ("..") != -1)) + { + throw new IllegalArgumentException (name); + } + + // already loaded? + clazz = findLoadedClass (name); + + // we need access to the boot class loader here + if (clazz == null && !fromBootLoader) + clazz = findBootClass (name); + + if (clazz == null) + { + StringBuffer res = new StringBuffer (); + + // here we do actually replace .'s with /'s because + // we're going to find something in the file system. + res.append (name.replace ('.', file_seperator_char)); + res.append (".class"); + + byte[] data = getResourceAsBytes (res.toString ()); + + if (data == null) + throw new ClassNotFoundException (name); + + clazz = defineClass (name, data, 0, data.length); + + } + + if (resolve && clazz != null) + resolveClass (clazz); + + return clazz; + } + + private native Class findBootClass (String name); + + public InputStream getResourceAsStream(String name) + { + return path.getStream (name); + } + + public URL getResource(String name) + { + return path.getURL (name); + } + + public byte[] getResourceAsBytes(String name) + { + return path.getBytes (name); + } +} diff --git a/libjava/java/lang/VirtualMachineError.java b/libjava/java/lang/VirtualMachineError.java index dcc907c5d03..ee6d8e829bd 100644 --- a/libjava/java/lang/VirtualMachineError.java +++ b/libjava/java/lang/VirtualMachineError.java @@ -20,6 +20,11 @@ package java.lang; * Status: Believed complete and correct. */ +/* FIXME: We should consider adding some special error message when this + * exception is thrown, or maybe if it being caught at top-level. Such + * a message would direct the user to send a bug report to + * gcj-bugs@cygnus.com, or something like that. --KKT */ + public abstract class VirtualMachineError extends Error { public VirtualMachineError () diff --git a/libjava/java/lang/natClass.cc b/libjava/java/lang/natClass.cc index 1768df5ecb2..367f14abb0f 100644 --- a/libjava/java/lang/natClass.cc +++ b/libjava/java/lang/natClass.cc @@ -39,6 +39,8 @@ details. */ #include <java/lang/System.h> #include <java/lang/SecurityManager.h> +#include <java-cpool.h> + #define CloneableClass _CL_Q34java4lang9Cloneable @@ -59,28 +61,6 @@ static _Jv_Utf8Const *void_signature = _Jv_makeUtf8Const ("()V", 3); static _Jv_Utf8Const *clinit_name = _Jv_makeUtf8Const ("<clinit>", 8); static _Jv_Utf8Const *init_name = _Jv_makeUtf8Const ("<init>", 6); -// These are the possible values for the `state' field. They more or -// less follow the section numbers in the Java Language Spec. Right -// now we don't bother to represent other interesting states, e.g. the -// states a class might inhabit before it is prepared. Note that -// ordering is important here; in particular `resolved' must come -// between `nothing' and the other states. -#define STATE_NOTHING 0 -#define STATE_RESOLVED 1 -#define STATE_IN_PROGRESS 6 -#define STATE_DONE 9 -#define STATE_ERROR 10 - -// Size of local hash table. -#define HASH_LEN 256 - -// Hash function for Utf8Consts. -#define HASH_UTF(Utf) (((Utf)->hash) % HASH_LEN) - -// This is the table we use to keep track of loaded classes. See Spec -// section 12.2. -static jclass loaded_classes[HASH_LEN]; - jclass @@ -111,6 +91,9 @@ java::lang::Class::forName (jstring className) #endif if (! klass) JvThrow (new java::lang::ClassNotFoundException (className)); + + _Jv_InitClass (klass); + return klass; } @@ -380,33 +363,13 @@ java::lang::Class::newInstance (void) return r; } -// Initialize the constants. void -java::lang::Class::resolveConstants (void) +java::lang::Class::finalize (void) { - for (int i = 0; i < constants.size; ++i) - { - if (constants.tags[i] == CONSTANT_String) - { - jstring str; - str = _Jv_NewStringUtf8Const ((_Jv_Utf8Const *) constants.data[i]); - constants.data[i] = (void *) str; - constants.tags[i] = CONSTANT_ResolvedString; - } - else if (constants.tags[i] == CONSTANT_Class) - { - _Jv_Utf8Const *name = (_Jv_Utf8Const *) constants.data[i]; - jclass klass = _Jv_FindClassFromSignature (name->data, loader); - if (! klass) - { - jstring str = _Jv_NewStringUtf8Const (name); - JvThrow (new java::lang::ClassNotFoundException (str)); - } - - constants.data[i] = (void *) klass; - constants.tags[i] = CONSTANT_ResolvedClass; - } - } +#ifdef INTERPRETER + JvAssert (_Jv_IsInterpretedClass (this)); + _Jv_UnregisterClass (this); +#endif } // FIXME. @@ -424,38 +387,52 @@ void java::lang::Class::initializeClass (void) { // Short-circuit to avoid needless locking. - if (state == STATE_DONE) + if (state == JV_STATE_DONE) return; - // Step 1. - _Jv_MonitorEnter (this); + // do this before we enter the monitor below, since this can cause + // exceptions. Here we assume, that reading "state" is an atomic + // operation, I pressume that is true? --Kresten + if (state < JV_STATE_LINKED) + { +#ifdef INTERPRETER + if (_Jv_IsInterpretedClass (this)) + { + java::lang::ClassLoader::resolveClass (this); - // FIXME: This should actually be handled by calling into the class - // loader. For now we put it here. - if (state < STATE_RESOLVED) + // Step 1. + _Jv_MonitorEnter (this); + } + else +#endif + { + // Step 1. + _Jv_MonitorEnter (this); + _Jv_InternClassStrings (this); + } + } + else { - // We set the state before calling resolveConstants to avoid - // infinite recursion when processing String or Class. - state = STATE_RESOLVED; - resolveConstants (); + // Step 1. + _Jv_MonitorEnter (this); } // Step 2. java::lang::Thread *self = java::lang::Thread::currentThread(); // FIXME: `self' can be null at startup. Hence this nasty trick. self = (java::lang::Thread *) ((long) self | 1); - while (state == STATE_IN_PROGRESS && thread && thread != self) + while (state == JV_STATE_IN_PROGRESS && thread && thread != self) wait (); // Steps 3 & 4. - if (state == STATE_DONE || state == STATE_IN_PROGRESS || thread == self) + if (state == JV_STATE_DONE || state == JV_STATE_IN_PROGRESS || thread == self) { _Jv_MonitorExit (this); return; } // Step 5. - if (state == STATE_ERROR) + if (state == JV_STATE_ERROR) { _Jv_MonitorExit (this); JvThrow (new java::lang::NoClassDefFoundError); @@ -463,7 +440,7 @@ java::lang::Class::initializeClass (void) // Step 6. thread = self; - state = STATE_IN_PROGRESS; + state = JV_STATE_IN_PROGRESS; _Jv_MonitorExit (this); // Step 7. @@ -477,7 +454,7 @@ java::lang::Class::initializeClass (void) { // Caught an exception. _Jv_MonitorEnter (this); - state = STATE_ERROR; + state = JV_STATE_ERROR; notify (); _Jv_MonitorExit (this); JvThrow (except); @@ -492,7 +469,7 @@ java::lang::Class::initializeClass (void) if (! except) { _Jv_MonitorEnter (this); - state = STATE_DONE; + state = JV_STATE_DONE; } else { @@ -503,7 +480,7 @@ java::lang::Class::initializeClass (void) except = hackTrampoline(2, except); } _Jv_MonitorEnter (this); - state = STATE_ERROR; + state = JV_STATE_ERROR; } notify (); _Jv_MonitorExit (this); @@ -530,6 +507,64 @@ _Jv_GetMethodLocal (jclass klass, _Jv_Utf8Const *name, return NULL; } +#define MCACHE_SIZE 1013 + +struct _Jv_mcache { + jclass klass; + _Jv_Method *method; +}; + +static _Jv_mcache method_cache[MCACHE_SIZE]; +static int method_cache_count; + +static void* +_Jv_FindMethodInCache (jclass klass, + _Jv_Utf8Const *name, + _Jv_Utf8Const *signature) +{ + for (int index = name->hash % MCACHE_SIZE; + method_cache[index].klass != NULL; + index = (index+1) % MCACHE_SIZE) + { + _Jv_mcache *mc = (method_cache+index); + _Jv_Method *m = mc->method; + + if (mc->klass == klass + && m != NULL // thread safe check + && _Jv_equalUtf8Consts (m->name, name) + && _Jv_equalUtf8Consts (m->signature, signature)) + { + return mc->method->ncode; + } + } + return NULL; +} + +static void +_Jv_AddMethodToCache (jclass klass, + _Jv_Method *method) +{ + _Jv_MonitorEnter (&ClassClass); + + if (method_cache_count > MCACHE_SIZE*2/3) + { + for (int i = 0; i < MCACHE_SIZE; i++) + method_cache[i].klass = 0; + } + + for (int index = method->name->hash % MCACHE_SIZE; + method_cache[index].klass != NULL; + index = (index+1) % MCACHE_SIZE) + { + method_cache[index].method = method; + method_cache[index].klass = klass; + } + + method_cache_count += 1; + + _Jv_MonitorExit (&ClassClass); +} + void * _Jv_LookupInterfaceMethod (jclass klass, _Jv_Utf8Const *name, _Jv_Utf8Const *signature) @@ -539,6 +574,14 @@ _Jv_LookupInterfaceMethod (jclass klass, _Jv_Utf8Const *name, // call a method of a class until the class is linked. But this // captures the general idea. // klass->getClassLoader()->resolveClass(klass); + // + // KKT: This is unnessecary, exactly for the reason you present: + // _Jv_LookupInterfaceMethod is only called on object instances, and + // such have already been initialized (which includes resolving). + + void *ncode = _Jv_FindMethodInCache (klass, name, signature); + if (ncode != 0) + return ncode; for (; klass; klass = klass->getSuperclass()) { @@ -553,6 +596,8 @@ _Jv_LookupInterfaceMethod (jclass klass, _Jv_Utf8Const *name, if (! java::lang::reflect::Modifier::isPublic(meth->accflags)) JvThrow (new java::lang::IllegalAccessError); + _Jv_AddMethodToCache (klass, meth); + return meth->ncode; } JvThrow (new java::lang::IncompatibleClassChangeError); @@ -565,219 +610,6 @@ _Jv_InitClass (jclass klass) klass->initializeClass(); } -// This function is called many times during startup, before main() is -// run. We do our runtime initialization here the very first time we -// are called. At that point in time we know for certain we are -// running single-threaded, so we don't need to lock when modifying -// `init'. CLASSES is NULL-terminated. -void -_Jv_RegisterClasses (jclass *classes) -{ - static bool init = false; - - if (! init) - { - init = true; - _Jv_InitThreads (); - _Jv_InitGC (); - _Jv_InitializeSyncMutex (); - } - - JvSynchronize sync (&ClassClass); - for (; *classes; ++classes) - { - jclass klass = *classes; - jint hash = HASH_UTF (klass->name); - klass->next = loaded_classes[hash]; - loaded_classes[hash] = klass; - } -} - -void -_Jv_RegisterClass (jclass klass) -{ - jclass classes[2]; - classes[0] = klass; - classes[1] = NULL; - _Jv_RegisterClasses (classes); -} - -jclass -_Jv_FindClassInCache (_Jv_Utf8Const *name, java::lang::ClassLoader *loader) -{ - JvSynchronize sync (&ClassClass); - jint hash = HASH_UTF (name); - jclass klass; - for (klass = loaded_classes[hash]; klass; klass = klass->next) - { - if (loader == klass->loader && _Jv_equalUtf8Consts (name, klass->name)) - break; - } - return klass; -} - -#if 0 -jclass -_Jv_FindClassInCache (jstring name, java::lang::ClassLoader *loader) -{ - JvSynchronize sync (&ClassClass); - jint hash = name->hashCode(); - jclass klass = loaded_classes[(_Jv_ushort) hash % HASH_LEN]; - for ( ; klass; klass = klass->next) - { - if (loader == klass->loader - && _Jv_equalUtf8Consts (klass->name, name, hash)) - break; - } - return klass; -} -#endif - -jclass -_Jv_FindClass (_Jv_Utf8Const* name, java::lang::ClassLoader *loader) -{ - jclass klass = _Jv_FindClassInCache (name, loader); - if (loader && ! klass) - { - klass = loader->loadClass(_Jv_NewStringUtf8Const (name)); - if (klass) - _Jv_RegisterClass (klass); - } - return klass; -} - -#if 0 -jclass -_Jv_FindClass (jstring name, java::lang::ClassLoader *loader) -{ - jclass klass = _Jv_FindClassInCache (name, loader); - if (loader && ! klass) - { - klass = loader->loadClass(name); - if (klass) - _Jv_RegisterClass (klass); - } - return klass; -} -#endif - -jclass -_Jv_NewClass (_Jv_Utf8Const *name, jclass superclass, - java::lang::ClassLoader *loader) -{ - jclass ret = (jclass) JvAllocObject (&ClassClass); - - ret->next = NULL; - ret->name = name; - ret->accflags = 0; - ret->superclass = superclass; - ret->constants.size = 0; - ret->constants.tags = NULL; - ret->constants.data = NULL; - ret->methods = NULL; - ret->method_count = 0; - ret->vtable_method_count = 0; - ret->fields = NULL; - ret->size_in_bytes = 0; - ret->field_count = 0; - ret->static_field_count = 0; - ret->vtable = NULL; - ret->interfaces = NULL; - ret->loader = loader; - ret->interface_count = 0; - ret->state = 0; - ret->thread = NULL; - - _Jv_RegisterClass (ret); - - return ret; -} - -jclass -_Jv_FindArrayClass (jclass element) -{ - _Jv_Utf8Const *array_name; - int len; - if (element->isPrimitive()) - { - // For primitive types the array is cached in the class. - jclass ret = (jclass) element->methods; - if (ret) - return ret; - len = 3; - } - else - len = element->name->length + 5; - - { - char signature[len]; - int index = 0; - signature[index++] = '['; - // Compute name of array class to see if we've already cached it. - if (element->isPrimitive()) - { - signature[index++] = (char) element->method_count; - } - else - { - size_t length = element->name->length; - const char *const name = element->name->data; - if (name[0] != '[') - signature[index++] = 'L'; - memcpy (&signature[index], name, length); - index += length; - if (name[0] != '[') - signature[index++] = ';'; - } - array_name = _Jv_makeUtf8Const (signature, index); - } - - jclass array_class = _Jv_FindClassInCache (array_name, element->loader); - - if (! array_class) - { - // Create new array class. - array_class = _Jv_NewClass (array_name, &ObjectClass, element->loader); - - // Note that `vtable_method_count' doesn't include the initial - // NULL slot. - int dm_count = ObjectClass.vtable_method_count + 1; - - // Create a new vtable by copying Object's vtable (except the - // class pointer, of course). Note that we allocate this as - // unscanned memory -- the vtables are handled specially by the - // GC. - int size = (sizeof (_Jv_VTable) + - ((dm_count - 1) * sizeof (void *))); - _Jv_VTable *vtable = (_Jv_VTable *) _Jv_AllocBytes (size); - vtable->clas = array_class; - memcpy (vtable->method, ObjectClass.vtable->method, - dm_count * sizeof (void *)); - array_class->vtable = vtable; - array_class->vtable_method_count = ObjectClass.vtable_method_count; - - // Stash the pointer to the element type. - array_class->methods = (_Jv_Method *) element; - - // Register our interfaces. - // FIXME: for JDK 1.2 we need Serializable. - static jclass interfaces[] = { &CloneableClass }; - array_class->interfaces = interfaces; - array_class->interface_count = 1; - - // FIXME: initialize other Class instance variables, - // e.g. `fields'. - - array_class->state = STATE_DONE; - } - - // For primitive types, point back at this array. - if (element->isPrimitive()) - element->methods = (_Jv_Method *) array_class; - - return array_class; -} - jboolean _Jv_IsInstanceOf(jobject obj, jclass cl) { diff --git a/libjava/java/lang/natClassLoader.cc b/libjava/java/lang/natClassLoader.cc new file mode 100644 index 00000000000..13452eccd99 --- /dev/null +++ b/libjava/java/lang/natClassLoader.cc @@ -0,0 +1,624 @@ +// natClassLoader.cc - Implementation of java.lang.ClassLoader native methods. + +/* Copyright (C) 1999 Cygnus Solutions + + This file is part of libgcj. + +This software is copyrighted work licensed under the terms of the +Libgcj License. Please consult the file "LIBGCJ_LICENSE" for +details. */ + +/* Author: Kresten Krab Thorup <krab@gnu.org> */ + +#include <config.h> + +#include <stdlib.h> +#include <string.h> + +#include <cni.h> +#include <jvm.h> +#include <java/lang/Character.h> +#include <java/lang/Thread.h> +#include <java/lang/ClassLoader.h> +#include <java/lang/VMClassLoader.h> +#include <java/lang/InternalError.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/reflect/Modifier.h> + +#include <java-interp.h> + +#define CloneableClass _CL_Q34java4lang9Cloneable +extern java::lang::Class CloneableClass; +#define ObjectClass _CL_Q34java4lang6Object +extern java::lang::Class ObjectClass; +#define ClassClass _CL_Q34java4lang5Class +extern java::lang::Class ClassClass; +#define VMClassLoaderClass _CL_Q34java4lang17VMClassLoader +extern java::lang::Class VMClassLoader; +#define ClassLoaderClass _CL_Q34java4lang11ClassLoader +extern java::lang::Class ClassLoaderClass; + +/////////// java.lang.ClassLoader native methods //////////// + +#ifdef INTERPRETER +java::lang::VMClassLoader *redirect = 0; +#endif + +java::lang::ClassLoader* +java::lang::ClassLoader::getVMClassLoader0 () +{ +#ifdef INTERPRETER + if (redirect == 0) + redirect = new java::lang::VMClassLoader; + return redirect; +#else + return 0; +#endif +} + +void +java::lang::ClassLoader::defineClass2 (jclass klass, jbyteArray data, + jint offset, jint length) +{ +#ifdef INTERPRETER + _Jv_DefineClass (klass, data, offset, length); +#endif +} + +java::lang::Class * +java::lang::ClassLoader::defineClass0 (jstring name, + jbyteArray data, + jint offset, + jint length) +{ +#ifdef INTERPRETER + jclass klass; + klass = (jclass) JvAllocObject (&ClassClass, sizeof (_Jv_InterpClass)); + + // 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); + + _Jv_VerifyClassName (name2); + + klass->name = name2; + } + + // this will do the magic. loadInto also operates + // as an exception trampoline for now... + Throwable *ex = defineClass1 (klass, data, offset, length); + + if (ex) // we failed to load it + { + 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. + + JvThrow (ex); + } + + // 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) + { + _Jv_MonitorExit (klass); + _Jv_InternClassStrings (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) + { + _Jv_Throw (new java::lang::LinkageError ()); + } +} + +// Finish linking a class. Only called from ClassLoader::resolveClass. +void +java::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_InternClassStrings (klass); +} + +void +java::lang::ClassLoader::markClassErrorState0 (java::lang::Class *klass) +{ + klass->state = JV_STATE_ERROR; + klass->notifyAll (); +} + + +/** this is the only native method in VMClassLoader, so + we define it here. */ +jclass +java::lang::VMClassLoader::findBootClass (jstring name) +{ + return _Jv_FindClassInCache (_Jv_makeUtf8Const (name), 0); +} + +jclass +java::lang::ClassLoader::findLoadedClass (jstring name) +{ + return _Jv_FindClassInCache (_Jv_makeUtf8Const (name), this); +} + +jclass +java::lang::ClassLoader::findSystemClass (jstring name) +{ + return _Jv_FindClass (_Jv_makeUtf8Const (name), 0); +} + + +/* This is the final step of linking, internalizing the constant strings + * of a class. This is called for both compiled and interpreted + * classes, and it is *only* called from ClassLoader::linkClass0, + * which is always in a context where the current thread has a lock on + * the class in question. We define it here, and not in resolve.cc, so that + * the entire resolve.cc can be #ifdef'ed away when not using the + * interpreter. */ +void +_Jv_InternClassStrings(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 i = 1; i < pool->size; ++i) + { + if (pool->tags[i] == JV_CONSTANT_String) + { + jstring str; + str = _Jv_NewStringUtf8Const ((_Jv_Utf8Const *) pool->data[i]); + pool->data[i] = (void *) str; + pool->tags[i] |= JV_CONSTANT_ResolvedFlag; + } + } + + 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; +}; + +_Jv_LoaderInfo *initiated_classes[HASH_LEN]; +jclass loaded_classes[HASH_LEN]; + +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 + 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; + } + } + } + + _Jv_MonitorExit (&ClassClass); + + return klass; +} + +void +_Jv_UnregisterClass (jclass the_class) +{ + _Jv_MonitorEnter (&ClassClass); + 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 = &((*info)->next)) + { + while ((*info)->klass == the_class) + { + *info = (*info)->next; + } + } + + _Jv_MonitorExit (&ClassClass); +} + +void +_Jv_RegisterInitiatingLoader (jclass klass, java::lang::ClassLoader *loader) +{ + _Jv_LoaderInfo *info = new _Jv_LoaderInfo; // non-gc alloc! + jint hash = HASH_UTF(klass->name); + + _Jv_MonitorEnter (&ClassClass); + info->loader = loader; + info->klass = klass; + info->next = initiated_classes[hash]; + initiated_classes[hash] = info; + _Jv_MonitorExit (&ClassClass); + +} + +// This function is called many times during startup, before main() is +// run. We do our runtime initialization here the very first time we +// are called. At that point in time we know for certain we are +// running single-threaded, so we don't need to lock when modifying +// `init'. CLASSES is NULL-terminated. +void +_Jv_RegisterClasses (jclass *classes) +{ + static bool init = false; + + if (! init) + { + init = true; + _Jv_InitThreads (); + _Jv_InitGC (); + _Jv_InitializeSyncMutex (); + } + + JvSynchronize sync (&ClassClass); + for (; *classes; ++classes) + { + jclass klass = *classes; + jint hash = HASH_UTF (klass->name); + klass->next = loaded_classes[hash]; + loaded_classes[hash] = klass; + + // registering a compiled class causes + // it to be immediately "prepared". + if (klass->state == JV_STATE_NOTING) + klass->state = JV_STATE_COMPILED; + } +} + +void +_Jv_RegisterClass (jclass klass) +{ + jclass classes[2]; + classes[0] = klass; + classes[1] = NULL; + _Jv_RegisterClasses (classes); +} + +#if 0 +// NOTE: this one is out of date with the new loader stuff... +jclass +_Jv_FindClassInCache (jstring name, java::lang::ClassLoader *loader) +{ + JvSynchronize sync (&ClassClass); + jint hash = name->hashCode(); + jclass klass = loaded_classes[(_Jv_ushort) hash % HASH_LEN]; + for ( ; klass; klass = klass->next) + { + if (loader == klass->loader + && _Jv_equal (klass->name, name, hash)) + break; + } + _Jv_MonitorExit (&ClassClass); + return klass; +} +#endif + +jclass _Jv_FindClass (_Jv_Utf8Const *name, + java::lang::ClassLoader *loader) +{ + jclass klass = _Jv_FindClassInCache (name, loader); + +#ifdef INTERPRETER + if (! klass) + { + jstring sname = _Jv_NewStringUTF (name->data); + + if (loader) + { + // Load using a user-defined loader, jvmspec 5.3.2 + klass = loader->loadClass(sname, false); + + // if "loader" delegateted the loadClass operation + // to another loader, register explicitly + // that he is also an initiating loader of the + // given class. + + if (klass && (klass->getClassLoader () != loader)) + _Jv_RegisterInitiatingLoader (klass, 0); + } + else + { + if (redirect == NULL) + { + _Jv_InitClass (&ClassLoaderClass); + java::lang::ClassLoader::getSystemClassLoader (); + } + + // Load using the bootstrap loader jmspec 5.3.1 + klass = redirect -> loadClassInternal (sname, false, true); + + // register that we're an initiating loader + if (klass) + { + _Jv_RegisterInitiatingLoader (klass, 0); + } + } + } + else + { + // we need classes to be in the hash while + // we're loading, so that they can refer to themselves. + _Jv_WaitForState (klass, JV_STATE_LOADED); + } +#endif + + return klass; +} + +#if 0 +// NOTE: this one is out of date with the new class loader stuff... +jclass +_Jv_FindClass (jstring name, java::lang::ClassLoader *loader) +{ + jclass klass = _Jv_FindClassInCache (name, loader); + if (! klass) + { + if (loader) + { + klass = loader->loadClass(name); + } + else + { + // jmspec 5.3.1.2 + + // delegate to the system loader + klass = java::lang::ClassLoader::system.loadClass (sname); + + // register that we're an initiating loader + if (klass) + _Jv_RegisterInitiatingLoader (klass, 0); + } + } + else + { + _Jv_WaitForState (klass, JV_STATE_LOADED); + } + + return klass; +} +#endif + +jclass +_Jv_NewClass (_Jv_Utf8Const *name, jclass superclass, + java::lang::ClassLoader *loader) +{ + jclass ret = (jclass) JvAllocObject (&ClassClass); + + ret->next = NULL; + ret->name = name; + ret->accflags = 0; + ret->superclass = superclass; + ret->constants.size = 0; + ret->constants.tags = NULL; + ret->constants.data = NULL; + ret->methods = NULL; + ret->method_count = 0; + ret->vtable_method_count = 0; + ret->fields = NULL; + ret->size_in_bytes = 0; + ret->field_count = 0; + ret->static_field_count = 0; + ret->vtable = NULL; + ret->interfaces = NULL; + ret->loader = loader; + ret->interface_count = 0; + ret->state = 0; + ret->thread = NULL; + + _Jv_RegisterClass (ret); + + return ret; +} + +jclass +_Jv_FindArrayClass (jclass element, java::lang::ClassLoader *loader) +{ + _Jv_Utf8Const *array_name; + int len; + if (element->isPrimitive()) + { + // For primitive types the array is cached in the class. + jclass ret = (jclass) element->methods; + if (ret) + return ret; + len = 3; + } + else + len = element->name->length + 5; + + { + char signature[len]; + int index = 0; + signature[index++] = '['; + // Compute name of array class to see if we've already cached it. + if (element->isPrimitive()) + { + signature[index++] = (char) element->method_count; + } + else + { + size_t length = element->name->length; + const char *const name = element->name->data; + if (name[0] != '[') + signature[index++] = 'L'; + memcpy (&signature[index], name, length); + index += length; + if (name[0] != '[') + signature[index++] = ';'; + } + array_name = _Jv_makeUtf8Const (signature, index); + } + + jclass array_class = _Jv_FindClassInCache (array_name, element->loader); + + if (! array_class) + { + // Create new array class. + array_class = _Jv_NewClass (array_name, &ObjectClass, element->loader); + + // Note that `vtable_method_count' doesn't include the initial + // NULL slot. + int dm_count = ObjectClass.vtable_method_count + 1; + + // Create a new vtable by copying Object's vtable (except the + // class pointer, of course). Note that we allocate this as + // unscanned memory -- the vtables are handled specially by the + // GC. + int size = (sizeof (_Jv_VTable) + + ((dm_count - 1) * sizeof (void *))); + _Jv_VTable *vtable = (_Jv_VTable *) _Jv_AllocBytes (size); + vtable->clas = array_class; + memcpy (vtable->method, ObjectClass.vtable->method, + dm_count * sizeof (void *)); + array_class->vtable = vtable; + array_class->vtable_method_count = ObjectClass.vtable_method_count; + + // Stash the pointer to the element type. + array_class->methods = (_Jv_Method *) element; + + // Register our interfaces. + // FIXME: for JDK 1.2 we need Serializable. + static jclass interfaces[] = { &CloneableClass }; + array_class->interfaces = interfaces; + array_class->interface_count = 1; + + // as per vmspec 5.3.3.2 + array_class->accflags = element->accflags; + + // FIXME: initialize other Class instance variables, + // e.g. `fields'. + + // say this class is initialized and ready to go! + array_class->state = JV_STATE_DONE; + + // vmspec, section 5.3.3 describes this + if (element->loader != loader) + _Jv_RegisterInitiatingLoader (array_class, loader); + } + + // For primitive types, point back at this array. + if (element->isPrimitive()) + element->methods = (_Jv_Method *) array_class; + + return array_class; +} + + diff --git a/libjava/java/lang/natFirstThread.cc b/libjava/java/lang/natFirstThread.cc index d47446be4a8..319e487b8ab 100644 --- a/libjava/java/lang/natFirstThread.cc +++ b/libjava/java/lang/natFirstThread.cc @@ -27,7 +27,7 @@ details. */ typedef void main_func (jobject); void -java::lang::FirstThread::run (void) +java::lang::FirstThread::run0 (void) { Utf8Const* main_signature = _Jv_makeUtf8Const ("([Ljava.lang.String;)V", 22); Utf8Const* main_name = _Jv_makeUtf8Const ("main", 4); @@ -41,6 +41,12 @@ java::lang::FirstThread::run (void) DIE ("class must be public"); #endif + if (klass == NULL) + { + klass = java::lang::Class::forName (klass_name); + if (klass != NULL) _Jv_InitClass (klass); + } + _Jv_Method *meth = _Jv_GetMethodLocal (klass, main_name, main_signature); // Some checks from Java Spec section 12.1.4. diff --git a/libjava/java/lang/reflect/natArray.cc b/libjava/java/lang/reflect/natArray.cc index 3da8b5c01ce..2c951db0046 100644 --- a/libjava/java/lang/reflect/natArray.cc +++ b/libjava/java/lang/reflect/natArray.cc @@ -45,10 +45,11 @@ java::lang::reflect::Array::newInstance (jclass componentType, jintArray dimensi if (ndims == 1) return newInstance (componentType, dims[0]); jclass arrayType = componentType; - for (int i = 0; i < ndims; i++) - arrayType = _Jv_FindArrayClass (arrayType); - return _Jv_NewMultiArray (arrayType, ndims, dims); + for (int i = 0; i < ndims; i++) // FIXME 2nd arg should + // be "current" loader + arrayType = _Jv_FindArrayClass (arrayType, 0); + return _Jv_NewMultiArray (arrayType, ndims, dims); } jint diff --git a/libjava/java/lang/reflect/natMethod.cc b/libjava/java/lang/reflect/natMethod.cc index 720bbc3d74e..a62d1ffeac2 100644 --- a/libjava/java/lang/reflect/natMethod.cc +++ b/libjava/java/lang/reflect/natMethod.cc @@ -378,8 +378,10 @@ java::lang::reflect::Method::getType () while (*ptr != ';' && ptr[1] != '\0'); break; } + + // FIXME: 2'nd argument should be "current loader" while (--num_arrays >= 0) - type = _Jv_FindArrayClass (type); + type = _Jv_FindArrayClass (type, 0); *argPtr++ = type; } parameter_types = args; diff --git a/libjava/java/net/natPlainSocketImpl.cc b/libjava/java/net/natPlainSocketImpl.cc index d42b821614f..e652ba7d8e0 100644 --- a/libjava/java/net/natPlainSocketImpl.cc +++ b/libjava/java/net/natPlainSocketImpl.cc @@ -354,7 +354,7 @@ java::net::PlainSocketImpl::getOption (jint optID) if (l_val.l_onoff) return new java::lang::Integer (l_val.l_linger); else - return new java::lang::Boolean (false); + return new java::lang::Boolean ((__java_boolean)false); #else JvThrow (new java::lang::InternalError ( JvNewStringUTF ("SO_LINGER not supported"))); diff --git a/libjava/libgcj.spec.in b/libjava/libgcj.spec.in index a46acefed10..e9ee02deb5f 100644 --- a/libjava/libgcj.spec.in +++ b/libjava/libgcj.spec.in @@ -4,7 +4,7 @@ # to link with libgcj. # %rename lib liborig -*lib: -lgcj -lm @GCSPEC@ @THREADSPEC@ @ZLIBSPEC@ @SYSTEMSPEC@ %(liborig) +*lib: -lgcj -lm @INTERPSPEC@ @GCSPEC@ @THREADSPEC@ @ZLIBSPEC@ @SYSTEMSPEC@ %(liborig) %rename cc1 cc1orig *cc1: @DIVIDESPEC@ %(cc1orig) diff --git a/libjava/prims.cc b/libjava/prims.cc index c26c2996a21..90fd96de9f3 100644 --- a/libjava/prims.cc +++ b/libjava/prims.cc @@ -180,6 +180,24 @@ _Jv_makeUtf8Const (char* s, int len) return (m); } +_Jv_Utf8Const * +_Jv_makeUtf8Const (jstring string) +{ + jint hash = string->hashCode (); + jint len = _Jv_GetStringUTFLength (string); + + Utf8Const* m = (Utf8Const*) + _Jv_AllocBytesChecked (sizeof(Utf8Const) + len + 1); + + m->hash = hash; + m->length = len; + + _Jv_GetStringUTFRegion (string, 0, string->length (), m->data); + m->data[len] = 0; + + return m; +} + #ifdef DEBUG @@ -298,7 +316,10 @@ _Jv_NewObjectArray (jsize count, jclass elementClass, jobject init) JvThrow (no_memory); size_t size = count * sizeof (jobject) + sizeof (__JArray); - jclass clas = _Jv_FindArrayClass (elementClass); + + // FIXME: second argument should be "current loader" // + jclass clas = _Jv_FindArrayClass (elementClass, 0); + jobjectArray obj = (jobjectArray) _Jv_AllocArray (size); if (! obj) JvThrow (no_memory); @@ -338,7 +359,7 @@ _Jv_NewPrimArray (jclass eltype, jint count) arr->length = count; // Note that we assume we are given zeroed memory by the allocator. - jclass klass = _Jv_FindArrayClass (eltype); + jclass klass = _Jv_FindArrayClass (eltype, 0); // Set the vtbl last to avoid problems if the GC happens during the // window in this function between the allocation and this // assignment. @@ -531,9 +552,11 @@ _Jv_FindClassFromSignature (char *sig, java::lang::ClassLoader *loader) ; _Jv_Utf8Const *name = _Jv_makeUtf8Const (&sig[1], i - 1); return _Jv_FindClass (name, loader); + } case '[': - return _Jv_FindArrayClass (_Jv_FindClassFromSignature (&sig[1], loader)); + return _Jv_FindArrayClass (_Jv_FindClassFromSignature (&sig[1], loader), + loader); } JvFail ("couldn't understand class signature"); return NULL; // Placate compiler. @@ -588,9 +611,20 @@ JvRunMain (jclass klass, int argc, const char **argv) LTDL_SET_PRELOADED_SYMBOLS (); #endif - arg_vec = JvConvertArgv (argc - 1, argv + 1); - main_group = new java::lang::ThreadGroup (23); - main_thread = new java::lang::FirstThread (main_group, klass, arg_vec); + if (klass == NULL) + { + arg_vec = JvConvertArgv (argc - 2, argv + 2); + main_group = new java::lang::ThreadGroup (23); + main_thread = new java::lang::FirstThread (main_group, + JvNewStringLatin1 (argv[1]), + arg_vec); + } + else + { + arg_vec = JvConvertArgv (argc - 1, argv + 1); + main_group = new java::lang::ThreadGroup (23); + main_thread = new java::lang::FirstThread (main_group, klass, arg_vec); + } main_thread->start(); _Jv_ThreadWait (); @@ -598,6 +632,7 @@ JvRunMain (jclass klass, int argc, const char **argv) java::lang::Runtime::getRuntime ()->exit (0); } + void * @@ -630,7 +665,7 @@ _Jv_divI (jint dividend, jint divisor) if (divisor == 0) _Jv_Throw (arithexception); - if (dividend == 0x80000000L && divisor == -1) + if (dividend == (jint) 0x80000000L && divisor == -1) return dividend; return dividend / divisor; @@ -642,7 +677,7 @@ _Jv_remI (jint dividend, jint divisor) if (divisor == 0) _Jv_Throw (arithexception); - if (dividend == 0x80000000L && divisor == -1) + if (dividend == (jint) 0x80000000L && divisor == -1) return 0; return dividend % divisor; @@ -654,7 +689,7 @@ _Jv_divJ (jlong dividend, jlong divisor) if (divisor == 0) _Jv_Throw (arithexception); - if (dividend == 0x8000000000000000LL && divisor == -1) + if (dividend == (jlong) 0x8000000000000000LL && divisor == -1) return dividend; return dividend / divisor; @@ -666,9 +701,15 @@ _Jv_remJ (jlong dividend, jlong divisor) if (divisor == 0) _Jv_Throw (arithexception); - if (dividend == 0x8000000000000000LL && divisor == -1) + if (dividend == (jlong) 0x8000000000000000LL && divisor == -1) return 0; return dividend % divisor; } + + + + + + diff --git a/libjava/resolve.cc b/libjava/resolve.cc new file mode 100644 index 00000000000..a2d61c96210 --- /dev/null +++ b/libjava/resolve.cc @@ -0,0 +1,1101 @@ +// resolve.cc - Code for linking and resolving classes and pool entries. + +/* Copyright (C) 1999 Cygnus Solutions + + This file is part of libgcj. + +This software is copyrighted work licensed under the terms of the +Libgcj License. Please consult the file "LIBGCJ_LICENSE" for +details. */ + +/* Author: Kresten Krab Thorup <krab@gnu.org> */ + +#include <java-interp.h> + +#ifdef INTERPRETER + +#include <cni.h> +#include <jvm.h> +#include <string.h> +#include <java-cpool.h> +#include <java/lang/Class.h> +#include <java/lang/String.h> +#include <java/lang/Thread.h> +#include <java/lang/InternalError.h> +#include <java/lang/VirtualMachineError.h> +#include <java/lang/NoSuchFieldError.h> +#include <java/lang/ClassFormatError.h> +#include <java/lang/IllegalAccessError.h> +#include <java/lang/AbstractMethodError.h> +#include <java/lang/ClassNotFoundException.h> +#include <java/lang/IncompatibleClassChangeError.h> + +static void throw_internal_error (char *msg) + __attribute__ ((__noreturn__)); +static void throw_class_format_error (jstring msg) + __attribute__ ((__noreturn__)); +static void throw_class_format_error (char *msg) + __attribute__ ((__noreturn__)); + +#define StringClass _CL_Q34java4lang6String +extern java::lang::Class StringClass; +#define ClassObject _CL_Q34java4lang6Object +extern java::lang::Class ClassObject; +#define ObjectClass _CL_Q34java4lang6Object +extern java::lang::Class ObjectClass; + + +static int get_alignment_from_class (jclass); + +static _Jv_ResolvedMethod* +_Jv_BuildResolvedMethod (_Jv_Method*, + jclass, + jboolean, + jint); + + +static const int PUBLIC = 0x001; +static const int PRIVATE = 0x002; +static const int PROTECTED = 0x004; +static const int STATIC = 0x008; +static const int FINAL = 0x010; +static const int SYNCHRONIZED = 0x020; +static const int VOLATILE = 0x040; +static const int TRANSIENT = 0x080; +static const int NATIVE = 0x100; +static const int INTERFACE = 0x200; +static const int ABSTRACT = 0x400; +static const int ALL_FLAGS = 0x7FF; + +// We need to know the name of a constructor. +static _Jv_Utf8Const *init_name = _Jv_makeUtf8Const ("<init>", 6); + +static void throw_incompatible_class_change_error (jstring msg) +{ + JvThrow (new java::lang::IncompatibleClassChangeError (msg)); +} + +void* +_Jv_ResolvePoolEntry (jclass klass, int index) +{ + _Jv_Constants *pool = &klass->constants; + + if ((pool->tags[index] & JV_CONSTANT_ResolvedFlag) != 0) + return pool->data[index]; + + switch (pool->tags[index]) { + case JV_CONSTANT_Class: + { + _Jv_Utf8Const *name = (_Jv_Utf8Const *) pool->data[index]; + + 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); + JvThrow (new java::lang::ClassNotFoundException (str)); + } + + if ((found->accflags & PUBLIC) == PUBLIC + || (_Jv_ClassNameSamePackage (found->name, + klass->name))) + { + pool->data[index] = (void *) found; + pool->tags[index] |= JV_CONSTANT_ResolvedFlag; + } + else + { + JvThrow (new java::lang::IllegalAccessError (found->getName())); + } + } + break; + + case JV_CONSTANT_String: + { + jstring str; + str = _Jv_NewStringUtf8Const ((_Jv_Utf8Const *) pool->data[index]); + pool->data[index] = (void *) str; + pool->tags[index] |= JV_CONSTANT_ResolvedFlag; + } + break; + + case JV_CONSTANT_Fieldref: + { + _Jv_ushort class_index, name_and_type_index; + _Jv_loadIndexes ((const void**) &pool->data[index], + class_index, + name_and_type_index); + jclass owner = (jclass) _Jv_ResolvePoolEntry (klass, class_index); + + if (owner != klass) + _Jv_InitClass (owner); + + _Jv_ushort name_index, type_index; + _Jv_loadIndexes ((const void**) &pool->data[name_and_type_index], + name_index, + type_index); + + _Jv_Utf8Const *field_name = (_Jv_Utf8Const*) pool->data[name_index]; + _Jv_Utf8Const *field_type_name = + (_Jv_Utf8Const*) pool->data[type_index]; + + // FIXME: The implementation of this function + // (_Jv_FindClassFromSignature) will generate an instance of + // _Jv_Utf8Const for each call if the field type is a class name + // (Lxx.yy.Z;). This may be too expensive to do for each and + // every fieldref being resolved. For now, we fix the problem by + // only doing it when we have a loader different from the class + // declaring the field. + + jclass field_type = 0; + + if (owner->loader != klass->loader) + field_type = _Jv_FindClassFromSignature (field_type_name->data, + klass->loader); + + _Jv_Field* the_field = 0; + + for (jclass cls = owner; cls != 0; cls = cls->getSuperclass ()) + { + for (int i = 0; i < cls->field_count; i++) + { + _Jv_Field *field = &cls->fields[i]; + if (! _Jv_equalUtf8Consts (field->name, field_name)) + continue; + + // now, check field access. + + if ( (cls == klass) + || ((field->flags & PUBLIC) != 0) + || (((field->flags & PROTECTED) != 0) + && cls->isAssignableFrom (klass)) + || (((field->flags & PRIVATE) == 0) + && _Jv_ClassNameSamePackage (cls->name, + klass->name))) + { + /* resove the field using the class' own loader + if necessary */ + + if (!field->isResolved ()) + _Jv_ResolveField (field, cls->loader); + + if (field_type != 0 && field->type != field_type) + JvThrow + (new java::lang::LinkageError + (JvNewStringLatin1 + ("field type mismatch with different loaders"))); + + the_field = field; + goto end_of_field_search; + } + else + { + JvThrow (new java::lang::IllegalAccessError); + } + } + } + + end_of_field_search: + if (the_field == 0) + { + jstring msg = JvNewStringLatin1 ("field "); + msg = msg->concat (owner->getName ()); + msg = msg->concat (JvNewStringLatin1(".")); + msg = msg->concat (_Jv_NewStringUTF (field_name->data)); + msg = msg->concat (JvNewStringLatin1(" was not found.")); + throw_incompatible_class_change_error (msg); + } + + pool->data[index] = (void*)the_field; + pool->tags[index] |= JV_CONSTANT_ResolvedFlag; + } + break; + + case JV_CONSTANT_Methodref: + case JV_CONSTANT_InterfaceMethodref: + { + _Jv_ushort class_index, name_and_type_index; + _Jv_loadIndexes ((const void**) &pool->data[index], + class_index, + name_and_type_index); + jclass owner = (jclass) _Jv_ResolvePoolEntry (klass, class_index); + + if (owner != klass) + _Jv_InitClass (owner); + + _Jv_ushort name_index, type_index; + _Jv_loadIndexes ((const void**) &pool->data[name_and_type_index], + name_index, + type_index); + + _Jv_Utf8Const *method_name = (_Jv_Utf8Const*) pool->data[name_index]; + _Jv_Utf8Const *method_signature = + (_Jv_Utf8Const*) pool->data[type_index]; + + int vtable_index = -1; + _Jv_Method *the_method = 0; + jclass found_class = 0; + + // we make a loop here, because methods are allowed to be moved to + // a super class, and still be visible.. (binary compatibility). + + for (jclass cls = owner; cls != 0; cls = cls->getSuperclass ()) + { + for (int i = 0; i < cls->method_count; i++) + { + _Jv_Method *method = &cls->methods[i]; + if ( (!_Jv_equalUtf8Consts (method->name, + method_name)) + || (!_Jv_equalUtf8Consts (method->signature, + method_signature))) + continue; + + if (cls == klass + || ((method->accflags & PUBLIC) != 0) + || (((method->accflags & PROTECTED) != 0) + && cls->isAssignableFrom (klass)) + || (((method->accflags & PRIVATE) == 0) + && _Jv_ClassNameSamePackage (cls->name, + klass->name))) + { + // FIXME: if (cls->loader != klass->loader), then we + // must actually check that the types of arguments + // correspond. That is, for each argument type, and + // the return type, doing _Jv_FindClassFromSignature + // with either loader should produce the same result, + // i.e., exactly the same jclass object. JVMS 5.4.3.3 + + the_method = method; + found_class = cls; + + + if (pool->tags[index] == JV_CONSTANT_InterfaceMethodref) + vtable_index = -1; + else + vtable_index = _Jv_DetermineVTableIndex + (cls, method_name, method_signature); + + if (vtable_index == 0) + throw_incompatible_class_change_error + (JvNewStringLatin1 ("method not found")); + + goto end_of_method_search; + } + else + { + JvThrow (new java::lang::IllegalAccessError); + } + } + } + + end_of_method_search: + if (the_method == 0) + { + jstring msg = JvNewStringLatin1 ("method "); + msg = msg->concat (owner->getName ()); + msg = msg->concat (JvNewStringLatin1(".")); + msg = msg->concat (_Jv_NewStringUTF (method_name->data)); + msg = msg->concat (JvNewStringLatin1(" was not found.")); + JvThrow(new java::lang::NoSuchFieldError (msg)); + } + + pool->data[index] = (void*) + _Jv_BuildResolvedMethod(the_method, + found_class, + ((the_method->accflags & STATIC) != 0), + vtable_index); + pool->tags[index] |= JV_CONSTANT_ResolvedFlag; + } + break; + + } + + return pool->data[index]; +} + +void +_Jv_ResolveField (_Jv_Field *field, java::lang::ClassLoader *loader) +{ + if (! field->isResolved ()) + { + _Jv_Utf8Const *sig = (_Jv_Utf8Const*)field->type; + field->type = _Jv_FindClassFromSignature (sig->data, loader); + field->flags &= ~_Jv_FIELD_UNRESOLVED_FLAG; + } +} + +_Jv_Method* +_Jv_LookupDeclaredMethod (jclass klass, _Jv_Utf8Const *name, + _Jv_Utf8Const *signature) +{ + for (; klass; klass = klass->getSuperclass()) + { + _Jv_Method *meth = _Jv_GetMethodLocal (klass, name, signature); + + if (meth) + return meth; + } + + return NULL; +} + +/** FIXME: this is a terribly inefficient algorithm! It would improve + things if compiled classes to know vtable offset, and _Jv_Method had + a field for this. + + Returns 0 if this class does not declare the given method. + Returns -1 if the given method does not appear in the vtable. + i.e., it is static, private, final or a constructor. + Otherwise, returns the vtable index. */ +int +_Jv_DetermineVTableIndex (jclass klass, + _Jv_Utf8Const *name, + _Jv_Utf8Const *signature) +{ + jclass super_class = klass->getSuperclass (); + + if (super_class != NULL) + { + int prev = _Jv_DetermineVTableIndex (super_class, + name, + signature); + if (prev != 0) + return prev; + } + + /* at this point, we know that the super-class does not declare + * the method. Otherwise, the above call would have found it, and + * determined the result of this function (-1 or some positive + * number). + */ + + _Jv_Method *meth = _Jv_GetMethodLocal (klass, name, signature); + + /* now, if we do not declare this method, return zero */ + if (meth == NULL) + return 0; + + /* so now, we know not only that the super class does not declare the + * method, but we do! So, this is a first declaration of the method. */ + + /* now, the checks for things that are declared in this class, but do + * not go into the vtable. There are three cases. + * 1) the method is static, private or final + * 2) the class itself is final, or + * 3) it is the method <init> + */ + + if ( (meth->accflags & (STATIC|PRIVATE|FINAL)) != 0 + || (klass->accflags & FINAL) != 0 + || _Jv_equalUtf8Consts (name, init_name)) + return -1; + + /* reaching this point, we know for sure, that the method in question + * will be in the vtable. The question is where. */ + + /* the base offset, is where we will start assigning vtable + * indexes for this class. It is 1 for base classes + * (vtable->method[0] is unused), and for non-base classes it is the + * number of entries in the super class' vtable plus 1. */ + + int base_offset; + if (super_class == 0) + base_offset = 1; + else + base_offset = super_class->vtable_method_count+1; + + /* we will consider methods 0..this_method_index-1. And for each one, + * determine if it is new (i.e., if it appears in the super class), + * and if it should go in the vtable. If so, increment base_offset */ + + int this_method_index = meth - (&klass->methods[0]); + + for (int i = 0; i < this_method_index; i++) + { + _Jv_Method *m = &klass->methods[i]; + + /* fist some checks for things that surely do not go in the + * vtable */ + + if ((m->accflags & (STATIC|PRIVATE)) != 0) + continue; + if (_Jv_equalUtf8Consts (m->name, init_name)) + continue; + + /* Then, we need to know if this method appears in the + superclass. (This is where this function gets expensive) */ + _Jv_Method *sm = _Jv_LookupDeclaredMethod (super_class, + m->name, + m->signature); + + /* if it was somehow declared in the superclass, skip this */ + if (sm != NULL) + continue; + + /* but if it is final, and not declared in the super class, + * then we also skip it */ + if ((m->accflags & FINAL) != 0) + continue; + + /* finally, we can assign the index of this method */ + /* m->vtable_index = base_offset */ + base_offset += 1; + } + + return base_offset; +} + +/* this is installed in place of abstract methods */ +static void +_Jv_abstractMethodError () +{ + JvThrow (new java::lang::AbstractMethodError); +} + +void +_Jv_PrepareClass(jclass klass) +{ + /* + * The job of this function is to: 1) assign storage to fields, and 2) + * build the vtable. static fields are assigned real memory, instance + * fields are assigned offsets. + * + * NOTE: we have a contract with the garbage collector here. Static + * reference fields must not be resolved, until after they have storage + * assigned which is the check used by the collector to see if it + * should indirect the static field reference and mark the object + * pointed to. + * + * Most fields are resolved lazily (i.e. have their class-type + * assigned) when they are accessed the first time by calling as part + * of _Jv_ResolveField, which is allways called after _Jv_PrepareClass. + * Static fields with initializers are resolved as part of this + * function, as are fields with primitive types. + */ + + if (! _Jv_IsInterpretedClass (klass)) + return; + + if (klass->state >= JV_STATE_PREPARED) + return; + + // make sure super-class is linked. This involves taking a lock on + // the super class, so we use the Java method resolveClass, which will + // unlock it properly, should an exception happen. + + java::lang::ClassLoader::resolveClass (klass->superclass); + + _Jv_InterpClass *clz = (_Jv_InterpClass*)klass; + + /************ PART ONE: OBJECT LAYOUT ***************/ + + int instance_size; + int static_size; + + // java.lang.Object is never interpreted! + instance_size = clz->superclass->size (); + static_size = 0; + + for (int i = 0; i < clz->field_count; i++) + { + int field_size; + int field_align; + + _Jv_Field *field = &clz->fields[i]; + + if (! field->isRef ()) + { + // it's safe to resolve the field here, since it's + // a primitive class, which does not cause loading to happen. + _Jv_ResolveField (field, clz->loader); + + field_size = field->type->size (); + field_align = get_alignment_from_class (field->type); + } + else + { + field_size = sizeof (jobject); + field_align = __alignof__ (jobject); + } + +#ifndef COMPACT_FIELDS + field->bsize = field_size; +#endif + + if (field->flags & STATIC) + { + /* this computes an offset into a region we'll allocate + shortly, and then add this offset to the start address */ + + static_size = ROUND (static_size, field_align); + field->u.boffset = static_size; + static_size += field_size; + } + else + { + instance_size = ROUND (instance_size, field_align); + field->u.boffset = instance_size; + instance_size += field_size; + } + } + + // set the instance size for the class + clz->size_in_bytes = instance_size; + + // allocate static memory + if (static_size != 0) + { + char *static_data = (char*)_Jv_AllocBytesChecked (static_size); + + memset (static_data, 0, static_size); + + for (int i = 0; i < clz->field_count; i++) + { + _Jv_Field *field = &clz->fields[i]; + + if ((field->flags & STATIC) != 0) + { + field->u.addr = static_data + field->u.boffset; + + if (clz->field_initializers[i] != 0) + { + _Jv_ResolveField (field, clz->loader); + _Jv_InitField (0, clz, i); + } + } + } + + // now we don't need the field_initializers anymore, so let the + // collector get rid of it! + + clz->field_initializers = 0; + } + + /************ PART TWO: VTABLE LAYOUT ***************/ + + /* preparation: build the vtable stubs (even interfaces can) + have code -- for static constructors. */ + for (int i = 0; i < clz->method_count; i++) + { + _Jv_InterpMethod *imeth = clz->interpreted_methods[i]; + + if (imeth != 0) // it could be abstract or native + { + clz->methods[i].ncode = imeth->ncode (); + } + else + { + if ((clz->methods[i].accflags & NATIVE) != 0) + { + JvThrow + (new java::lang::VirtualMachineError + (JvNewStringLatin1 + ("the interpreter does not support native methods"))); + } + } + } + + if (clz->accflags & INTERFACE) + { + clz->state = JV_STATE_PREPARED; + clz->notifyAll (); + return; + } + + /* FIXME: native methods for interpreted classes should be handled, I + * dunno exactly how, but it seems that we should try to find them at + * this point, and if we fail, try again after <clinit>, since it + * could have caused additional code to be loaded. Interfaces cannot + * have native methods (not even for static initialization). */ + + + /* Now onto the actual job: vtable layout. First, count how many new + methods we have */ + int new_method_count = 0; + + jclass super_class = clz->getSuperclass (); + + if (super_class == 0) + throw_internal_error ("cannot handle interpreted base classes"); + + for (int i = 0; i < clz->method_count; i++) + { + _Jv_Method *this_meth = &clz->methods[i]; + + if ((this_meth->accflags & (STATIC|PRIVATE)) != 0 + || _Jv_equalUtf8Consts (this_meth->name, init_name)) + { + /* skip this, it doesn't go in the vtable */ + continue; + } + + _Jv_Method *orig_meth = _Jv_LookupDeclaredMethod (super_class, + this_meth->name, + this_meth->signature); + + if (orig_meth == 0) + { + // new methods that are final, also don't go in the vtable + if ((this_meth->accflags & FINAL) != 0) + continue; + + new_method_count += 1; + continue; + } + + if ((orig_meth->accflags & (STATIC|PRIVATE|FINAL)) != 0 + || ((orig_meth->accflags & ABSTRACT) == 0 + && (this_meth->accflags & ABSTRACT) != 0 + && (klass->accflags & ABSTRACT) == 0)) + { + clz->state = JV_STATE_ERROR; + clz->notifyAll (); + JvThrow (new java::lang::IncompatibleClassChangeError + (clz->getName ())); + } + + /* FIXME: At this point, if (loader != super_class->loader), we + * need to "impose class loader constraints" for the types + * involved in the signature of this method */ + } + + /* determine size */ + int vtable_count = (super_class->vtable_method_count) + new_method_count; + clz->vtable_method_count = vtable_count; + + /* allocate vtable structure */ + _Jv_VTable *vtable = (_Jv_VTable*) + _Jv_AllocBytesChecked (sizeof (_Jv_VTable) + + (sizeof (void*) * (vtable_count))); + vtable->clas = clz; + + /* copy super class' vtable entries (index 0 goes unused). */ + memcpy ((void*)&vtable->method[1], + (void*)&super_class->vtable->method[1], + sizeof (void*) * super_class->vtable_method_count); + + /* now, install our own vtable entries, reprise... */ + for (int i = 0; i < clz->method_count; i++) + { + _Jv_Method *this_meth = &clz->methods[i]; + + int index = _Jv_DetermineVTableIndex (clz, + this_meth->name, + this_meth->signature); + + if (index == 0) + throw_internal_error ("method now found in own class"); + + if (index != -1) + { + if (index > clz->vtable_method_count+1) + throw_internal_error ("vtable problem..."); + + if (clz->interpreted_methods[i] == 0) + vtable->method[index] = (void*)&_Jv_abstractMethodError; + else + vtable->method[index] = this_meth->ncode; + } + } + + /* finally, assign the vtable! */ + clz->vtable = vtable; + + /* wooha! we're done. */ + clz->state = JV_STATE_PREPARED; + clz->notifyAll (); +} + +/** Do static initialization for fields with a constant initializer */ +void +_Jv_InitField (jobject obj, jclass klass, int index) +{ + if (obj != 0 && klass == 0) + klass = obj->getClass (); + + if (!_Jv_IsInterpretedClass (klass)) + return; + + _Jv_InterpClass *clz = (_Jv_InterpClass*)klass; + + _Jv_Field * field = (&clz->fields[0]) + index; + + if (index > clz->field_count) + throw_internal_error ("field out of range"); + + int init = clz->field_initializers[index]; + if (init == 0) + return; + + _Jv_Constants *pool = &clz->constants; + int tag = pool->tags[init]; + + if (! field->isResolved ()) + throw_internal_error ("initializing unresolved field"); + + if (obj==0 && ((field->flags & STATIC) == 0)) + throw_internal_error ("initializing non-static field with no object"); + + void *addr = 0; + + if ((field->flags & STATIC) != 0) + addr = (void*) field->u.addr; + else + addr = (void*) (((char*)obj) + field->u.boffset); + + switch (tag) + { + case JV_CONSTANT_String: + { + _Jv_MonitorEnter (clz); + jstring str; + str = _Jv_NewStringUtf8Const ((_Jv_Utf8Const *) pool->data[init]); + pool->data[init] = (void *) str; + pool->tags[init] = JV_CONSTANT_ResolvedString; + _Jv_MonitorExit (clz); + } + /* fall through */ + + case JV_CONSTANT_ResolvedString: + if (! (field->type == &StringClass || field->type == &ObjectClass)) + throw_class_format_error ("string initialiser to non-string field"); + + *(jstring*)addr = *(jstring*) (pool->data + init); + break; + + case JV_CONSTANT_Integer: + { + int value = *(jint*)(pool->data + init); + + if (field->type == JvPrimClass (boolean)) + *(jboolean*)addr = (jboolean)value; + + else if (field->type == JvPrimClass (byte)) + *(jbyte*)addr = (jbyte)value; + + else if (field->type == JvPrimClass (char)) + *(jchar*)addr = (jchar)value; + + else if (field->type == JvPrimClass (short)) + *(jshort*)addr = (jshort)value; + + else if (field->type == JvPrimClass (int)) + *(jint*)addr = (jint)value; + + else + throw_class_format_error ("erroneous field initializer"); + } + break; + + case JV_CONSTANT_Long: + if (field->type != JvPrimClass (long)) + throw_class_format_error ("erroneous field initializer"); + + memcpy (addr, pool->data+init, 8); + break; + + case JV_CONSTANT_Float: + if (field->type != JvPrimClass (float)) + throw_class_format_error ("erroneous field initializer"); + + memcpy (addr, pool->data+init, 4); + break; + + case JV_CONSTANT_Double: + if (field->type != JvPrimClass (double)) + throw_class_format_error ("erroneous field initializer"); + + memcpy (addr, pool->data+init, 8); + break; + + default: + throw_class_format_error ("erroneous field initializer"); + } +} + +static int +get_alignment_from_class (jclass klass) +{ + if (klass == JvPrimClass (byte)) + return __alignof__ (jbyte); + else if (klass == JvPrimClass (short)) + return __alignof__ (jshort); + else if (klass == JvPrimClass (int)) + return __alignof__ (jint); + else if (klass == JvPrimClass (long)) + return __alignof__ (jlong); + else if (klass == JvPrimClass (boolean)) + return __alignof__ (jboolean); + else if (klass == JvPrimClass (char)) + return __alignof__ (jchar); + else if (klass == JvPrimClass (float)) + return __alignof__ (jfloat); + else if (klass == JvPrimClass (double)) + return __alignof__ (jdouble); + else + return __alignof__ (jobject); +} + + +inline static unsigned char* +skip_one_type (unsigned char* ptr) +{ + int ch = *ptr++; + + while (ch == '[') + { + ch = *ptr++; + } + + if (ch == 'L') + { + do { ch = *ptr++; } while (ch != ';'); + } + + return ptr; +} + +static ffi_type* +get_ffi_type_from_signature (unsigned char* ptr) +{ + switch (*ptr) + { + case 'L': + case '[': + return &ffi_type_pointer; + break; + + case 'Z': + case 'B': + return &ffi_type_sint8; + break; + + case 'C': + return &ffi_type_uint16; + break; + + case 'S': + return &ffi_type_sint16; + break; + + case 'I': + return &ffi_type_sint32; + break; + + case 'J': + return &ffi_type_sint64; + break; + + case 'F': + return &ffi_type_float; + break; + + case 'D': + return &ffi_type_double; + break; + + case 'V': + return &ffi_type_void; + break; + } + + throw_internal_error ("unknown type in signature"); +} + +/* this function yields the number of actual arguments, that is, if the + * function is non-static, then one is added to the number of elements + * found in the signature */ + +static int +count_arguments (_Jv_Utf8Const *signature, + jboolean staticp) +{ + unsigned char *ptr = (unsigned char*) signature->data; + int arg_count = staticp ? 0 : 1; + + /* first, count number of arguments */ + + // skip '(' + ptr++; + + // count args + while (*ptr != ')') + { + ptr = skip_one_type (ptr); + arg_count += 1; + } + + return arg_count; +} + +/* This beast will build a cif, given the signature. Memory for + * the cif itself and for the argument types must be allocated by the + * caller. + */ + +static int +init_cif (_Jv_Utf8Const* signature, + int arg_count, + jboolean staticp, + ffi_cif *cif, + ffi_type **arg_types) +{ + unsigned char *ptr = (unsigned char*) signature->data; + + int arg_index = 0; // arg number + int item_count = 0; // stack-item count + + // setup receiver + if (!staticp) + { + arg_types[arg_index++] = &ffi_type_pointer; + item_count += 1; + } + + // skip '(' + ptr++; + + // assign arg types + while (*ptr != ')') + { + arg_types[arg_index++] = get_ffi_type_from_signature (ptr); + + if (*ptr == 'J' || *ptr == 'D') + item_count += 2; + else + item_count += 1; + + ptr = skip_one_type (ptr); + } + + // skip ')' + ptr++; + ffi_type *rtype = get_ffi_type_from_signature (ptr); + + ptr = skip_one_type (ptr); + if (ptr != (unsigned char*)signature->data + signature->length) + throw_internal_error ("did not find end of signature"); + + if (ffi_prep_cif (cif, FFI_DEFAULT_ABI, + arg_count, rtype, arg_types) != FFI_OK) + throw_internal_error ("ffi_prep_cif failed"); + + return item_count; +} + + +/* we put this one here, and not in interpret.cc because it + * calls the utility routines count_arguments + * which are static to this module. The following struct defines the + * layout we use for the stubs, it's only used in the ncode method. */ + +typedef struct { + ffi_raw_closure closure; + ffi_cif cif; + ffi_type *arg_types[0]; +} ncode_closure; + +typedef void (*ffi_closure_fun) (ffi_cif*,void*,ffi_raw*,void*); + +void* _Jv_InterpMethod::ncode () +{ + if (self->ncode != 0) + return self->ncode; + + jboolean staticp = (self->accflags & STATIC) != 0; + int arg_count = count_arguments (self->signature, staticp); + + ncode_closure *closure = + (ncode_closure*)_Jv_AllocBytesChecked (sizeof (ncode_closure) + + arg_count * sizeof (ffi_type*)); + + init_cif (self->signature, + arg_count, + staticp, + &closure->cif, + &closure->arg_types[0]); + + ffi_closure_fun fun; + + args_raw_size = ffi_raw_size (&closure->cif); + + if ((self->accflags & SYNCHRONIZED) != 0) + { + if (staticp) + fun = (ffi_closure_fun)&_Jv_InterpMethod::run_synch_class; + else + fun = (ffi_closure_fun)&_Jv_InterpMethod::run_synch_object; + } + else + { + fun = (ffi_closure_fun)&_Jv_InterpMethod::run_normal; + } + + ffi_prep_raw_closure (&closure->closure, + &closure->cif, + fun, + (void*)this); + + self->ncode = (void*)closure; + return self->ncode; +} + + +/* A _Jv_ResolvedMethod is what is put in the constant pool for a + * MethodRef or InterfacemethodRef. */ +static _Jv_ResolvedMethod* +_Jv_BuildResolvedMethod (_Jv_Method* method, + jclass klass, + jboolean staticp, + jint vtable_index) +{ + int arg_count = count_arguments (method->signature, staticp); + + _Jv_ResolvedMethod* result = (_Jv_ResolvedMethod*) + _Jv_AllocBytesChecked (sizeof (_Jv_ResolvedMethod) + + arg_count*sizeof (ffi_type*)); + + result->stack_item_count + = init_cif (method->signature, + arg_count, + staticp, + &result->cif, + &result->arg_types[0]); + + result->vtable_index = vtable_index; + result->method = method; + result->klass = klass; + + return result; +} + + +static void +throw_class_format_error (jstring msg) +{ + if (msg == 0) + JvThrow (new java::lang::ClassFormatError); + else + JvThrow (new java::lang::ClassFormatError (msg)); +} + +static void +throw_class_format_error (char *msg) +{ + throw_class_format_error (JvNewStringLatin1 (msg)); +} + +static void +throw_internal_error (char *msg) +{ + JvThrow + (new java::lang::InternalError (JvNewStringLatin1 (msg))); +} + + +#endif diff --git a/libjava/testsuite/Makefile.in b/libjava/testsuite/Makefile.in index 139abea6854..ac2e92d1b0d 100644 --- a/libjava/testsuite/Makefile.in +++ b/libjava/testsuite/Makefile.in @@ -71,6 +71,7 @@ COMPPATH = @COMPPATH@ CPP = @CPP@ CXX = @CXX@ CXXCPP = @CXXCPP@ +DIVIDESPEC = @DIVIDESPEC@ DLLTOOL = @DLLTOOL@ EH_COMMON_INCLUDE = @EH_COMMON_INCLUDE@ EXEEXT = @EXEEXT@ @@ -79,6 +80,7 @@ GCINCS = @GCINCS@ GCLIBS = @GCLIBS@ GCOBJS = @GCOBJS@ GCSPEC = @GCSPEC@ +INTERPSPEC = @INTERPSPEC@ LD = @LD@ LIBGCJ_CFLAGS = @LIBGCJ_CFLAGS@ LIBGCJ_CXXFLAGS = @LIBGCJ_CXXFLAGS@ |