Initial revision
authorAnthony Green <green@gcc.gnu.org>
Sun, 6 Sep 1998 15:36:06 +0000 (15:36 +0000)
committerAnthony Green <green@gcc.gnu.org>
Sun, 6 Sep 1998 15:36:06 +0000 (15:36 +0000)
From-SVN: r22299

43 files changed:
gcc/java/Make-lang.in [new file with mode: 0644]
gcc/java/Makefile.in [new file with mode: 0644]
gcc/java/buffer.c [new file with mode: 0644]
gcc/java/buffer.h [new file with mode: 0644]
gcc/java/class.c [new file with mode: 0644]
gcc/java/config-lang.in [new file with mode: 0644]
gcc/java/constants.c [new file with mode: 0644]
gcc/java/convert.h [new file with mode: 0644]
gcc/java/decl.c [new file with mode: 0644]
gcc/java/except.c [new file with mode: 0644]
gcc/java/expr.c [new file with mode: 0644]
gcc/java/gjavah.c [new file with mode: 0644]
gcc/java/java-except.h [new file with mode: 0644]
gcc/java/java-opcodes.h [new file with mode: 0644]
gcc/java/java-tree.def [new file with mode: 0644]
gcc/java/java-tree.h [new file with mode: 0644]
gcc/java/javaop.def [new file with mode: 0644]
gcc/java/javaop.h [new file with mode: 0644]
gcc/java/jcf-dump.c [new file with mode: 0644]
gcc/java/jcf-io.c [new file with mode: 0644]
gcc/java/jcf-parse.c [new file with mode: 0644]
gcc/java/jcf-reader.c [new file with mode: 0644]
gcc/java/jcf-write.c [new file with mode: 0644]
gcc/java/jcf.h [new file with mode: 0644]
gcc/java/jv-scan.c [new file with mode: 0644]
gcc/java/jvgenmain.c [new file with mode: 0644]
gcc/java/jvspec.c [new file with mode: 0644]
gcc/java/keyword.gperf [new file with mode: 0644]
gcc/java/keyword.h [new file with mode: 0644]
gcc/java/lang-options.h [new file with mode: 0644]
gcc/java/lang-specs.h [new file with mode: 0644]
gcc/java/lang.c [new file with mode: 0644]
gcc/java/lex.c [new file with mode: 0644]
gcc/java/lex.h [new file with mode: 0644]
gcc/java/mangle.c [new file with mode: 0644]
gcc/java/parse-scan.y [new file with mode: 0644]
gcc/java/parse.c [new file with mode: 0644]
gcc/java/parse.h [new file with mode: 0644]
gcc/java/parse.y [new file with mode: 0644]
gcc/java/typeck.c [new file with mode: 0644]
gcc/java/verify.c [new file with mode: 0644]
gcc/java/zextract.c [new file with mode: 0644]
gcc/java/zipfile.h [new file with mode: 0644]

diff --git a/gcc/java/Make-lang.in b/gcc/java/Make-lang.in
new file mode 100644 (file)
index 0000000..de1adf4
--- /dev/null
@@ -0,0 +1,209 @@
+# Top level makefile fragment for the GNU compiler for the Java(TM)
+# language.
+#   Copyright (C) 1996, 1998 Free Software Foundation, Inc.
+
+#This file is part of GNU CC.
+
+#GNU CC is free software; you can redistribute it and/or modify
+#it under the terms of the GNU General Public License as published by
+#the Free Software Foundation; either version 2, or (at your option)
+#any later version.
+
+#GNU CC is distributed in the hope that it will be useful,
+#but WITHOUT ANY WARRANTY; without even the implied warranty of
+#MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+#GNU General Public License for more details.
+
+#You should have received a copy of the GNU General Public License
+#along with GNU CC; see the file COPYING.  If not, write to
+#the Free Software Foundation, 59 Temple Place - Suite 330,
+#Boston, MA 02111-1307, USA.
+
+#Java and all Java-based marks are trademarks or registered trademarks
+#of Sun Microsystems, Inc. in the United States and other countries.
+#The Free Software Foundation is independent of Sun Microsystems, Inc.
+
+# This file provides the language dependent support in the main Makefile.
+# Each language makefile fragment must provide the following targets:
+#
+# foo.all.build, foo.all.cross, foo.start.encap, foo.rest.encap,
+# foo.info, foo.dvi,
+# foo.install-normal, foo.install-common, foo.install-info, foo.install-man,
+# foo.uninstall, foo.distdir,
+# foo.mostlyclean, foo.clean, foo.distclean, foo.extraclean,
+# foo.maintainer-clean, foo.stage1, foo.stage2, foo.stage3, foo.stage4
+#
+# where `foo' is the name of the language.
+#
+# It should also provide rules for:
+#
+# - making any compiler driver (eg: g++)
+# - the compiler proper (eg: jc1)
+# - define the names for selecting the language in LANGUAGES.
+\f
+# Extra flags to pass to recursive makes.
+JAVA_FLAGS_TO_PASS = \
+       "JAVA_FOR_BUILD=$(JAVA_FOR_BUILD)" \
+       "JAVAFLAGS=$(JAVAFLAGS)" \
+       "JAVA_FOR_TARGET=$(JAVA_FOR_TARGET)"
+
+# Actual names to use when installing a native compiler.
+JAVA_INSTALL_NAME = `t='$(program_transform_name)'; echo gcj | sed $$t`
+
+# Actual names to use when installing a cross-compiler.
+JAVA_CROSS_NAME = `t='$(program_transform_cross_name)'; echo gcj | sed $$t`
+
+\f
+# Define the names for selecting java in LANGUAGES.
+java: jc1$(exeext) $(GCJ)$(exeext) jvgenmain$(exeext) gcjh$(exeext)
+
+# Define the name of target independant tools to be installed in $(bindir)
+# Names are subject to changes
+JAVA_TARGET_INDEPENDENT_BIN_TOOLS = gcjh jv-scan jcf-dump
+
+# Tell GNU make to ignore these if they exist.
+.PHONY: java
+
+GCJ = gcj
+
+# Remember to keep this list in sync with JAVA_OBJS in Makefile.in!!!
+#
+JAVA_SRCS = $(srcdir)/java/parse.y $(srcdir)/java/class.c \
+  $(srcdir)/java/decl.c $(srcdir)/java/expr.c $(srcdir)/java/constants.c \
+  $(srcdir)/java/lang.c $(srcdir)/java/typeck.c $(srcdir)/java/except.c \
+  $(srcdir)/java/verify.c $(srcdir)/java/zextract.c $(srcdir)/java/jcf-io.c \
+  $(srcdir)/java/jcf-parse.c $(srcdir)/java/mangle.c \
+  $(srcdir)/java/jcf-write.c $(srcdir)/java/buffer.c
+
+jc1$(exeext): $(P) $(JAVA_SRCS) $(LIBDEPS) stamp-objlist
+       cd java; $(MAKE) $(FLAGS_TO_PASS) $(JAVA_FLAGS_TO_PASS) ../jc1$(exeext)
+
+$(GCJ).c: $(srcdir)/gcc.c
+       -rm -f $@
+       $(LN_S) $(srcdir)/gcc.c $@
+
+jvspec.o: $(srcdir)/java/jvspec.c
+       $(CC) -c -DWITH_THREAD_$(GCC_THREAD_FILE) \
+               $(ALL_CFLAGS) $(ALL_CPPFLAGS) \
+               $(INCLUDES) $(srcdir)/java/jvspec.c
+
+# N.B.: This is a copy of the gcc.o rule, with -DLANG_SPECIFIC_DRIVER added.
+# It'd be nice if we could find an easier way to do this---rather than have
+# to track changes to the toplevel gcc Makefile as well.
+# We depend on $(GCJ).c last, to make it obvious where it came from.
+$(GCJ).o: $(CONFIG_H) multilib.h config.status $(lang_specs_files) $(GCJ).c \
+       system.h
+       $(CC) $(ALL_CFLAGS) $(ALL_CPPFLAGS) $(INCLUDES) \
+       $(DRIVER_DEFINES) \
+       -DLANG_SPECIFIC_DRIVER \
+  -c $(GCJ).c
+
+# Create the compiler driver for $(GCJ).
+$(GCJ)$(exeext): $(GCJ).o jvspec.o version.o choose-temp.o\
+          pexecute.o prefix.o mkstemp.o $(LIBDEPS) $(EXTRA_GCC_OBJS)
+       $(CC) $(ALL_CFLAGS) $(LDFLAGS) -o $@ $(GCJ).o jvspec.o prefix.o \
+         version.o choose-temp.o pexecute.o mkstemp.o $(EXTRA_GCC_OBJS) $(LIBS)
+
+# Create a version of the $(GCJ) driver which calls the cross-compiler.
+$(GCJ)-cross$(exeext): $(GCJ)$(exeext)
+       -rm -f $(GCJ)-cross$(exeext)
+       cp $(GCJ)$(exeext) $(GCJ)-cross$(exeext)
+
+# Dependencies here must be kept in sync with dependencies in Makefile.in.
+jvgenmain$(exeext): $(srcdir)/java/jvgenmain.c $(srcdir)/java/mangle.c \
+               $(OBSTACK)
+       cd java && $(MAKE) $(FLAGS_TO_PASS) $(JAVA_FLAGS_TO_PASS) ../jvgenmain$(exeext)
+
+# This must be kept in sync with dependencies in Makefile.in.
+GCJH_SOURCES = $(srcdir)/java/gjavah.c $(srcdir)/java/jcf-io.c \
+       $(srcdir)/java/zextract.c $(srcdir)/java/jcf-reader.c \
+       $(srcdir)/java/jcf.h $(srcdir)/java/javaop.h \
+       $(srcdir)/java/javaop.def
+
+gcjh$(exeext): $(GCJH_SOURCES)
+       cd java && $(MAKE) $(FLAGS_TO_PASS) $(JAVA_FLAGS_TO_PASS) ../gcjh$(exeext)
+
+
+\f
+# Build hooks:
+
+java.all.build: $(GCJ)$(exeext) jvgenmain$(exeext) gcjh$(exeext)
+java.all.cross: $(GCJ)-cross$(exeext) jvgenmain$(exeext) gcjh$(exeext)
+java.start.encap: $(GCJ)$(exeext)
+java.rest.encap: jvgenmain$(exeext) gcjh$(exeext)
+
+java.info:
+
+# Install hooks:
+# jc1, gcj, jvgenmain, and gcjh are installed elsewhere as part
+# of $(COMPILERS).
+
+# Nothing to do here.
+java.install-normal:
+
+java.install-common:
+       -if [ -f $(GCJ)$(exeext) ]; then \
+         if [ -f $(GCJ)-cross$(exeext) ]; then \
+           rm -f $(bindir)/$(JAVA_CROSS_NAME)$(exeext); \
+           $(INSTALL_PROGRAM) $(GCJ)-cross$(exeext) $(bindir)/$(JAVA_CROSS_NAME)$(exeext); \
+           chmod a+x $(bindir)/$(JAVA_CROSS_NAME)$(exeext); \
+         else \
+           rm -f $(bindir)/$(JAVA_INSTALL_NAME)$(exeext); \
+           $(INSTALL_PROGRAM) $(GCJ)$(exeext) $(bindir)/$(JAVA_INSTALL_NAME)$(exeext); \
+           chmod a+x $(bindir)/$(JAVA_INSTALL_NAME)$(exeext); \
+         fi ; \
+       fi ; \
+        for tool in $(JAVA_TARGET_INDEPENDENT_BIN_TOOLS); do \
+          if [ -f $$tool$(exeext) ]; then \
+           rm -f $(bindir)/$$tool$(exeext); \
+           $(INSTALL_PROGRAM) $$tool$(exeext) $(bindir)/$$tool$(exeext); \
+           chmod a+x $(bindir)/$$tool$(exeext); \
+          fi ; \
+       done
+
+java.install-man:
+
+java.uninstall:
+       -rm -rf $(bindir)/$(JAVA_INSTALL_NAME)$(exeext)
+       -rm -rf $(bindir)/$(JAVA_CROSS_NAME)$(exeext)
+
+java.install-info:
+
+\f
+# Clean hooks:
+# A lot of the ancillary files are deleted by the main makefile.
+# We just have to delete files specific to us.
+
+java.mostlyclean:
+       -rm -f java/*$(objext) $(DEMANGLER_PROG)
+java.clean:
+java.distclean:
+       -rm -f java/config.status java/Makefile
+       -rm -f java/parse.output
+java.extraclean:
+java.maintainer-clean:
+       -rm -f java/parse.h
+\f
+# Stage hooks:
+# The main makefile has already created stage?/java.
+
+java.stage1:
+       -mv java/*$(objext) stage1/java
+java.stage2:
+       -mv java/*$(objext) stage2/java
+java.stage3:
+       -mv java/*$(objext) stage3/java
+java.stage4:
+       -mv java/*$(objext) stage4/java
+\f
+# Maintenance hooks:
+
+# This target creates the files that can be rebuilt, but go in the
+# distribution anyway.  It then copies the files to the distdir directory.
+java.distdir:
+       mkdir tmp/java
+       cd java ; $(MAKE) $(FLAGS_TO_PASS) $(JAVA_FLAGS_TO_PASS) parse.c hash.h
+       cd java; \
+       for file in *[0-9a-zA-Z+]; do \
+         ln $$file ../tmp/java >/dev/null 2>&1 || cp $$file ../tmp/java; \
+       done
diff --git a/gcc/java/Makefile.in b/gcc/java/Makefile.in
new file mode 100644 (file)
index 0000000..6f043dc
--- /dev/null
@@ -0,0 +1,258 @@
+# Makefile for GNU compiler for the Java(TM) language.
+#   Copyright (C) 1987, 88, 90-4, 1995, 1998 Free Software Foundation, Inc.
+
+#This file is part of GNU CC.
+
+#GNU CC is free software; you can redistribute it and/or modify
+#it under the terms of the GNU General Public License as published by
+#the Free Software Foundation; either version 2, or (at your option)
+#any later version.
+
+#GNU CC is distributed in the hope that it will be useful,
+#but WITHOUT ANY WARRANTY; without even the implied warranty of
+#MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+#GNU General Public License for more details.
+
+#You should have received a copy of the GNU General Public License
+#along with GNU CC; see the file COPYING.  If not, write to
+#the Free Software Foundation, 59 Temple Place - Suite 330,
+#Boston, MA 02111-1307, USA.
+
+#Java and all Java-based marks are trademarks or registered trademarks
+#of Sun Microsystems, Inc. in the United States and other countries.
+#The Free Software Foundation is independent of Sun Microsystems, Inc.
+
+# The makefile built from this file lives in the language subdirectory.
+# Its purpose is to provide support for:
+#
+# 1) recursion where necessary, and only then (building .o's), and
+# 2) building and debugging cc1 from the language subdirectory, and
+# 3) nothing else.
+#
+# The parent makefile handles all other chores, with help from the
+# language makefile fragment, of course.
+#
+# The targets for external use are:
+# all, TAGS, ???mostlyclean, ???clean.
+
+# Suppress smart makes who think they know how to automake Yacc files
+.y.c:
+
+# Variables that exist for you to override.
+# See below for how to change them for certain systems.
+
+ALLOCA =
+
+# Various ways of specifying flags for compilations:  
+# CFLAGS is for the user to override to, e.g., do a bootstrap with -O2.
+# BOOT_CFLAGS is the value of CFLAGS to pass
+# to the stage2 and stage3 compilations
+# XCFLAGS is used for most compilations but not when using the GCC just built.
+XCFLAGS =
+CFLAGS = -g
+BOOT_CFLAGS = -O $(CFLAGS)
+# These exists to be overridden by the x-* and t-* files, respectively.
+X_CFLAGS =
+T_CFLAGS =
+
+X_CPPFLAGS =
+T_CPPFLAGS =
+
+CC = @CC@
+BISON = `if [ -f ../../bison/bison ] ; then echo ../../bison/bison -L $(srcdir)/../../bison/ ;  else echo bison ; fi`
+BISONFLAGS =
+JAVABISONFLAGS = --name-prefix=java_
+LEX = `if [ -f ../../flex/flex ] ; then echo ../../flex/flex ;  else echo flex ; fi`
+LEXFLAGS =
+AR = ar
+AR_FLAGS = rc
+SHELL = /bin/sh
+MAKEINFO = makeinfo
+TEXI2DVI = texi2dvi
+
+# Define this as & to perform parallel make on a Sequent.
+# Note that this has some bugs, and it seems currently necessary 
+# to compile all the gen* files first by hand to avoid erroneous results.
+P =
+
+# This is used in the definition of SUBDIR_USE_ALLOCA.
+# ??? Perhaps it would be better if it just looked for *gcc*.
+OLDCC = cc
+
+# This is used instead of ALL_CFLAGS when compiling with GCC_FOR_TARGET.
+# It omits XCFLAGS, and specifies -B./.
+# It also specifies -B$(tooldir)/ to find as and ld for a cross compiler.
+GCC_CFLAGS=$(INTERNAL_CFLAGS) $(X_CFLAGS) $(T_CFLAGS) $(CFLAGS)
+
+# Tools to use when building a cross-compiler.
+# These are used because `configure' appends `cross-make'
+# to the makefile when making a cross-compiler.
+
+# We don't use cross-make.  Instead we use the tools
+# from the build tree, if they are available.
+# program_transform_name and objdir are set by configure.in.
+program_transform_name =
+objdir = .
+
++target=@target@
++xmake_file=@dep_host_xmake_file@
++tmake_file=@dep_tmake_file@
+#version=`sed -e 's/.*\"\([^ \"]*\)[ \"].*/\1/' < $(srcdir)/version.c`
+#mainversion=`sed -e 's/.*\"\([0-9]*\.[0-9]*\).*/\1/' < $(srcdir)/version.c`
+
+# Directory where sources are, from where we are.
+srcdir = @srcdir@
+VPATH = @srcdir@
+
+# Additional system libraries to link with.
+CLIB=
+
+# Change this to a null string if obstacks are installed in the
+# system library.
+OBSTACK=obstack.o
+
+# Choose the real default target.
+ALL=all
+
+# End of variables for you to override.
+
+# Definition of `all' is here so that new rules inserted by sed
+# do not specify the default target.
+all: all.indirect
+
+# This tells GNU Make version 3 not to put all variables in the environment.
+.NOEXPORT:
+
+# sed inserts variable overrides after the following line.
+####target overrides
+@target_overrides@
+####host overrides
+@host_overrides@
+####cross overrides
+@cross_overrides@
+####build overrides
+@build_overrides@
+####site overrides
+\f
+# Now figure out from those variables how to compile and link.
+
+all.indirect: Makefile ../jc1$(exeext) ../jcf-dump$(exeext) \
+       ../jvgenmain$(exeext) ../gcjh$(exeext) ../jv-scan$(exeext)
+
+# IN_GCC tells obstack.h that we are using gcc's <stddef.h> file.
+INTERNAL_CFLAGS = $(CROSS) -DIN_GCC @extra_c_flags@
+
+# This is the variable actually used when we compile.
+ALL_CFLAGS = $(INTERNAL_CFLAGS) $(X_CFLAGS) $(T_CFLAGS) $(CFLAGS) $(XCFLAGS)
+
+# Likewise.
+ALL_CPPFLAGS = $(CPPFLAGS) $(X_CPPFLAGS) $(T_CPPFLAGS)
+
+# Even if ALLOCA is set, don't use it if compiling with GCC.
+
+SUBDIR_OBSTACK = `if [ x$(OBSTACK) != x ]; then echo ../$(OBSTACK); else true; fi`
+SUBDIR_USE_ALLOCA = `case "${CC}" in "${OLDCC}") if [ x$(ALLOCA) != x ]; then echo ../$(ALLOCA); else true; fi ;; esac`
+SUBDIR_MALLOC = `if [ x$(MALLOC) != x ]; then echo ../$(MALLOC); else true; fi`
+
+# How to link with both our special library facilities
+# and the system's installed libraries.
+LIBS = $(SUBDIR_OBSTACK) $(SUBDIR_USE_ALLOCA) $(SUBDIR_MALLOC) $(CLIB)
+
+# Specify the directories to be searched for header files.
+# Both . and srcdir are used, in that order,
+# so that tm.h and config.h will be found in the compilation
+# subdirectory rather than in the source directory.
+INCLUDES = -I. -I.. -I$(srcdir) -I$(srcdir)/.. -I$(srcdir)/../config
+
+# Always use -I$(srcdir)/config when compiling.
+.c.o:
+       $(CC) -c $(ALL_CFLAGS) $(ALL_CPPFLAGS) $(INCLUDES) $<
+
+# This tells GNU make version 3 not to export all the variables
+# defined in this file into the environment.
+.NOEXPORT:
+\f
+# Lists of files for various purposes.
+
+# Language-specific object files for Gcc/Java:
+
+# Remember to keep this list in sync with JAVA_SRCS in Make-lang.in!!!
+#
+JAVA_OBJS = parse.o class.o decl.o expr.o constants.o lang.o typeck.o \
+  except.o verify.o zextract.o jcf-io.o jcf-parse.o mangle.o jcf-write.o \
+  buffer.o
+
+JAVA_OBJS_LITE = parse-scan.o jv-scan.o
+
+# Language-independent object files.
+OBJS = `cat ../stamp-objlist`
+OBJDEPS = ../stamp-objlist
+
+compiler: ../jc1$(exeext) ../jv-scan$(exeext)
+../jc1$(exeext): $(P) $(JAVA_OBJS) $(OBJDEPS) $(LIBDEPS)
+       rm -f ../jc1$(exeext)
+       $(CC) $(ALL_CFLAGS) $(LDFLAGS) -o $@ \
+             $(JAVA_OBJS) $(OBJS) $(LIBS)
+../jv-scan$(exeext): $(P) $(JAVA_OBJS_LITE) $(OBJDEPS) $(LIBDEPS)
+       rm -f ../jv-scan$(exeext)
+       $(CC) $(ALL_CFLAGS) $(LDFLAGS) -o $@ \
+             $(JAVA_OBJS_LITE) $(LIBS)
+
+../jcf-dump$(exeext): jcf-dump.o jcf-io.o zextract.o
+       $(CC) $(ALL_CFLAGS) $(LDFLAGS) -o $@ jcf-dump.o jcf-io.o zextract.o
+
+# Dependencies here must be kept in sync with dependencies in Make-lang.in.
+../jvgenmain$(exeext):  jvgenmain.o mangle.o
+       $(CC) $(ALL_CFLAGS) $(LDFLAGS) -o $@ jvgenmain.o mangle.o ../obstack.o
+
+../gcjh$(exeext): gjavah.o jcf-io.o zextract.o
+       $(CC) $(ALL_CFLAGS) $(LDFLAGS) -o $@ gjavah.o jcf-io.o zextract.o
+
+Makefile: $(srcdir)/Makefile.in $(srcdir)/../configure
+       cd ..; $(SHELL) config.status
+
+native: config.status ../jc1$(exeext)
+\f
+# Compiling object files from source files.
+
+PARSE_H = $(srcdir)/parse.h
+PARSE_C = $(srcdir)/parse.c
+PARSE_SCAN_C = $(srcdir)/parse-scan.c
+
+$(PARSE_H): $(PARSE_C)
+$(PARSE_C):  $(srcdir)/parse.y $(srcdir)/lex.c $(PARSE_H) $(srcdir)/lex.h
+       $(BISON) -t -v $(BISONFLAGS) $(JAVABISONFLAGS) -o $(PARSE_C) \
+         $(srcdir)/parse.y
+$(PARSE_SCAN_C):  $(srcdir)/parse-scan.y $(srcdir)/lex.c $(PARSE_H) \
+               $(srcdir)/lex.h
+       $(BISON) -t -v $(BISONFLAGS) -o $(PARSE_SCAN_C) $(srcdir)/parse-scan.y
+
+lex.c: keyword.h lex.h
+
+keyword.h: keyword.gperf
+       gperf -p -t -j1 -i 1 -g -o -N java_keyword -k1,3,$$ \
+       keyword.gperf > keyword.h
+
+# These exist for maintenance purposes.
+
+# Update the tags table.
+TAGS: force
+       cd $(srcdir) ;                          \
+       etags *.c *.h ;                         \
+       echo 'l' | tr 'l' '\f' >> TAGS ;        \
+       echo 'parse.y,0' >> TAGS ;              \
+       etags -a ../*.h ../*.c;
+
+.PHONY: TAGS
+
+mostlyclean:
+       rm -f *.o
+
+clean:  mostlyclean
+       rm -f parse.c
+
+force:
+
+parse.o: $(PARSE_C) jcf-reader.c
+jcf-dump.o: jcf-reader.c jcf.h javaop.h javaop.def
+gjavah.o: jcf-reader.c jcf.h javaop.h
diff --git a/gcc/java/buffer.c b/gcc/java/buffer.c
new file mode 100644 (file)
index 0000000..cea7899
--- /dev/null
@@ -0,0 +1,50 @@
+/* A "buffer" utility type.
+   Copyright (C) 1998 Free Software Foundation, Inc.
+
+This file is part of GNU CC.
+
+GNU CC is free software; you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation; either version 2, or (at your option)
+any later version.
+
+GNU CC is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with GNU CC; see the file COPYING.  If not, write to
+the Free Software Foundation, 59 Temple Place - Suite 330,
+Boston, MA 02111-1307, USA.  */
+
+/* Written by Per Bothner <bothner@cygnus.com>, July 1998. */
+
+#include "gansidecl.h"
+#include "buffer.h"
+
+/* Grow BUFP so there is room for at least SIZE more bytes. */
+
+void
+buffer_grow (bufp, size)
+     struct buffer *bufp;
+     int size;
+{
+  if (bufp->limit - bufp->ptr >= size)
+    return;
+  if (bufp->data == 0)
+    {
+      if (size < 120)
+       size = 120;
+      bufp->data = (unsigned char*) xmalloc (size);
+      bufp->ptr = bufp->data;
+    }
+  else
+    {
+      int index = bufp->ptr - bufp->data;
+      size += 2 * (bufp->limit - bufp->data);
+      bufp->data = (unsigned char *) xrealloc (bufp->data, size);
+      bufp->ptr = bufp->data + index;
+    }
+  bufp->limit = bufp->data + size;
+}
diff --git a/gcc/java/buffer.h b/gcc/java/buffer.h
new file mode 100644 (file)
index 0000000..aa63840
--- /dev/null
@@ -0,0 +1,43 @@
+/* A "buffer" utility type.
+   Copyright (C) 1998 Free Software Foundation, Inc.
+
+This file is part of GNU CC.
+
+GNU CC is free software; you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation; either version 2, or (at your option)
+any later version.
+
+GNU CC is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with GNU CC; see the file COPYING.  If not, write to
+the Free Software Foundation, 59 Temple Place - Suite 330,
+Boston, MA 02111-1307, USA.  */
+
+/* Written by Per Bothner <bothner@cygnus.com>, July 1998. */
+
+/* A simple data structure for an expandable buffer. */
+
+struct buffer
+{
+  /* The start of the actual data buffer. */
+  unsigned char *data;
+
+  /* Where to write next in the buffer. */
+  unsigned char *ptr;
+
+  /* The end of the allocated data buffer. */
+  unsigned char *limit;
+};
+
+#define NULL_BUFFER { (void*) 0, (void*) 0, (void*) 0 }
+
+#define BUFFER_LENGTH(BUFP) ((BUFP)->ptr - (BUFP)->data)
+
+#define BUFFER_RESET(BUFP) ((BUFP)->ptr = (BUFP)->data)
+
+extern void buffer_grow PROTO ((struct buffer*, int));
diff --git a/gcc/java/class.c b/gcc/java/class.c
new file mode 100644 (file)
index 0000000..9dc49dc
--- /dev/null
@@ -0,0 +1,1601 @@
+/* Functions related to building classes and their related objects.
+   Copyright (C) 1996, 1997, 1998 Free Software Foundation, Inc.
+
+This file is part of GNU CC.
+
+GNU CC is free software; you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation; either version 2, or (at your option)
+any later version.
+
+GNU CC is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with GNU CC; see the file COPYING.  If not, write to
+the Free Software Foundation, 59 Temple Place - Suite 330,
+Boston, MA 02111-1307, USA.
+
+Java and all Java-based marks are trademarks or registered trademarks
+of Sun Microsystems, Inc. in the United States and other countries.
+The Free Software Foundation is independent of Sun Microsystems, Inc.  */
+
+/* Written by Per Bothner <bothner@cygnus.com> */
+
+#include <stdio.h>
+#include "config.h"
+#include "tree.h"
+#include "rtl.h"
+#include "java-tree.h"
+#include "jcf.h"
+#include "obstack.h"
+
+static tree mangle_class_field PROTO ((tree class));
+
+static rtx registerClass_libfunc;
+
+extern struct obstack permanent_obstack;
+extern struct obstack temporary_obstack;
+
+/* Return an IDENTIFIER_NODE the same as (OLD_NAME, OLD_LENGTH).
+   except that characters matching OLD_CHAR are substituted by NEW_CHAR.
+   Also, PREFIX is prepended, and SUFFIX is appended. */
+
+tree
+ident_subst (old_name, old_length, prefix, old_char, new_char, suffix)
+     const char* old_name;
+     int old_length;
+     const char *prefix;
+     int old_char;
+     int new_char;
+     const char *suffix;
+{
+  int prefix_len = strlen (prefix);
+  int suffix_len = strlen (suffix);
+  int i = prefix_len + old_length + suffix_len + 1;
+#ifdef __GNUC__
+  char buffer[i];
+#else
+  char *buffer = (char *)alloca  (i);
+#endif
+  strcpy (buffer, prefix);
+  for (i = 0; i < old_length; i++)
+    {
+      char ch = old_name[i];
+      if (ch == old_char)
+       ch = new_char;
+      buffer[prefix_len + i] = ch;
+    }
+  strcpy (buffer + prefix_len + old_length, suffix);
+  return get_identifier (buffer);
+}
+
+/* Return an IDENTIFIER_NODE the same as OLD_ID,
+   except that characters matching OLD_CHAR are substituted by NEW_CHAR.
+   Also, PREFIX is prepended, and SUFFIX is appended. */
+
+tree
+identifier_subst (old_id, prefix, old_char, new_char, suffix)
+     const tree old_id;
+     const char *prefix;
+     int old_char;
+     int new_char;
+     const char *suffix;
+{
+  return ident_subst (IDENTIFIER_POINTER (old_id), IDENTIFIER_LENGTH (old_id),
+                     prefix, old_char, new_char, suffix);
+}
+
+/* Generate a valid C identifier from the name of the class TYPE,
+   prefixed by PREFIX. */
+
+tree
+mangled_classname (prefix, type)
+     char *prefix;
+     tree type;
+{
+  tree ident = TYPE_NAME (type);
+  if (TREE_CODE (ident) != IDENTIFIER_NODE)
+    ident = DECL_NAME (ident);
+  return identifier_subst (ident, prefix, '/', '_', "");
+}
+
+tree
+make_class ()
+{
+  tree type;
+  push_obstacks (&permanent_obstack, &permanent_obstack);
+  type = make_node (RECORD_TYPE);
+#ifdef JAVA_USE_HANDLES
+  tree field1 = build_decl (FIELD_DECL, get_identifier ("obj"),
+                           build_pointer_type (type));
+  tree field2 = build_decl (FIELD_DECL, get_identifier ("methods"),
+                           methodtable_ptr_type);
+  tree handle_type = make_node (RECORD_TYPE);
+  TREE_CHAIN (field1) = field2;
+  TYPE_FIELDS (handle_type) = field1;
+  TYPE_BINFO (type) = make_tree_vec (7);
+  TYPE_BINFO (handle_type) = make_tree_vec (7);
+  BINFO_HANDLE (TYPE_BINFO (handle_type)) = type;
+  BINFO_HANDLE (TYPE_BINFO (type)) = handle_type;
+#else
+  TYPE_BINFO (type) = make_tree_vec (6);
+#endif
+  CLASS_P (type) = 1;
+  pop_obstacks ();
+
+  return type;
+}
+
+/* Given a fully-qualified classname in NAME (whose length is NAME_LENGTH),
+   and where each of the constituents is separated by '/',
+   return a corresponding IDENTIFIER_NODE, except using '.' as separator. */
+
+tree
+unmangle_classname (name, name_length)
+     const char *name;  int name_length;
+{
+  return ident_subst (name, name_length, "", '/', '.', "");
+}
+
+tree
+push_class (class_type, class_name)
+     tree class_type, class_name;
+{
+  tree decl, signature;
+  char *save_input_filename = input_filename;
+  int save_lineno = lineno;
+  tree source_name = identifier_subst (class_name, "", '.', '/', ".java");
+  push_obstacks (&permanent_obstack, &permanent_obstack);
+  input_filename = IDENTIFIER_POINTER (source_name);
+  lineno = 0;
+  decl = build_decl (TYPE_DECL, class_name, class_type);
+  input_filename = save_input_filename;
+  lineno = save_lineno;
+  signature = identifier_subst (class_name, "L", '.', '/', ";");
+  IDENTIFIER_SIGNATURE_TYPE (signature) = class_type;
+
+  /* Setting DECL_ARTIFICAL forces dbxout.c to specific the type is
+     both a typedef and in the struct name-space.  We may want to re-visit
+     this later, but for now it reduces the changes needed for gdb. */
+  DECL_ARTIFICIAL (decl) = 1;
+
+  pushdecl_top_level (decl);
+#ifdef JAVA_USE_HANDLES
+  {
+    tree handle_name = identifier_subst (class_name,
+                                        "Handle$", '.', '.', "");
+    tree handle_decl = build_decl (TYPE_DECL, handle_name,
+                                  CLASS_TO_HANDLE_TYPE (class_type));
+    pushdecl (handle_decl);
+  }
+#endif
+
+  pop_obstacks ();
+  return decl;
+}
+
+/* Finds the (global) class named NAME.  Creates the class if not found.
+   Also creates associated TYPE_DECL.
+   Does not check if the class actually exists, load the class,
+   fill in field or methods, or do layout_type. */
+
+tree
+lookup_class (name)
+     tree name;
+{
+  tree decl = IDENTIFIER_CLASS_VALUE (name);
+  if (decl == NULL_TREE)
+    decl = push_class (make_class (), name);
+  return TREE_TYPE (decl);
+}
+
+void
+set_super_info (access_flags, this_class, super_class, interfaces_count)
+     int access_flags;
+     tree this_class;
+     tree super_class;
+     int interfaces_count;
+{
+  int total_supers = interfaces_count;
+  tree class_decl = TYPE_NAME (this_class);
+  if (super_class)
+    total_supers++;
+
+  push_obstacks (&permanent_obstack, &permanent_obstack);
+  TYPE_BINFO_BASETYPES (this_class) = make_tree_vec (total_supers);
+  if (super_class)
+    {
+      tree super_binfo = make_tree_vec (6);
+      BINFO_TYPE (super_binfo) = super_class;
+      BINFO_OFFSET (super_binfo) = integer_zero_node;
+      TREE_VIA_PUBLIC (super_binfo) = 1;
+      TREE_VEC_ELT (BINFO_BASETYPES (TYPE_BINFO (this_class)), 0)
+       = super_binfo;
+      CLASS_HAS_SUPER (this_class) = 1;
+    }
+  pop_obstacks ();
+  
+  if (access_flags & ACC_PUBLIC)    CLASS_PUBLIC (class_decl) = 1;
+  if (access_flags & ACC_FINAL)     CLASS_FINAL (class_decl) = 1;
+  if (access_flags & ACC_SUPER)     CLASS_SUPER (class_decl) = 1;
+  if (access_flags & ACC_INTERFACE) CLASS_INTERFACE (class_decl) = 1;
+  if (access_flags & ACC_ABSTRACT)  CLASS_ABSTRACT (class_decl) = 1;
+}
+
+/* Return length of inheritance chain of CLAS, where java.lang.Object is 0,
+   direct sub-classes of Object are 1, and so on. */
+
+int
+class_depth (clas)
+     tree clas;
+{
+  int depth = 0;
+  if (! CLASS_LOADED_P (clas))
+    load_class (clas, 1);
+  while (clas != object_type_node)
+    {
+      depth++;
+      clas = TYPE_BINFO_BASETYPE (clas, 0);
+    }
+  return depth;
+}
+
+/* Return true iff TYPE2 is an interface that extends interface TYPE1 */
+
+int
+interface_of_p (type1, type2)
+     tree type1, type2;
+{
+  int n, i;
+  tree basetype_vec;
+
+  if (!(basetype_vec = TYPE_BINFO_BASETYPES (type2)))
+    return 0;
+  n = TREE_VEC_LENGTH (basetype_vec);
+  for (i = 0; i < n; i++)
+    {
+      tree vec_elt = TREE_VEC_ELT (basetype_vec, i);
+      if (vec_elt && BINFO_TYPE (vec_elt) == type1)
+       return 1;
+    }
+  for (i = 0; i < n; i++)
+    {
+      tree vec_elt = TREE_VEC_ELT (basetype_vec, i);
+      if (vec_elt && BINFO_TYPE (vec_elt) 
+         && interface_of_p (type1, BINFO_TYPE (vec_elt)))
+       return 1;
+    }
+  return 0;
+}
+
+/* Return true iff TYPE1 inherits from TYPE2. */
+
+int
+inherits_from_p (type1, type2)
+     tree type1, type2;
+{
+  while (type1 != NULL_TREE && TREE_CODE (type1) == RECORD_TYPE)
+    {
+      if (type1 == type2)
+       return 1;
+      type1 = CLASSTYPE_SUPER (type1);
+    }
+  return 0;
+}
+
+static void
+add_interface_do (basetype_vec, interface_class, i)
+     tree basetype_vec, interface_class;
+     int i;
+{
+  tree interface_binfo = make_tree_vec (6);
+  BINFO_TYPE (interface_binfo) = interface_class;
+  BINFO_OFFSET (interface_binfo) = integer_zero_node;
+  TREE_VIA_VIRTUAL (interface_binfo) = 1;
+  TREE_VIA_PUBLIC (interface_binfo) = 1;
+  TREE_VEC_ELT (basetype_vec, i) = interface_binfo;
+}
+
+/* Add INTERFACE_CLASS to THIS_CLASS iff INTERFACE_CLASS can't be
+   found in THIS_CLASS. Returns NULL_TREE upon success, INTERFACE_CLASS
+   if attempt is made to add it twice. */
+
+tree
+maybe_add_interface (this_class, interface_class)
+     tree this_class, interface_class;
+{
+  tree basetype_vec = TYPE_BINFO_BASETYPES (this_class);
+  tree interface_binfo = make_tree_vec (6);
+  int i;
+  int n = TREE_VEC_LENGTH (basetype_vec);
+  for (i = 0; ; i++)
+    {
+      if (i >= n)
+       {
+         error ("internal error - too many interface type");
+         return NULL_TREE;
+       }
+      else if (TREE_VEC_ELT (basetype_vec, i) == NULL_TREE)
+       break;
+      else if (BINFO_TYPE (TREE_VEC_ELT (basetype_vec, i)) == interface_class)
+       return interface_class;
+    } 
+  add_interface_do (basetype_vec, interface_class, i);
+  return NULL_TREE;
+}
+
+/* Add the INTERFACE_CLASS as one of the interfaces of THIS_CLASS. */
+
+void
+add_interface (this_class, interface_class)
+     tree this_class, interface_class;
+{
+  tree basetype_vec = TYPE_BINFO_BASETYPES (this_class);
+  int i;
+  int n = TREE_VEC_LENGTH (basetype_vec);
+  for (i = 0; ; i++)
+    {
+      if (i >= n)
+       {
+         error ("internal error - too many interface type");
+         return;
+       }
+      else if (TREE_VEC_ELT (basetype_vec, i) == NULL_TREE)
+       break;
+    }
+  add_interface_do (basetype_vec, interface_class, i);
+}
+
+/* Return the address of a pointer to the first FUNCTION_DECL
+   in the list (*LIST) whose DECL_NAME is NAME. */
+
+static tree *
+find_named_method (list, name)
+     tree *list;
+     tree name;
+{
+  while (*list && DECL_NAME (*list) != name)
+    list = &TREE_CHAIN (*list);
+  return list;
+}
+
+tree
+build_java_method_type (fntype, this_class, access_flags)
+     tree fntype;
+     tree this_class;
+     int access_flags;
+{
+  if (access_flags & ACC_STATIC)
+    return fntype;
+  return build_method_type (CLASS_TO_HANDLE_TYPE (this_class), fntype);
+}
+
+tree
+add_method_1 (handle_class, access_flags, name, function_type)
+     tree handle_class;
+     int access_flags;
+     tree name;
+     tree function_type;
+{
+  tree method_type, fndecl;
+  push_obstacks (&permanent_obstack, &permanent_obstack);
+
+  method_type = build_java_method_type (function_type,
+                                       handle_class, access_flags);
+
+  fndecl = build_decl (FUNCTION_DECL, name, method_type);
+  DECL_CONTEXT (fndecl) = handle_class;
+
+  DECL_LANG_SPECIFIC (fndecl)
+    = (struct lang_decl *) permalloc (sizeof (struct lang_decl));
+  bzero (DECL_LANG_SPECIFIC (fndecl), sizeof (struct lang_decl));
+
+  TREE_CHAIN (fndecl) = TYPE_METHODS (handle_class);
+  TYPE_METHODS (handle_class) = fndecl;
+  pop_obstacks ();
+
+  if (access_flags & ACC_PUBLIC) METHOD_PUBLIC (fndecl) = 1;
+  if (access_flags & ACC_PROTECTED) METHOD_PROTECTED (fndecl) = 1;
+  if (access_flags & ACC_PRIVATE) METHOD_PRIVATE (fndecl) = 1;
+  if (access_flags & ACC_NATIVE) METHOD_NATIVE (fndecl) = 1;
+  if (access_flags & ACC_STATIC) METHOD_STATIC (fndecl) = 1;
+  if (access_flags & ACC_FINAL) METHOD_FINAL (fndecl) = 1;
+  if (access_flags & ACC_SYNCHRONIZED) METHOD_SYNCHRONIZED (fndecl) = 1;
+  if (access_flags & ACC_ABSTRACT) METHOD_ABSTRACT (fndecl) = 1;
+  if (access_flags & ACC_TRANSIENT) METHOD_TRANSIENT (fndecl) = 1;
+  return fndecl;
+}
+
+/* Add a method to THIS_CLASS.
+   The method's name is NAME.
+   Its signature (mangled type) is METHOD_SIG (an IDENTIFIER_NODE). */
+
+tree
+add_method (this_class, access_flags, name, method_sig)
+     tree this_class;
+     int access_flags;
+     tree name;
+     tree method_sig;
+{
+  tree handle_class = CLASS_TO_HANDLE_TYPE (this_class);
+  tree function_type, method_type, fndecl;
+  unsigned char *sig = (unsigned char*)IDENTIFIER_POINTER (method_sig);
+  push_obstacks (&permanent_obstack, &permanent_obstack);
+  if (sig[0] != '(')
+    fatal ("bad method signature");
+  function_type = get_type_from_signature (method_sig);
+  fndecl = add_method_1 (handle_class, access_flags, name, function_type);
+  set_java_signature (TREE_TYPE (fndecl), method_sig);
+  pop_obstacks ();
+  return fndecl;
+}
+
+tree
+add_field (class, name, field_type, flags)
+     tree class;
+     tree name;
+     tree field_type;
+     int flags;
+{
+  int is_static = (flags & ACC_STATIC) != 0;
+  tree field;
+  /* Push the obstack of field_type ? FIXME */
+  push_obstacks (&permanent_obstack, &permanent_obstack);
+#if ! JAVA_PROMOTE_TO_INT
+  if (TREE_CODE (field_type) == RECORD_TYPE)
+#endif
+    field_type = promote_type (field_type);
+  field = build_decl (is_static ? VAR_DECL : FIELD_DECL, name, field_type);
+  pop_obstacks ();
+  TREE_CHAIN (field) = TYPE_FIELDS (class);
+  TYPE_FIELDS (class) = field;
+  DECL_CONTEXT (field) = class;
+
+  if (flags & ACC_PUBLIC) FIELD_PUBLIC (field) = 1;
+  if (flags & ACC_PROTECTED) FIELD_PROTECTED (field) = 1;
+  if (flags & ACC_PRIVATE) FIELD_PRIVATE (field) = 1;
+  if (flags & ACC_FINAL) FIELD_FINAL (field) = 1;
+  if (flags & ACC_VOLATILE) FIELD_VOLATILE (field) = 1;
+  if (flags & ACC_TRANSIENT) FIELD_TRANSIENT (field) = 1;
+  if (is_static)
+    {
+      FIELD_STATIC (field) = 1;
+      if (! FIELD_PRIVATE (field) || FIELD_PROTECTED (field))
+       TREE_PUBLIC (field) = 1;
+    }
+  return field;
+}
+
+/* Associate a constant value CONSTANT with VAR_DECL FIELD. */
+
+void
+set_constant_value (field, constant)
+     tree field, constant;
+{
+  if (field == NULL_TREE)
+    warning ("misplaced ConstantValue attribute (not in any field)");
+  else if (DECL_INITIAL (field) != NULL_TREE)
+    warning ("duplicate ConstanValue atribute for field '%s'",
+            IDENTIFIER_POINTER (DECL_NAME (field)));
+  else
+    DECL_INITIAL (field) = constant;
+}
+
+/* Count the number of Unicode chars encoded in a given Ut8 string. */
+
+int
+strLengthUtf8 (str, len)
+     char *str;
+     int len;
+{
+  register unsigned char* ptr = (unsigned char*) str;
+  register unsigned char *limit = ptr + len;
+  int str_length = 0;
+  for (; ptr < limit; str_length++) {
+    if (UTF8_GET (ptr, limit) < 0)
+      return -1;
+  }
+  return str_length;
+}
+
+
+/* Calculate a hash value for a string encoded in Utf8 format.
+ * This returns the same hash value as specified for java.lang.String.hashCode.
+ */
+
+int32
+hashUtf8String (str, len)
+     char *str;
+     int len;
+{
+  register unsigned char* ptr = (unsigned char*) str;
+  register unsigned char *limit = ptr + len;
+  int32 hash = 0;
+  for (; ptr < limit;)
+    {
+      int ch = UTF8_GET (ptr, limit);
+      /* Updated specification from
+        http://www.javasoft.com/docs/books/jls/clarify.html. */
+      hash = (31 * hash) + ch;
+    }
+  return hash;
+}
+
+tree utf8_decl_list = NULL_TREE;
+
+tree
+build_utf8_ref (name)
+     tree name;
+{
+  char* name_ptr = IDENTIFIER_POINTER(name);
+  int name_len = IDENTIFIER_LENGTH(name);
+  char buf[60];
+  char *buf_ptr;
+  tree ctype, field, str_type, cinit, string;
+  static int utf8_count = 0;
+  int name_hash;
+  tree ref = IDENTIFIER_UTF8_REF (name);
+  tree decl;
+  if (ref != NULL_TREE)
+    return ref;
+
+  push_obstacks (&permanent_obstack, &permanent_obstack);
+  ctype = make_node (RECORD_TYPE);
+  str_type = build_prim_array_type (unsigned_byte_type_node,
+                                   name_len + 1); /* Allow for final '\0'. */
+  PUSH_FIELD (ctype, field, "hash", unsigned_short_type_node);
+  PUSH_FIELD (ctype, field, "length", unsigned_short_type_node);
+  PUSH_FIELD (ctype, field, "data", str_type);
+  FINISH_RECORD (ctype);
+  START_RECORD_CONSTRUCTOR (cinit, ctype);
+  name_hash = hashUtf8String (name_ptr, name_len) & 0xFFFF;
+  PUSH_FIELD_VALUE (cinit, "hash", build_int_2 (name_hash, 0));
+  PUSH_FIELD_VALUE (cinit, "length", build_int_2 (name_len, 0));
+  string = build_string (name_len, name_ptr);
+  TREE_TYPE (string) = str_type;
+  PUSH_FIELD_VALUE (cinit, "data", string);
+  FINISH_RECORD_CONSTRUCTOR (cinit);
+
+  /* Build a unique identifier based on buf. */
+  sprintf(buf, "_Utf%d", ++utf8_count);
+  buf_ptr = &buf[strlen (buf)];
+  while (--name_len >= 0)
+    {
+      char c = *name_ptr++;
+      if (c & 0x80)
+       continue;
+      if (!isalpha(c) && !isdigit(c))
+       c = '_';
+      *buf_ptr++ = c;
+      if (buf_ptr >= buf + 50)
+       break;
+    }
+  *buf_ptr = '\0';
+
+  decl = build_decl (VAR_DECL, get_identifier (buf), utf8const_type);
+  /* FIXME get some way to force this into .text, not .data. */
+  TREE_STATIC (decl) = 1;
+  DECL_ARTIFICIAL (decl) = 1;
+  DECL_IGNORED_P (decl) = 1;
+  TREE_READONLY (decl) = 1;
+  DECL_INITIAL (decl) = cinit;
+  TREE_CHAIN (decl) = utf8_decl_list;
+  layout_decl (decl, 0);
+  pushdecl (decl);
+  rest_of_decl_compilation (decl, (char*) 0, global_bindings_p (), 0);
+  utf8_decl_list = decl;
+  make_decl_rtl (decl, (char*) 0, 1);
+  ref = build1 (ADDR_EXPR, utf8const_ptr_type, decl);
+  IDENTIFIER_UTF8_REF (name) = ref;
+  pop_obstacks ();
+  return ref;
+}
+
+/* Build a reference to the class TYPE.
+   Also handles primitive types and array types. */
+
+tree
+build_class_ref (type)
+     tree type;
+{
+  int is_compiled = is_compiled_class (type);
+  if (is_compiled)
+    {
+      tree ref, decl_name, decl;
+      if (TREE_CODE (type) == POINTER_TYPE)
+       type = TREE_TYPE (type);
+      if (TREE_CODE (type) == RECORD_TYPE)
+       {
+         if (TYPE_SIZE (type) == error_mark_node)
+           return null_pointer_node;
+         decl_name = identifier_subst (DECL_NAME (TYPE_NAME (type)),
+                                       "", '/', '/', ".class");
+         decl = IDENTIFIER_GLOBAL_VALUE (decl_name);
+         if (decl == NULL_TREE)
+           {
+             push_obstacks (&permanent_obstack, &permanent_obstack);
+             decl = build_decl (VAR_DECL, decl_name, class_type_node);
+             DECL_SIZE (decl) = TYPE_SIZE (class_type_node);
+             TREE_STATIC (decl) = 1;
+             TREE_PUBLIC (decl) = 1;
+             DECL_IGNORED_P (decl) = 1;
+             DECL_ARTIFICIAL (decl) = 1;
+             DECL_ASSEMBLER_NAME (decl) = mangle_class_field (type);
+             make_decl_rtl (decl, NULL, 1);
+             pushdecl_top_level (decl);
+             if (is_compiled == 1)
+               DECL_EXTERNAL (decl) = 1;
+             pop_obstacks ();
+           }
+       }
+      else
+       {
+         char *name;
+         char buffer[20];
+         decl_name = TYPE_NAME (type);
+         if (TREE_CODE (decl_name) == TYPE_DECL)
+           decl_name = DECL_NAME (decl_name);
+         name = IDENTIFIER_POINTER (decl_name);
+         if (strncmp (name, "promoted_", 9) == 0)
+           name += 9;
+         sprintf (buffer, "%sClass", name);
+         decl_name = get_identifier (buffer);
+         decl = IDENTIFIER_GLOBAL_VALUE (decl_name);
+         if (decl == NULL_TREE)
+           {
+             push_obstacks (&permanent_obstack, &permanent_obstack);
+             decl = build_decl (VAR_DECL, decl_name, class_type_node);
+             TREE_STATIC (decl) = 1;
+             TREE_PUBLIC (decl) = 1;
+             make_decl_rtl (decl, NULL, 1);
+             pushdecl_top_level (decl);
+             if (is_compiled == 1)
+               DECL_EXTERNAL (decl) = 1;
+             pop_obstacks ();
+           }
+       }
+
+      ref = build1 (ADDR_EXPR, class_ptr_type, decl);
+      return ref;
+    }
+  else
+    {
+      int index;
+      tree cl;
+      push_obstacks (&permanent_obstack, &permanent_obstack);
+      index = alloc_class_constant (type);
+      cl = build_ref_from_constant_pool (index); 
+      TREE_TYPE (cl) = promote_type (class_ptr_type);
+      pop_obstacks ();
+      return cl;
+    }
+}
+
+tree
+build_static_field_ref (fdecl)
+     tree fdecl;
+{
+  tree fclass = DECL_CONTEXT (fdecl);
+  int is_compiled = is_compiled_class (fclass);
+  if (is_compiled)
+    {
+      if (DECL_RTL (fdecl) == 0)
+       {
+         push_obstacks (&permanent_obstack, &permanent_obstack);
+         make_decl_rtl (fdecl, NULL, 1);
+         pop_obstacks ();
+         if (is_compiled == 1)
+           DECL_EXTERNAL (fdecl) = 1;
+       }
+      return fdecl;
+    }
+  else
+    {
+      /* Compile as:
+       * *(FTYPE*)build_class_ref(FCLASS)->fields[INDEX].info.addr
+       */
+      static tree fields_ident = NULL_TREE;
+      static tree info_ident = NULL_TREE;
+      tree ref = build_class_ref (fclass);
+      tree fld;
+      int field_index = 0;
+      ref = build1 (INDIRECT_REF, class_type_node, ref);
+      if (fields_ident == NULL_TREE)
+       fields_ident = get_identifier ("fields");
+      if (info_ident == NULL_TREE)
+       info_ident = get_identifier ("info");
+      ref = build (COMPONENT_REF, field_ptr_type_node, ref,
+                  lookup_field (&class_type_node, fields_ident));
+
+      for (fld = TYPE_FIELDS (fclass); ; fld = TREE_CHAIN (fld))
+       {
+         if (fld == fdecl)
+           break;
+         if (fld == NULL_TREE)
+           fatal ("field '%s' not found in class",
+                  IDENTIFIER_POINTER (DECL_NAME (fdecl)));
+         if (FIELD_STATIC (fld))
+           field_index++;
+       }
+      field_index *= int_size_in_bytes (field_type_node);
+      ref = fold (build (PLUS_EXPR, field_ptr_type_node,
+                        ref, build_int_2 (field_index, 0)));
+      ref = build1 (INDIRECT_REF, field_type_node, ref);
+      ref = build (COMPONENT_REF, field_info_union_node,
+                  ref, lookup_field (&field_type_node, info_ident));
+      ref = build (COMPONENT_REF, ptr_type_node,
+                  ref, TREE_CHAIN (TYPE_FIELDS (field_info_union_node)));
+      return fold (build1 (INDIRECT_REF, TREE_TYPE(fdecl), ref));
+    }
+}
+
+int
+get_access_flags_from_decl (decl)
+     tree decl;
+{
+  int access_flags = 0;
+  if (TREE_CODE (decl) == FIELD_DECL || TREE_CODE (decl) == VAR_DECL)
+    {
+      if (FIELD_STATIC (decl))
+       access_flags |= ACC_STATIC;
+      if (FIELD_PUBLIC (decl))
+       access_flags |= ACC_PUBLIC;
+      if (FIELD_PROTECTED (decl))
+       access_flags |= ACC_PROTECTED;
+      if (FIELD_PRIVATE (decl))
+       access_flags |= ACC_PRIVATE;
+      if (FIELD_FINAL (decl))
+       access_flags |= ACC_FINAL;
+      if (FIELD_VOLATILE (decl))
+       access_flags |= ACC_VOLATILE;
+      if (FIELD_TRANSIENT (decl))
+       access_flags |= ACC_TRANSIENT;
+      return access_flags;
+    }
+  if (TREE_CODE (decl) == TYPE_DECL)
+    {
+      if (CLASS_PUBLIC (decl))
+       access_flags |= ACC_PUBLIC;
+      if (CLASS_FINAL (decl))
+       access_flags |= ACC_FINAL;
+      if (CLASS_SUPER (decl))
+       access_flags |= ACC_SUPER;
+      if (CLASS_INTERFACE (decl))
+       access_flags |= ACC_INTERFACE;
+      if (CLASS_ABSTRACT (decl))
+       access_flags |= ACC_ABSTRACT;
+      return access_flags;
+    }
+  if (TREE_CODE (decl) == FUNCTION_DECL)
+    {
+      if (METHOD_PUBLIC (decl))
+       access_flags |= ACC_PUBLIC;
+      if (METHOD_PRIVATE (decl))
+       access_flags |= ACC_PRIVATE;
+      if (METHOD_PROTECTED (decl))
+       access_flags |= ACC_PROTECTED;
+      if (METHOD_STATIC (decl))
+       access_flags |= ACC_STATIC;
+      if (METHOD_FINAL (decl))
+       access_flags |= ACC_FINAL;
+      if (METHOD_SYNCHRONIZED (decl))
+       access_flags |= ACC_SYNCHRONIZED;
+      if (METHOD_NATIVE (decl))
+       access_flags |= ACC_NATIVE;
+      if (METHOD_ABSTRACT (decl))
+       access_flags |= ACC_ABSTRACT;
+      if (METHOD_TRANSIENT (decl))
+       access_flags |= ACC_TRANSIENT;
+      return access_flags;
+    }
+  abort ();
+}
+
+tree
+make_field_value (tree fdecl)
+{
+  tree finit, info;
+  int bsize, flags;
+  tree type = TREE_TYPE (fdecl);
+  int resolved = is_compiled_class (type);
+  START_RECORD_CONSTRUCTOR (finit, field_type_node);
+  PUSH_FIELD_VALUE (finit, "name", build_utf8_ref (DECL_NAME (fdecl)));
+  if (resolved)
+    type = build_class_ref (type);
+  else
+    type = build_utf8_ref (build_java_signature (type));
+  PUSH_FIELD_VALUE (finit, "type", type);
+  flags = get_access_flags_from_decl (fdecl);
+  if (! resolved)
+    flags |= 0x8000 /* FIELD_UNRESOLVED_FLAG */;
+  PUSH_FIELD_VALUE (finit, "accflags", build_int_2 (flags, 0));
+  bsize = TREE_INT_CST_LOW (TYPE_SIZE (TREE_TYPE (fdecl))) / BITS_PER_UNIT;
+  PUSH_FIELD_VALUE (finit, "bsize", build_int_2 (bsize, 0));
+  if (FIELD_STATIC (fdecl))
+    {
+      tree cfield = TREE_CHAIN (TYPE_FIELDS(field_info_union_node));
+      tree faddr = build_address_of (build_static_field_ref (fdecl));
+      info = build (CONSTRUCTOR, field_info_union_node, NULL_TREE,
+                   build_tree_list (cfield, faddr));
+    }
+  else
+    {
+      int boffset
+       = TREE_INT_CST_LOW (DECL_FIELD_BITPOS (fdecl)) / BITS_PER_UNIT;
+      info = build (CONSTRUCTOR, field_info_union_node, NULL_TREE,
+                   build_tree_list (TYPE_FIELDS(field_info_union_node),
+                                    build_int_2 (boffset, 0)));
+    }
+  PUSH_FIELD_VALUE (finit, "info", info);
+
+  FINISH_RECORD_CONSTRUCTOR (finit);
+  return finit;
+}
+
+tree
+make_method_value (mdecl, this_class_addr)
+     tree mdecl;
+     tree this_class_addr;
+{
+  tree minit;
+  tree code;
+#define ACC_TRANSLATED          0x4000
+  int accflags = get_access_flags_from_decl (mdecl) | ACC_TRANSLATED;
+  code = null_pointer_node;
+  if (DECL_RTL (mdecl))
+    code = build1 (ADDR_EXPR, nativecode_ptr_type_node, mdecl);
+  START_RECORD_CONSTRUCTOR (minit, method_type_node);
+  PUSH_FIELD_VALUE (minit, "name",
+                   build_utf8_ref (DECL_CONSTRUCTOR_P (mdecl) ?
+                                   init_identifier_node
+                                   : DECL_NAME (mdecl)));
+  PUSH_FIELD_VALUE (minit, "signature",
+                   build_utf8_ref (build_java_signature (TREE_TYPE (mdecl))));
+  PUSH_FIELD_VALUE (minit, "accflags", build_int_2 (accflags, 0));
+  PUSH_FIELD_VALUE (minit, "ncode", code);
+  FINISH_RECORD_CONSTRUCTOR (minit);
+  return minit;
+}
+
+tree
+get_dispatch_vector (type)
+     tree type;
+{
+  tree vtable = TYPE_VTABLE (type);
+  if (vtable == NULL)
+    {
+      int i;
+      tree method;
+      tree super = CLASSTYPE_SUPER (type);
+      int nvirtuals = TREE_INT_CST_LOW (TYPE_NVIRTUALS (type));
+      vtable = make_tree_vec (nvirtuals);
+      TYPE_VTABLE (type) = vtable;
+      if (super != NULL_TREE)
+       {
+         tree super_vtable = get_dispatch_vector (super);
+         for ( i = TREE_INT_CST_LOW (TYPE_NVIRTUALS (super));  --i >= 0; )
+           TREE_VEC_ELT (vtable, i) = TREE_VEC_ELT (super_vtable, i);
+       }
+      for (method = TYPE_METHODS (type);  method != NULL_TREE;
+          method = TREE_CHAIN (method))
+       {
+         if (DECL_VINDEX (method) != NULL_TREE
+             && TREE_CODE (DECL_VINDEX (method)) == INTEGER_CST)
+           {
+             TREE_VEC_ELT (vtable, TREE_INT_CST_LOW (DECL_VINDEX (method)))
+               = method;
+           }
+       }
+    }
+  return vtable;
+}
+
+tree
+get_dispatch_table (type, this_class_addr)
+     tree type, this_class_addr;
+{
+  tree vtable = get_dispatch_vector (type);
+  int i;
+  tree list = NULL_TREE;
+  int nvirtuals = TREE_VEC_LENGTH (vtable);
+  for (i = nvirtuals;  --i >= 0; )
+    {
+      tree method = TREE_VEC_ELT (vtable, i);
+      if (METHOD_ABSTRACT (method))
+       warning_with_decl (method, "abstract method in non-abstract class");
+      if (DECL_RTL (method) == 0)
+       make_decl_rtl (method, NULL, 1);
+      list = tree_cons (NULL_TREE /*DECL_VINDEX (method) + 2*/,
+                       build1 (ADDR_EXPR, nativecode_ptr_type_node, method),
+                       list);
+    }
+  /* Dummy entry for compatibility with G++ -fvtable-thunks. */
+  list = tree_cons (integer_zero_node, null_pointer_node, list);
+  list = tree_cons (integer_zero_node, this_class_addr, list);
+  return build (CONSTRUCTOR, build_prim_array_type (nativecode_ptr_type_node,
+                                                   nvirtuals + 2),
+                NULL_TREE, list);
+}
+
+void
+make_class_data (type)
+     tree type;
+{
+  tree decl, cons, temp;
+  tree field, fields_decl;
+  tree static_fields = NULL_TREE;
+  tree instance_fields = NULL_TREE;
+  HOST_WIDE_INT static_field_count = 0;
+  HOST_WIDE_INT instance_field_count = 0;
+  HOST_WIDE_INT field_count;
+  tree field_array_type;
+  tree method;
+  tree methods = NULL_TREE;
+  tree dtable_decl = NULL_TREE;
+  HOST_WIDE_INT method_count = 0;
+  tree method_array_type;
+  tree methods_decl;
+  tree super;
+  tree this_class_addr;
+  tree constant_pool_constructor;
+  tree interfaces = null_pointer_node;
+  int interface_len = 0;
+  tree type_decl = TYPE_NAME (type);
+
+  this_class_addr = build_class_ref (type);
+  decl = TREE_OPERAND (this_class_addr, 0);
+
+  /* Build Field array. */
+  field = TYPE_FIELDS (type);
+  if (DECL_NAME (field) == NULL_TREE)
+    field = TREE_CHAIN (field);  /* Skip dummy field for inherited data. */
+  for ( ;  field != NULL_TREE;  field = TREE_CHAIN (field))
+    {
+      if (! DECL_ARTIFICIAL (field))
+       {
+         tree init = make_field_value (field);
+         if (FIELD_STATIC (field))
+           {
+             static_field_count++;
+             static_fields = tree_cons (NULL_TREE, init, static_fields);
+             rest_of_decl_compilation (field, (char*) 0, 1, 1);
+           }
+         else
+           {
+             instance_field_count++;
+             instance_fields = tree_cons (NULL_TREE, init, instance_fields);
+           }
+       }
+    }
+  field_count = static_field_count + instance_field_count;
+  if (field_count > 0)
+    {
+      static_fields = nreverse (static_fields);
+      instance_fields = nreverse (instance_fields);
+      static_fields = chainon (static_fields, instance_fields);
+      field_array_type = build_prim_array_type (field_type_node, field_count);
+      fields_decl = build_decl (VAR_DECL, mangled_classname ("_FL_", type),
+                               field_array_type);
+      DECL_INITIAL (fields_decl) = build (CONSTRUCTOR, field_array_type,
+                                         NULL_TREE, static_fields);
+      TREE_STATIC (fields_decl) = 1;
+      DECL_ARTIFICIAL (fields_decl) = 1;
+      DECL_IGNORED_P (fields_decl) = 1;
+      rest_of_decl_compilation (fields_decl, (char*) 0, 1, 0);
+    }
+  else
+    fields_decl = NULL_TREE;
+
+  /* Build Method array. */
+  for (method = TYPE_METHODS (CLASS_TO_HANDLE_TYPE (type));
+       method != NULL_TREE; method = TREE_CHAIN (method))
+    {
+      tree init = make_method_value (method, this_class_addr);
+      method_count++;
+      methods = tree_cons (NULL_TREE, init, methods);
+    }
+  method_array_type = build_prim_array_type (method_type_node, method_count);
+  methods_decl = build_decl (VAR_DECL, mangled_classname ("_MT_", type),
+                            method_array_type);
+  DECL_INITIAL (methods_decl) = build (CONSTRUCTOR, method_array_type,
+                                      NULL_TREE, nreverse (methods));
+  TREE_STATIC (methods_decl) = 1;
+  DECL_ARTIFICIAL (methods_decl) = 1;
+  DECL_IGNORED_P (methods_decl) = 1;
+  rest_of_decl_compilation (methods_decl, (char*) 0, 1, 0);
+
+  if (flag_assume_compiled
+      && ! CLASS_ABSTRACT (type_decl) && ! CLASS_INTERFACE (type_decl))
+    {
+      tree dtable = get_dispatch_table (type, this_class_addr);
+      dtable_decl = build_dtable_decl (type);
+      DECL_INITIAL (dtable_decl) = dtable;
+      TREE_STATIC (dtable_decl) = 1;
+      DECL_ARTIFICIAL (dtable_decl) = 1;
+      DECL_IGNORED_P (dtable_decl) = 1;
+      TREE_PUBLIC (dtable_decl) = 1;
+      rest_of_decl_compilation (dtable_decl, (char*) 0, 1, 0);
+    }
+
+  super = CLASSTYPE_SUPER (type);
+  if (super == NULL_TREE)
+    super = null_pointer_node;
+  else if (flag_assume_compiled)
+    super = build_class_ref (super);
+  else
+    {
+      int super_index = alloc_class_constant (super);
+      super = build_int_2 (super_index, 0);
+      TREE_TYPE (super) == ptr_type_node;
+    }
+
+  /* Build and emit the array of implemented interfaces. */
+  if (type != object_type_node)
+      interface_len = TREE_VEC_LENGTH (TYPE_BINFO_BASETYPES (type)) - 1;
+  if (interface_len > 0)
+    {
+      tree init = NULL_TREE;
+      int i;
+      tree interface_array_type, idecl;
+      interface_array_type
+       = build_prim_array_type (class_ptr_type, interface_len);
+      idecl = build_decl (VAR_DECL, mangled_classname ("_IF_", type),
+                         interface_array_type);
+      for (i = interface_len;  i > 0; i--)
+       {
+         tree child = TREE_VEC_ELT (TYPE_BINFO_BASETYPES (type), i);
+         tree iclass = BINFO_TYPE (child);
+         tree index;
+         if (flag_assume_compiled)
+           index = build_class_ref (iclass);
+         else
+           {
+               int int_index = alloc_class_constant (iclass);
+               index = build_int_2 (int_index, 0);
+               TREE_TYPE (index) == ptr_type_node;
+           }
+         init = tree_cons (NULL_TREE, index, init); 
+       }
+      DECL_INITIAL (idecl) = build (CONSTRUCTOR, interface_array_type,
+                                   NULL_TREE, init);
+      TREE_STATIC (idecl) = 1;
+      DECL_ARTIFICIAL (idecl) = 1;
+      DECL_IGNORED_P (idecl) = 1;
+      interfaces = build1 (ADDR_EXPR, ptr_type_node, idecl);
+      rest_of_decl_compilation (idecl,  (char*) 0, 1, 0);
+    }
+
+  constant_pool_constructor = build_constants_constructor ();
+
+  START_RECORD_CONSTRUCTOR (temp, object_type_node);
+#if 0
+  PUSH_FIELD_VALUE (temp, "dtable", NULL_TREE);
+#else
+  PUSH_FIELD_VALUE (temp, "dtable",
+                   build1 (ADDR_EXPR, dtable_ptr_type, class_dtable_decl));
+#endif
+  PUSH_FIELD_VALUE (temp, "sync_info", null_pointer_node);
+  FINISH_RECORD_CONSTRUCTOR (temp);
+  START_RECORD_CONSTRUCTOR (cons, class_type_node);
+  PUSH_SUPER_VALUE (cons, temp);
+  PUSH_FIELD_VALUE (cons, "next", null_pointer_node);
+  PUSH_FIELD_VALUE (cons, "name",
+                   build_utf8_ref (build_internal_class_name (type)));
+  PUSH_FIELD_VALUE (cons, "accflags",
+                   build_int_2 (get_access_flags_from_decl (type_decl), 0));
+
+  PUSH_FIELD_VALUE (cons, "superclass", super);
+  PUSH_FIELD_VALUE (cons, "subclass_head", null_pointer_node);
+  PUSH_FIELD_VALUE (cons, "subclass_next", null_pointer_node);
+  PUSH_FIELD_VALUE (cons, "constants", constant_pool_constructor);
+  PUSH_FIELD_VALUE (cons, "methods",
+                   build1 (ADDR_EXPR, method_ptr_type_node, methods_decl));
+  PUSH_FIELD_VALUE (cons, "nmethods",  build_int_2 (method_count, 0));
+  PUSH_FIELD_VALUE (cons, "msize", TYPE_NVIRTUALS (type));
+  PUSH_FIELD_VALUE (cons, "fields",
+                   fields_decl == NULL_TREE ? null_pointer_node
+                   : build1 (ADDR_EXPR, field_ptr_type_node, fields_decl));
+  PUSH_FIELD_VALUE (cons, "bfsize", size_in_bytes (type));
+  PUSH_FIELD_VALUE (cons, "nfields", build_int_2 (field_count, 0));
+  PUSH_FIELD_VALUE (cons, "nsfields", build_int_2 (static_field_count, 0));
+  /* For now, we let Kaffe fill in the dtable.  */
+  PUSH_FIELD_VALUE (cons, "dtable",
+                   dtable_decl == NULL_TREE ? null_pointer_node
+                   : build1 (ADDR_EXPR, dtable_ptr_type, dtable_decl));
+  PUSH_FIELD_VALUE (cons, "interfaces", interfaces);
+  PUSH_FIELD_VALUE (cons, "loader", null_pointer_node);
+  PUSH_FIELD_VALUE (cons, "interface_len", build_int_2 (interface_len, 0));
+  PUSH_FIELD_VALUE (cons, "state",
+                   flag_assume_compiled ? integer_four_node
+                   : integer_two_node);
+
+  method = lookup_java_method (type,
+                              finalize_identifier_node, void_signature_node);
+  PUSH_FIELD_VALUE (cons, "final",
+                   method == NULL ? integer_zero_node : integer_one_node);
+
+  FINISH_RECORD_CONSTRUCTOR (cons);
+
+  DECL_INITIAL (decl) = cons;
+  rest_of_decl_compilation (decl, (char*) 0, 1, 0);
+}
+
+/* Return 2 if CLASS is compiled by this compilation job;
+   return 1 if CLASS can otherwise be assumed to be compiled;
+   return 0 if we cannot assume that CLASS is compiled.
+   Returns 1 for primitive and 0 for array types.  */
+int
+is_compiled_class (class)
+     tree class;
+{
+  if (TREE_CODE (class) == POINTER_TYPE)
+    class = TREE_TYPE (class);
+  if (TREE_CODE (class) != RECORD_TYPE)  /* Primitive types are static. */
+    return 1;
+  if (TYPE_ARRAY_P (class))
+    return 0;
+  if (class == current_class)
+    return 2;
+  if ((TYPE_LANG_SPECIFIC (class) &&  TYPE_LANG_SPECIFIC (class)->jcf && 
+       TYPE_LANG_SPECIFIC (class)->jcf->seen_in_zip))
+    {
+      /* The class was seen in the current ZIP file and will be
+        available as a compiled class in the future but may not have
+        been loaded already. Load it if necessary. This prevent
+        build_class_ref () from crashing.  This should take into
+        consideration class specified in a multiple class file
+        command line. FIXME if necessary.  */
+
+      if (!CLASS_LOADED_P (class))
+        load_class (class, 1);
+      return 2;
+    }
+
+  if (flag_assume_compiled)
+    {
+      if (!CLASS_LOADED_P (class))
+       load_class (class, 1);
+      return 1;
+    }
+
+  return 0;
+}
+
+/* Append the mangled name of TYPE onto OBSTACK. */
+
+void
+append_gpp_mangled_type (obstack, type)
+     struct obstack *obstack;
+     tree type;
+{
+  char buf[8];
+  int len;
+  char *ptr;
+  switch (TREE_CODE (type))
+    {
+      char code;
+    case BOOLEAN_TYPE: code = 'b';  goto primitive;
+    case CHAR_TYPE:    code = 'w';  goto primitive;
+    case VOID_TYPE:    code = 'v';  goto primitive;
+    case INTEGER_TYPE:
+      /* Get the original type instead of the arguments promoted type.
+        Avoid symbol name clashes. Should call a function to do that.
+        FIXME.  */
+      if (type == promoted_short_type_node)
+       type = short_type_node;
+      if (type == promoted_byte_type_node)
+        type = byte_type_node;
+      switch (TYPE_PRECISION (type))
+       {
+       case  8:       code = 'c';  goto primitive;
+       case 16:       code = 's';  goto primitive;
+       case 32:       code = 'i';  goto primitive;
+       case 64:       code = 'x';  goto primitive;
+       default:  goto bad_type;
+       }
+    primitive:
+      obstack_1grow (obstack, code);
+      break;
+    case REAL_TYPE:
+      switch (TYPE_PRECISION (type))
+       {
+       case 32:       code = 'f';  goto primitive;
+       case 64:       code = 'd';  goto primitive;
+       default:  goto bad_type;
+       }
+    case POINTER_TYPE:
+      type = TREE_TYPE (type);
+      obstack_1grow (obstack, 'P');
+    case RECORD_TYPE:
+      if (TYPE_ARRAY_P (type))
+       {
+         obstack_grow (obstack, "t6JArray1Z", sizeof("t6JArray1Z")-1);
+         append_gpp_mangled_type (obstack, TYPE_ARRAY_ELEMENT (type));
+       }
+      else
+       {
+         char *class_name = IDENTIFIER_POINTER (DECL_NAME (TYPE_NAME (type)));
+         append_gpp_mangled_classtype (obstack, class_name);
+       }
+      break;
+    bad_type:
+    default:
+      fatal ("internal error - trying to mangle unknown type");
+    }
+}
+
+/* Build the mangled name of the `class' field.  */
+
+static tree
+mangle_class_field (class)
+     tree class;
+{
+  tree name;
+  obstack_grow (&temporary_obstack, "_CL_", 4);
+  append_gpp_mangled_type (&temporary_obstack, class);
+  obstack_1grow (&temporary_obstack, '\0');
+  name = get_identifier (obstack_base (&temporary_obstack));
+  obstack_free (&temporary_obstack, obstack_base (&temporary_obstack));
+  return name;
+}
+
+/* Build the mangled (assembly-level) name of the static field FIELD. */
+
+tree
+mangle_static_field (field)
+     tree field;
+{
+  tree class = DECL_CONTEXT (field);
+  tree name = DECL_NAME (field);
+  int encoded_len;
+#if ! defined (NO_DOLLAR_IN_LABEL) || ! defined (NO_DOT_IN_LABEL)
+  obstack_1grow (&temporary_obstack, '_');
+#else
+  obstack_grow (&temporary_obstack, "__static_", 9);
+#endif
+  append_gpp_mangled_type (&temporary_obstack, class);
+  encoded_len = unicode_mangling_length (IDENTIFIER_POINTER (name),
+                                        IDENTIFIER_LENGTH (name));
+  if (encoded_len > 0)
+    {
+      obstack_1grow (&temporary_obstack, 'U');
+    }
+#ifndef NO_DOLLAR_IN_LABEL
+  obstack_1grow (&temporary_obstack, '$');
+#else /* NO_DOLLAR_IN_LABEL */
+#ifndef NO_DOT_IN_LABEL
+  obstack_1grow (&temporary_obstack, '.');
+#else /* NO_DOT_IN_LABEL */
+  obstack_1grow (&temporary_obstack, '_');
+#endif  /* NO_DOT_IN_LABEL */
+#endif  /* NO_DOLLAR_IN_LABEL */
+  if (encoded_len > 0)
+    {
+      emit_unicode_mangled_name (&temporary_obstack,
+                                IDENTIFIER_POINTER (name), 
+                                IDENTIFIER_LENGTH (name));
+    }
+  else
+    {
+      obstack_grow (&temporary_obstack,
+                   IDENTIFIER_POINTER (name),
+                   IDENTIFIER_LENGTH (name));
+    }
+  obstack_1grow (&temporary_obstack, '\0');
+  name = get_identifier (obstack_base (&temporary_obstack));
+  obstack_free (&temporary_obstack, obstack_base (&temporary_obstack));
+  return name;
+}
+
+/* Build a VAR_DECL for the dispatch table (vtable) for class TYPE. */
+
+tree
+build_dtable_decl (type)
+     tree type;
+{
+  tree name;
+  obstack_grow (&temporary_obstack, "__vt_", 5);
+  append_gpp_mangled_type (&temporary_obstack, type);
+  obstack_1grow (&temporary_obstack, '\0');
+  name = get_identifier (obstack_base (&temporary_obstack));
+  obstack_free (&temporary_obstack, obstack_base (&temporary_obstack));
+  return build_decl (VAR_DECL, name, dtable_type);
+}
+
+/* Pre-pend the TYPE_FIELDS of THIS_CLASS with a dummy FIELD_DECL for the
+   fields inherited from SUPER_CLASS. */
+
+void
+push_super_field (this_class, super_class)
+     tree this_class, super_class;
+{
+  tree base_decl;
+  push_obstacks (&permanent_obstack, &permanent_obstack);
+  base_decl = build_decl (FIELD_DECL, NULL_TREE, super_class);
+  pop_obstacks ();
+  DECL_IGNORED_P (base_decl) = 1;
+  TREE_CHAIN (base_decl) = TYPE_FIELDS (this_class);
+  TYPE_FIELDS (this_class) = base_decl;
+  DECL_SIZE (base_decl) = TYPE_SIZE (super_class);
+}
+
+void
+layout_class (this_class)
+     tree this_class;
+{
+  tree super_class = CLASSTYPE_SUPER (this_class);
+  tree handle_type = CLASS_TO_HANDLE_TYPE (this_class);
+  tree method_decl, field;
+  tree dtable_count;
+  int i;
+
+  if (super_class)
+    {
+      /* Class seen in source are now complete and can be layed out.
+        Once layed out, a class seen in the source has its
+        CLASS_LOADED_P flag set */
+      if (CLASS_FROM_SOURCE_P (super_class) && !CLASS_LOADED_P (super_class))
+       safe_layout_class (super_class);
+      if (! CLASS_LOADED_P (super_class))
+       load_class (super_class, 1);
+      if (TREE_CODE (TYPE_SIZE (super_class)) == ERROR_MARK)
+       {
+         TYPE_SIZE (this_class) = error_mark_node;
+         return;
+       }
+      dtable_count = TYPE_NVIRTUALS (super_class);
+
+      if (TYPE_SIZE (this_class) == NULL_TREE)
+       push_super_field (this_class, super_class);
+    }
+  else
+    {
+      dtable_count = integer_zero_node;
+    }
+
+  for (field = TYPE_FIELDS (this_class);
+       field != NULL_TREE;  field = TREE_CHAIN (field))
+    {
+      if (FIELD_STATIC (field))
+       {
+         /* Set DECL_ASSEMBLER_NAME to something suitably mangled. */
+         DECL_ASSEMBLER_NAME (field) = mangle_static_field (field);
+       }
+    }
+
+  layout_type (this_class);
+
+  TYPE_METHODS (handle_type) = nreverse (TYPE_METHODS (handle_type));
+
+  for (method_decl = TYPE_METHODS (handle_type), i = 0;
+       method_decl; method_decl = TREE_CHAIN (method_decl), i++)
+    {
+      char *ptr;
+      char buf[8];
+      char *asm_name;
+      tree method_name = DECL_NAME (method_decl);
+#if 1
+      /* Remove this once we no longer need old (Kaffe / JDK 1.0)  mangling. */
+      if (! flag_assume_compiled && METHOD_NATIVE (method_decl))
+       {
+         for (ptr = IDENTIFIER_POINTER (DECL_NAME (TYPE_NAME (this_class)));
+              *ptr; )
+           {
+             int ch = *ptr++;
+             if (ch == '.')
+               ch = '_';
+             obstack_1grow (&temporary_obstack, (char) ch);
+           }
+         obstack_1grow (&temporary_obstack, (char) '_');
+         if (method_name == init_identifier_node)
+           obstack_grow (&temporary_obstack, "INIT", 4);
+         else
+           obstack_grow (&temporary_obstack,
+                         IDENTIFIER_POINTER (method_name),
+                         IDENTIFIER_LENGTH (method_name));
+       }
+      else
+#endif
+       {
+         int len;  tree arg, arglist, t;
+         int method_name_needs_escapes = 0;
+         if (method_name != init_identifier_node)
+           {
+             int encoded_len
+               = unicode_mangling_length (IDENTIFIER_POINTER (method_name), 
+                                          IDENTIFIER_LENGTH (method_name));
+             if (encoded_len > 0)
+               {
+                 method_name_needs_escapes = 1;
+                 emit_unicode_mangled_name (&temporary_obstack,
+                                            IDENTIFIER_POINTER (method_name), 
+                                            IDENTIFIER_LENGTH (method_name));
+               }
+             else
+               {
+                 obstack_grow (&temporary_obstack,
+                               IDENTIFIER_POINTER (method_name),
+                               IDENTIFIER_LENGTH (method_name));
+               }
+           }
+
+         obstack_grow (&temporary_obstack, "__", 2);
+         append_gpp_mangled_type (&temporary_obstack, this_class);
+         TREE_PUBLIC (method_decl) = 1;
+
+         t = TREE_TYPE (method_decl);
+         arglist = TYPE_ARG_TYPES (t);
+         if (TREE_CODE (t) == METHOD_TYPE)
+           arglist = TREE_CHAIN (arglist);
+         for (arg = arglist; arg != NULL_TREE;  )
+           {
+             tree a = arglist;
+             tree argtype = TREE_VALUE (arg);
+             int tindex = 1;
+             if (TREE_CODE (argtype) == POINTER_TYPE)
+               {
+                 /* This is O(N**2).  Do we care?  Cfr gcc/cp/method.c. */
+                 while (a != arg && argtype != TREE_VALUE (a))
+                   a = TREE_CHAIN (a), tindex++;
+               }
+             else
+               a = arg;
+             if (a != arg)
+               {
+                 char buf[12];
+                 int nrepeats = 0;
+                 do
+                   {
+                     arg = TREE_CHAIN (arg); nrepeats++;
+                   }
+                 while (arg != NULL_TREE && argtype == TREE_VALUE (arg));
+                 if (nrepeats > 1)
+                   {
+                     obstack_1grow (&temporary_obstack, 'N');
+                     sprintf (buf, "%d", nrepeats);
+                     obstack_grow (&temporary_obstack, buf, strlen (buf));
+                     if (nrepeats > 9)
+                       obstack_1grow (&temporary_obstack, '_');
+                   }
+                 else
+                   obstack_1grow (&temporary_obstack, 'T');
+                 sprintf (buf, "%d", tindex);
+                 obstack_grow (&temporary_obstack, buf, strlen (buf));
+                 if (tindex > 9)
+                   obstack_1grow (&temporary_obstack, '_');
+               }
+             else
+               {
+                 append_gpp_mangled_type (&temporary_obstack, argtype);
+                 arg = TREE_CHAIN (arg);
+               }
+           }
+         if (method_name_needs_escapes)
+           obstack_1grow (&temporary_obstack, 'U');
+       }
+      obstack_1grow (&temporary_obstack, '\0');
+      asm_name = obstack_finish (&temporary_obstack);
+      DECL_ASSEMBLER_NAME (method_decl) = get_identifier (asm_name);
+      if (! METHOD_ABSTRACT (method_decl))
+       make_function_rtl (method_decl);
+      obstack_free (&temporary_obstack, asm_name);
+
+      if (method_name == init_identifier_node)
+       {
+         char *p = IDENTIFIER_POINTER (DECL_NAME (TYPE_NAME (this_class)));
+         for (ptr = p; *ptr; )
+           {
+             if (*ptr++ == '.')
+               p = ptr;
+           }
+         DECL_NAME (method_decl) = get_identifier (p);
+         DECL_CONSTRUCTOR_P (method_decl) = 1;
+       }
+      else if (! METHOD_STATIC (method_decl))
+       {
+         tree method_sig = build_java_argument_signature (TREE_TYPE (method_decl));
+         tree super_method = lookup_argument_method (super_class, method_name,
+                                                 method_sig);
+         if (super_method != NULL_TREE)
+           {
+             DECL_VINDEX (method_decl) = DECL_VINDEX (super_method);
+             if (DECL_VINDEX (method_decl) == NULL_TREE)
+               error_with_decl (method_decl,
+                        "non-static method '%s' overrides static method");
+#if 0
+             else if (TREE_TYPE (TREE_TYPE (method_decl))
+                      != TREE_TYPE (TREE_TYPE (super_method)))
+               {
+                 error_with_decl (method_decl,
+                                "Method `%s' redefined with different return type");  
+                 error_with_decl (super_method,
+                                "Overridden decl is here");
+               }
+#endif
+           }
+         else if (! METHOD_FINAL (method_decl)
+                  && ! CLASS_FINAL (TYPE_NAME (this_class)))
+           {
+             DECL_VINDEX (method_decl) = dtable_count;
+             dtable_count = build_int_2 (1+TREE_INT_CST_LOW (dtable_count), 0);
+           }
+       }
+    }
+  TYPE_NVIRTUALS (this_class) = dtable_count;
+
+#ifdef JAVA_USE_HANDLES
+  layout_type (handle_type);
+#endif
+}
+
+static tree registered_class = NULL_TREE;
+
+void
+register_class ()
+{
+  static tree end;
+  tree node    = TREE_OPERAND (build_class_ref (current_class), 0);
+  tree current = copy_node (node);
+
+  XEXP (DECL_RTL (current), 0) = copy_rtx (XEXP (DECL_RTL(node), 0));
+  if (!registered_class)
+    registered_class = current;
+  else
+    TREE_CHAIN (end) = current;
+
+  end = current;
+}
+
+/* Generate a function that gets called at start-up (static contructor) time,
+   which calls registerClass for all the compiled classes. */
+
+void
+emit_register_class ()
+{
+  tree decl = getdecls ();
+
+  extern tree get_file_function_name PROTO((int));
+  tree init_name = get_file_function_name ('I');
+  tree init_type = build_function_type (void_type_node, NULL_TREE);
+  tree init_decl;
+  tree t;
+
+  start_sequence ();
+  init_decl = build_decl (FUNCTION_DECL, init_name, init_type);
+  DECL_ASSEMBLER_NAME (init_decl) = init_name;
+  TREE_STATIC (init_decl) = 1;
+  current_function_decl = init_decl;
+  DECL_RESULT (init_decl) = build_decl(RESULT_DECL, NULL_TREE, void_type_node);
+  /*  DECL_EXTERNAL (init_decl) = 1;*/
+  TREE_PUBLIC (init_decl) = 1;
+  pushlevel (0);
+  make_function_rtl (init_decl);
+  init_function_start (init_decl, input_filename, 0);
+  expand_function_start (init_decl, 0);
+
+  for ( t = registered_class; t; t = TREE_CHAIN (t))
+    emit_library_call (registerClass_libfunc, 0, VOIDmode, 1,
+                      XEXP (DECL_RTL (t), 0), Pmode);
+
+  expand_function_end (input_filename, 0, 0);
+  poplevel (1, 0, 1);
+  { 
+    /* Force generation, even with -O3 or deeper. Gross hack. FIXME */
+    extern int flag_inline_functions;
+    int saved_flag = flag_inline_functions;
+    flag_inline_functions = 0; 
+    rest_of_compilation (init_decl);
+    flag_inline_functions = saved_flag;
+  }
+  current_function_decl = NULL_TREE;
+  assemble_constructor (IDENTIFIER_POINTER (init_name));
+}
+
+void
+init_class_processing ()
+{
+  registerClass_libfunc = gen_rtx (SYMBOL_REF, Pmode, "registerClass");
+}
diff --git a/gcc/java/config-lang.in b/gcc/java/config-lang.in
new file mode 100644 (file)
index 0000000..571e65a
--- /dev/null
@@ -0,0 +1,40 @@
+# Top level configure fragment for the GNU compiler for the Java(TM)
+# language.
+#  Copyright (C) 1994, 1995 Free Software Foundation, Inc.
+
+#This file is part of GNU CC.
+
+#GNU CC is free software; you can redistribute it and/or modify
+#it under the terms of the GNU General Public License as published by
+#the Free Software Foundation; either version 2, or (at your option)
+#any later version.
+
+#GNU CC is distributed in the hope that it will be useful,
+#but WITHOUT ANY WARRANTY; without even the implied warranty of
+#MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+#GNU General Public License for more details.
+
+#You should have received a copy of the GNU General Public License
+#along with GNU CC; see the file COPYING.  If not, write to
+#the Free Software Foundation, 59 Temple Place - Suite 330,
+#Boston, MA 02111-1307, USA.
+
+#Java and all Java-based marks are trademarks or registered trademarks
+#of Sun Microsystems, Inc. in the United States and other countries.
+#The Free Software Foundation is independent of Sun Microsystems, Inc.
+
+# Configure looks for the existence of this file to auto-config each language.
+# We define several parameters used by configure:
+#
+# language     - name of language as it would appear in $(LANGUAGES)
+# compilers    - value to add to $(COMPILERS)
+# stagestuff   - files to add to $(STAGESTUFF)
+# diff_excludes        - files to ignore when building diffs between two versions.
+
+language="java"
+
+compilers="jc1\$(exeext) jvgenmain\$(exeext)"
+
+stagestuff="jc1\$(exeext) gcj\$(exeext) jvgenmain\$(exeext) gcjh\$(exeext)"
+
+outputs=java/Makefile
diff --git a/gcc/java/constants.c b/gcc/java/constants.c
new file mode 100644 (file)
index 0000000..696f684
--- /dev/null
@@ -0,0 +1,453 @@
+/* Handle the constant pool of the Java(TM) Virtual Machine.
+   Copyright (C) 1997 Free Software Foundation, Inc.
+
+This file is part of GNU CC.
+
+GNU CC is free software; you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation; either version 2, or (at your option)
+any later version.
+
+GNU CC is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+You should have received a copy of the GNU General Public License
+along with GNU CC; see the file COPYING.  If not, write to
+the Free Software Foundation, 59 Temple Place - Suite 330,
+Boston, MA 02111-1307, USA. 
+
+Java and all Java-based marks are trademarks or registered trademarks
+of Sun Microsystems, Inc. in the United States and other countries.
+The Free Software Foundation is independent of Sun Microsystems, Inc.  */
+
+#include "config.h"
+#include "tree.h"
+#include "java-tree.h"
+#include "jcf.h"
+#include "system.h"
+
+extern struct obstack permanent_obstack;
+
+/* Set the INDEX'th constant in CPOOL to have the given TAG and VALUE. */
+
+void
+set_constant_entry (cpool, index, tag, value)
+     CPool *cpool;
+     int index;
+     int tag;
+     jword value;
+{
+  if (cpool->data == NULL)
+    {
+      cpool->capacity = 100;
+      cpool->tags = (uint8*) xmalloc (sizeof(uint8) * cpool->capacity);
+      cpool->data = (jword*) xmalloc (sizeof(jword) * cpool->capacity);
+      cpool->count = 1;
+    }
+  if (index >= cpool->capacity)
+    {
+      cpool->capacity *= 2;
+      if (index >= cpool->capacity)
+       cpool->capacity = index + 10;
+      cpool->tags = (uint8*) xrealloc (cpool->tags,
+                                      sizeof(uint8) * cpool->capacity);
+      cpool->data = (jword*) xrealloc (cpool->data,
+                                      sizeof(jword) * cpool->capacity);
+    }
+  if (index >= cpool->count)
+    cpool->count = index + 1;
+  cpool->tags[index] = tag;
+  cpool->data[index] = value;
+}
+
+/* Find (or create) a constant pool entry matching TAG and VALUE. */
+
+int
+find_constant1 (cpool, tag, value)
+     CPool *cpool;
+     int tag;
+     jword value;
+{
+  int i;
+  for (i = cpool->count;  --i > 0; )
+    {
+      if (cpool->tags[i] == tag && cpool->data[i] == value)
+       return i;
+    }
+  i = cpool->count == 0 ? 1 : cpool->count;
+  set_constant_entry (cpool, i, tag, value);
+  return i;
+}
+
+/* Find a double-word constant pool entry matching TAG and WORD1/WORD2. */
+
+int
+find_constant2 (cpool, tag, word1, word2)
+     CPool *cpool;
+     int tag;
+     jword word1, word2;
+{
+  int i;
+  for (i = cpool->count - 1;  --i > 0; )
+    {
+      if (cpool->tags[i] == tag
+         && cpool->data[i] == word1
+         && cpool->data[i+1] == word2)
+       return i;
+    }
+  i = cpool->count == 0 ? 1 : cpool->count;
+  set_constant_entry (cpool, i, tag, word1);
+  set_constant_entry (cpool, i+1, 0, word2);
+  return i;
+}
+
+int
+find_utf8_constant (cpool, name)
+     CPool *cpool;
+     tree name;
+{
+  if (name == NULL_TREE)
+    return 0;
+  return find_constant1 (cpool, CONSTANT_Utf8, (jword) name);
+}
+
+int
+find_class_or_string_constant (cpool, tag, name)
+     CPool *cpool;
+     int tag;
+     tree name;
+{
+  int j = find_utf8_constant (cpool, name);
+  int i;
+  for (i = cpool->count;  --i > 0; )
+    {
+      if (cpool->tags[i] == tag && cpool->data[i] == j)
+       return i;
+    }
+  i = cpool->count;
+  set_constant_entry (cpool, i, tag, (jword) j);
+  return i;
+}
+
+int
+find_class_constant (cpool, type)
+     CPool *cpool;
+     tree type;
+{
+  return find_class_or_string_constant (cpool, CONSTANT_Class,
+                                       build_internal_class_name (type));
+}
+
+/* Find (or create) a CONSTANT_NameAndType matching NAME and TYPE.
+   Return its index in the constant pool CPOOL. */
+
+int
+find_name_and_type_constant (cpool, name, type)
+     CPool *cpool;
+     tree name;
+     tree type;
+{
+  int name_index = find_utf8_constant (cpool, name);
+  int type_index = find_utf8_constant (cpool, build_java_signature (type));
+  return find_constant1 (cpool, CONSTANT_NameAndType,
+                        (name_index << 16) | type_index);
+}
+
+/* Find (or create) a CONSTANT_Fieldref for DECL (a FIELD_DECL or VAR_DECL).
+   Return its index in the constant pool CPOOL. */
+
+int
+find_fieldref_index (cpool, decl)
+     CPool *cpool;
+     tree decl;
+{
+  int class_index = find_class_constant (cpool, DECL_CONTEXT (decl));
+  int name_type_index
+    = find_name_and_type_constant (cpool, DECL_NAME (decl), TREE_TYPE (decl));
+  return find_constant1 (cpool, CONSTANT_Fieldref,
+                        (class_index << 16) | name_type_index);
+}
+
+/* Find (or create) a CONSTANT_Methodref for DECL (a FUNCTION_DECL).
+   Return its index in the constant pool CPOOL. */
+
+int
+find_methodref_index (cpool, decl)
+     CPool *cpool;
+     tree decl;
+{
+  int class_index = find_class_constant (cpool, DECL_CONTEXT (decl));
+  tree name = DECL_CONSTRUCTOR_P (decl) ? init_identifier_node
+    : DECL_NAME (decl);
+  int name_type_index
+    = find_name_and_type_constant (cpool, name, TREE_TYPE (decl));
+  /* Methodref or INterfacemethodRef - FIXME */
+  return find_constant1 (cpool, CONSTANT_Methodref,
+                        (class_index << 16) | name_type_index);
+}
+
+#define PUT1(X)  (*ptr++ = (X))
+#define PUT2(X)  (PUT1((X) >> 8), PUT1(X))
+#define PUT4(X)  (PUT2((X) >> 16), PUT2(X))
+#define PUTN(P, N)  (bcopy(P, ptr, N), ptr += (N))
+
+/* Give the number of bytes needed in a .class file for the CPOOL
+   constant pool.  Includes the 2-byte constant_pool_count. */
+
+int
+count_constant_pool_bytes (cpool)
+     CPool *cpool;
+{
+  int size = 2;
+  int i = 1;
+  jword *datap = &cpool->data[1];;
+  for ( ;  i < cpool->count;  i++, datap++)
+    {
+      size++;
+      switch (cpool->tags[i])
+       {
+       case CONSTANT_NameAndType:
+       case CONSTANT_Fieldref:
+       case CONSTANT_Methodref:
+       case CONSTANT_InterfaceMethodref:
+       case CONSTANT_Float:
+       case CONSTANT_Integer:
+         size += 4;
+         break;
+       case CONSTANT_Class:
+       case CONSTANT_String:
+         size += 2;
+         break;
+       case CONSTANT_Long:
+       case CONSTANT_Double:
+         size += 4;
+         break;
+       case CONSTANT_Utf8:
+         {
+           tree t = (tree) *datap;
+           int len = IDENTIFIER_LENGTH (t);
+           size += len + 2;
+         }
+         break;
+       }
+    }
+  return size;
+}
+
+/* Write the constant pool CPOOL into BUFFER.
+   The length of BUFFER is LENGTH, which must match the needed length. */
+
+void
+write_constant_pool (cpool, buffer, length)
+     CPool *cpool;
+     unsigned char* buffer;
+     int length;
+{
+  unsigned char* ptr = buffer;
+  int i = 1;
+  jword *datap = &cpool->data[1];
+  PUT2 (cpool->count);
+  for ( ;  i < cpool->count;  i++, datap++)
+    {
+      int tag = cpool->tags[i];
+      PUT1 (tag);
+      switch (tag)
+       {
+       case CONSTANT_NameAndType:
+       case CONSTANT_Fieldref:
+       case CONSTANT_Methodref:
+       case CONSTANT_InterfaceMethodref:
+       case CONSTANT_Float:
+       case CONSTANT_Integer:
+         PUT4 (*datap);
+         break;
+       case CONSTANT_Class:
+       case CONSTANT_String:
+         PUT2 (*datap);
+         break;
+         break;
+       case CONSTANT_Long:
+       case CONSTANT_Double:
+         PUT4(*datap);
+         i++;
+         datap++;
+         PUT4 (*datap);
+         break;
+       case CONSTANT_Utf8:
+         {
+           tree t = (tree) *datap;
+           int len = IDENTIFIER_LENGTH (t);
+           PUT2 (len);
+           PUTN (IDENTIFIER_POINTER (t), len);
+         }
+         break;
+       }
+    }
+  if (ptr != buffer + length)
+    fatal("internal error - incorrect constant pool");
+}
+
+CPool *outgoing_cpool;
+
+/* If non-NULL, an ADDR_EXPR referencing a VAR_DECL containing
+   the constant data array for the current class. */
+tree current_constant_pool_data_ref;
+
+/* A Cache for build_int_2 (CONSTANT_XXX, 0). */
+static tree tag_nodes[13];
+
+tree
+get_tag_node (tag)
+     int tag;
+{
+  if (tag_nodes[tag] == NULL_TREE)
+    {
+      push_obstacks (&permanent_obstack, &permanent_obstack);
+      tag_nodes[tag] = build_int_2 (tag, 0);
+      pop_obstacks ();
+    }
+  return tag_nodes[tag];
+}
+
+/* Look for a constant pool entry that matches TAG and NAME.
+   Creates a new entry if not found.
+   TAG is one of CONSTANT_Utf8, CONSTANT_String or CONSTANT_Class.
+   NAME is an IDENTIFIER_NODE naming the Utf8 constant, string, or class.
+   Returns the index of the entry. */
+
+int
+alloc_name_constant (tag, name)
+     int tag;
+     tree name;
+{
+  return find_constant1 (outgoing_cpool, tag, (jword) name);
+}
+
+/* Build an identifier for the internal name of reference type TYPE. */
+
+tree
+build_internal_class_name (type)
+     tree type;
+{
+  tree name;
+  if (TYPE_ARRAY_P (type))
+    name = build_java_signature (type);
+  else
+    {
+      name = TYPE_NAME (type);
+      if (TREE_CODE (name) != IDENTIFIER_NODE)
+       name = DECL_NAME (name);
+      name = identifier_subst (name, "", '.', '/', "");
+    }
+  return name;
+}
+
+/* Look for a CONSTANT_Class entry for CLAS, creating a new one if needed. */
+
+int
+alloc_class_constant (clas)
+     tree clas;
+{
+
+  return alloc_name_constant (CONSTANT_Class,
+                             build_internal_class_name (clas));
+}
+
+/* Return a reference to the data array of the current constant pool. */
+
+tree
+build_constant_data_ref ()
+{
+  if (current_constant_pool_data_ref == NULL_TREE)
+    {
+      tree decl;
+      tree decl_name = mangled_classname ("_CD_", current_class);
+      push_obstacks (&permanent_obstack, &permanent_obstack);
+      decl = build_decl (VAR_DECL, decl_name,
+                        build_array_type (ptr_type_node,
+                                          one_elt_array_domain_type));
+      TREE_STATIC (decl) = 1;
+      make_decl_rtl (decl, NULL, 1);
+      pop_obstacks ();
+      current_constant_pool_data_ref
+       = build1 (ADDR_EXPR, ptr_type_node, decl);
+    }
+  return current_constant_pool_data_ref;
+}
+
+/* Get the pointer value at the INDEX'th element of the constant pool. */
+
+tree
+build_ref_from_constant_pool (index)
+     int index;
+{
+  tree t = build_constant_data_ref ();
+  index *= int_size_in_bytes (ptr_type_node);
+  t = fold (build (PLUS_EXPR, ptr_type_node,
+                              t, build_int_2 (index, 0)));
+  return build1 (INDIRECT_REF, ptr_type_node, t);
+}
+
+/* Build an initializer for the constants field of the current constal pool.
+   Should only be called at top-level, since it may emit declarations. */
+
+tree
+build_constants_constructor ()
+{
+  tree tags_value, data_value;
+  tree cons;
+  tree tags_list = NULL_TREE;
+  tree data_list = NULL_TREE;
+  int i;
+  for (i = outgoing_cpool->count;  --i > 0; )
+    {
+      tags_list
+       = tree_cons (NULL_TREE, get_tag_node (outgoing_cpool->tags[i]),
+                    tags_list);
+      data_list
+       = tree_cons (NULL_TREE, build_utf8_ref ((tree)outgoing_cpool->data[i]),
+                    data_list);
+    }
+  if (outgoing_cpool->count > 0)
+    {
+      tree index_type;
+      tree data_decl, tags_decl, tags_type;
+      tree max_index = build_int_2 (outgoing_cpool->count - 1, 0);
+      TREE_TYPE (max_index) = sizetype;
+      index_type = build_index_type (max_index);
+
+      /* Add dummy 0'th element of constant pool. */
+      tags_list = tree_cons (NULL_TREE, get_tag_node (0), tags_list);
+      data_list = tree_cons (NULL_TREE, null_pointer_node, data_list);
+  
+      data_decl = TREE_OPERAND (build_constant_data_ref (), 0);
+      TREE_TYPE (data_decl) = build_array_type (ptr_type_node, index_type), 
+      DECL_INITIAL (data_decl) = build (CONSTRUCTOR, TREE_TYPE (data_decl),
+                                       NULL_TREE, data_list);
+      DECL_SIZE (data_decl) = TYPE_SIZE (TREE_TYPE (data_decl));
+      rest_of_decl_compilation (data_decl, (char*) 0, 1, 0);
+      data_value = build_address_of (data_decl);
+
+      tags_type = build_array_type (unsigned_byte_type_node, index_type);
+      tags_decl = build_decl (VAR_DECL, mangled_classname ("_CT_", 
+                                                          current_class),
+                             tags_type);
+      TREE_STATIC (tags_decl) = 1;
+      DECL_INITIAL (tags_decl) = build (CONSTRUCTOR, tags_type,
+                                       NULL_TREE, tags_list);
+      rest_of_decl_compilation (tags_decl, (char*) 0, 1, 0);
+      tags_value = build_address_of (tags_decl);
+    }
+  else
+    {
+      data_value = null_pointer_node;
+      tags_value = null_pointer_node;
+    }
+  START_RECORD_CONSTRUCTOR (cons, constants_type_node);
+  PUSH_FIELD_VALUE (cons, "size", build_int_2 (outgoing_cpool->count, 0));
+  PUSH_FIELD_VALUE (cons, "tags", tags_value);
+  PUSH_FIELD_VALUE (cons, "data", data_value);
+  FINISH_RECORD_CONSTRUCTOR (cons);
+  return cons;
+}
diff --git a/gcc/java/convert.h b/gcc/java/convert.h
new file mode 100644 (file)
index 0000000..1e1373e
--- /dev/null
@@ -0,0 +1,24 @@
+/* Definition of conversion functions.
+   Copyright (C) 1993 Free Software Foundation, Inc.
+
+This file is part of GNU CC.
+
+GNU CC is free software; you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation; either version 2, or (at your option)
+any later version.
+
+GNU CC is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with GNU CC; see the file COPYING.  If not, write to
+the Free Software Foundation, 59 Temple Place - Suite 330,
+Boston, MA 02111-1307, USA.  */
+
+/* Written by Jeffrey Hsu <hsu@cygnus.com> */
+
+extern tree convert_to_boolean PROTO ((tree, tree));
+extern tree convert_to_char    PROTO ((tree, tree));
diff --git a/gcc/java/decl.c b/gcc/java/decl.c
new file mode 100644 (file)
index 0000000..a12931f
--- /dev/null
@@ -0,0 +1,1561 @@
+/* Process declarations and variables for the GNU compiler for the
+   Java(TM) language.
+
+   Copyright (C) 1996, 1998 Free Software Foundation, Inc.
+
+This file is part of GNU CC.
+
+GNU CC is free software; you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation; either version 2, or (at your option)
+any later version.
+
+GNU CC is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with GNU CC; see the file COPYING.  If not, write to
+the Free Software Foundation, 59 Temple Place - Suite 330,
+Boston, MA 02111-1307, USA.
+
+Java and all Java-based marks are trademarks or registered trademarks
+of Sun Microsystems, Inc. in the United States and other countries.
+The Free Software Foundation is independent of Sun Microsystems, Inc.  */
+
+/* Hacked by Per Bothner <bothner@cygnus.com> February 1996. */
+
+#include "config.h"
+#include "tree.h"
+#include "java-tree.h"
+#include "jcf.h"
+#include "system.h"
+
+/* The DECL_MAP is a mapping from (index, type) to a decl node.
+   If index < max_locals, it is the index of a local variable.
+   if index >= max_locals, then index-max_locals is a stack slot.
+   The DECL_MAP mapping is represented as a TREE_VEC whose elements
+   are a list of decls (VAR_DECL or PARM_DECL) chained by
+   DECL_LOCAL_SLOT_CHAIN; the index finds the TREE_VEC element, and then
+   we search the chain for a decl with a matching TREE_TYPE. */
+
+tree decl_map;
+
+/* A list of local variables VAR_DECLs for this method that we have seen
+   debug information, but we have not reached their starting (byte) PC yet. */
+
+tree pending_local_decls = NULL_TREE;
+
+/* Push a local variable or stack slot into the decl_map,
+   and assign it an rtl. */
+
+tree
+push_jvm_slot (index, decl)
+     int index;
+     tree decl;
+{
+  struct rtx_def *rtl = NULL;
+  tree type = TREE_TYPE (decl);
+  tree tmp;
+
+  DECL_CONTEXT (decl) = current_function_decl;
+  layout_decl (decl, 0);
+
+  /* See if we have an appropriate rtl (i.e. same mode) at this index.
+     If so, we must use it. */ 
+  tmp = TREE_VEC_ELT (decl_map, index);
+  while (tmp != NULL_TREE)
+    {
+      if (TYPE_MODE (type) == TYPE_MODE (TREE_TYPE (tmp)))
+       rtl = DECL_RTL (tmp);
+      if (rtl != NULL)
+       break;
+     tmp = DECL_LOCAL_SLOT_CHAIN (tmp);
+    }
+  if (rtl != NULL)
+    DECL_RTL (decl) = rtl;
+  else
+    {
+      if (index >= DECL_MAX_LOCALS (current_function_decl))
+       DECL_REGISTER (decl) = 1;
+      expand_decl (decl);
+    }
+
+  /* Now link the decl into the decl_map. */
+  if (DECL_LANG_SPECIFIC (decl) == NULL)
+    {
+      DECL_LANG_SPECIFIC (decl)
+       = (struct lang_decl *) permalloc (sizeof (struct lang_decl_var));
+      DECL_LOCAL_START_PC (decl) = 0;
+      DECL_LOCAL_END_PC (decl) = DECL_CODE_LENGTH (current_function_decl);
+      DECL_LOCAL_SLOT_NUMBER (decl) = index;
+    }
+  DECL_LOCAL_SLOT_CHAIN (decl) = TREE_VEC_ELT (decl_map, index);
+  TREE_VEC_ELT (decl_map, index) = decl;
+  return decl;
+}
+
+/* Find a VAR_DECL (or PARM_DECL) at local index INDEX that has type TYPE,
+   that is valid at PC (or -1 if any pc).
+   If there is no existing matching decl, allocate one.
+   If we find a decl with matching modes but different types,
+   we re-use the rtl, but create a new decl. */
+
+tree
+find_local_variable (index, type, pc)
+     int index;
+     tree type;
+     int pc;
+{
+  struct rtx_def *rtl = NULL;
+  tree decl = TREE_VEC_ELT (decl_map, index);
+  tree best = NULL_TREE;
+  while (decl != NULL_TREE)
+    {
+      int in_range;
+      in_range = pc < 0
+       || (pc >= DECL_LOCAL_START_PC (decl)
+           && pc < DECL_LOCAL_END_PC (decl));
+
+      if ((TREE_TYPE (decl) == type
+          || (TREE_CODE (TREE_TYPE (decl)) == POINTER_TYPE
+              && type == ptr_type_node))
+          && in_range)
+       {
+         if (best == NULL_TREE
+             || (TREE_TYPE (decl) == type && TREE_TYPE (best) != type)
+             || DECL_LOCAL_START_PC (decl) > DECL_LOCAL_START_PC (best)
+             || DECL_LOCAL_END_PC (decl) < DECL_LOCAL_START_PC (decl))
+           best = decl;
+       }
+      decl = DECL_LOCAL_SLOT_CHAIN (decl);
+    }
+  if (best != NULL_TREE)
+    return best;
+  return push_jvm_slot (index, build_decl (VAR_DECL, NULL_TREE, type));
+}
+
+
+/* Same as find_local_index, except that INDEX is a stack index. */
+
+tree
+find_stack_slot (index, type)
+     int index;
+     tree type;
+{
+  return find_local_variable (index + DECL_MAX_LOCALS (current_function_decl),
+                             type, -1);
+}
+
+struct binding_level
+  {
+    /* A chain of _DECL nodes for all variables, constants, functions,
+     * and typedef types.  These are in the reverse of the order supplied.
+     */
+    tree names;
+
+    /* For each level, a list of shadowed outer-level local definitions
+       to be restored when this level is popped.
+       Each link is a TREE_LIST whose TREE_PURPOSE is an identifier and
+       whose TREE_VALUE is its old definition (a kind of ..._DECL node).  */
+    tree shadowed;
+
+    /* For each level (except not the global one),
+       a chain of BLOCK nodes for all the levels
+       that were entered and exited one level down.  */
+    tree blocks;
+
+    /* The BLOCK node for this level, if one has been preallocated.
+       If 0, the BLOCK is allocated (if needed) when the level is popped.  */
+    tree this_block;
+
+    /* The binding level which this one is contained in (inherits from).  */
+    struct binding_level *level_chain;
+
+    /* 1 means make a BLOCK for this level regardless of all else.
+       2 for temporary binding contours created by the compiler.  */
+    char keep;
+
+    /* Nonzero means make a BLOCK if this level has any subblocks.  */
+    char keep_if_subblocks;
+
+    /* Nonzero if this level can safely have additional
+       cleanup-needing variables added to it.  */
+    char more_cleanups_ok;
+    char have_cleanups;
+
+    /* The bytecode PC that marks the end of this level. */
+    int end_pc;
+  };
+
+#define NULL_BINDING_LEVEL (struct binding_level *) NULL
+
+/* The binding level currently in effect.  */
+
+static struct binding_level *current_binding_level;
+
+/* A chain of binding_level structures awaiting reuse.  */
+
+static struct binding_level *free_binding_level;
+
+/* The outermost binding level, for names of file scope.
+   This is created when the compiler is started and exists
+   through the entire run.  */
+
+static struct binding_level *global_binding_level;
+
+/* Binding level structures are initialized by copying this one.  */
+
+static struct binding_level clear_binding_level
+  = {NULL_TREE, NULL_TREE, NULL_TREE, NULL_TREE,
+       NULL_BINDING_LEVEL, 0, 0, 0, 0, 1000000000};
+
+#if 0
+/* A list (chain of TREE_LIST nodes) of all LABEL_DECLs in the function
+   that have names.  Here so we can clear out their names' definitions
+   at the end of the function.  */
+
+static tree named_labels;
+
+/* A list of LABEL_DECLs from outer contexts that are currently shadowed.  */
+
+static tree shadowed_labels;
+#endif
+
+int flag_traditional;
+
+/* Nonzero means unconditionally make a BLOCK for the next level pushed.  */
+
+static int keep_next_level_flag;
+
+/* Nonzero means make a BLOCK for the next level pushed
+   if it has subblocks.  */
+
+static int keep_next_if_subblocks;
+  
+/* The FUNCTION_DECL for the function currently being compiled,
+   or 0 if between functions.  */
+tree current_function_decl;
+
+/* The type node for the ordinary character type.  */
+tree char_type_node;
+
+tree object_type_node;
+tree object_ptr_type_node;
+tree string_type_node;
+tree throwable_type_node;
+
+tree boolean_type_node;
+
+tree float_type_node;
+tree double_type_node;
+
+/* a VOID_TYPE node.  */
+
+tree void_type_node;
+tree ptr_type_node;
+tree return_address_type_node;
+
+tree integer_type_node;
+
+tree byte_type_node;
+tree short_type_node;
+tree int_type_node;
+tree long_type_node;
+
+tree promoted_byte_type_node;
+tree promoted_short_type_node;
+tree promoted_char_type_node;
+tree promoted_boolean_type_node;
+
+tree unsigned_byte_type_node;
+tree unsigned_short_type_node;
+tree unsigned_int_type_node;
+tree unsigned_long_type_node;
+
+/* The type for struct methodtable. */
+tree methodtable_type;
+tree methodtable_ptr_type;
+
+tree utf8const_type;
+tree utf8const_ptr_type;
+tree class_type_node;
+tree class_ptr_type;
+tree field_type_node;
+tree field_ptr_type_node;
+tree field_info_union_node;
+tree jexception_type;
+tree jexception_ptr_type;
+tree lineNumberEntry_type;
+tree lineNumbers_type;
+tree constants_type_node;
+tree dtable_type;
+tree dtable_ptr_type;
+tree method_type_node;
+tree method_ptr_type_node;
+tree nativecode_ptr_array_type_node;
+tree one_elt_array_domain_type;
+tree access_flags_type_node;
+tree class_dtable_decl;
+
+/* a node which has tree code ERROR_MARK, and whose type is itself.
+   All erroneous expressions are replaced with this node.  All functions
+   that accept nodes as arguments should avoid generating error messages
+   if this node is one of the arguments, since it is undesirable to get
+   multiple error messages from one error in the input.  */
+
+tree error_mark_node;
+
+/* Two expressions that are constants with value zero.
+   The first is of type `int', the second of type `void *'.  */
+tree integer_zero_node;
+tree null_pointer_node;
+
+/* Nodes for boolean constants TRUE and FALSE. */
+tree boolean_true_node, boolean_false_node;
+
+tree TYPE_identifier_node;
+tree init_identifier_node;
+tree clinit_identifier_node;
+tree finalize_identifier_node;
+tree void_signature_node;
+tree length_identifier_node;
+tree this_identifier_node;
+tree super_identifier_node;
+
+/* References to internal libjava functions we use. */
+tree alloc_object_node;
+tree soft_instanceof_node;
+tree soft_checkcast_node;
+tree soft_initclass_node;
+tree soft_newarray_node;
+tree soft_anewarray_node;
+tree soft_multianewarray_node;
+tree soft_badarrayindex_node;
+tree throw_node;
+tree soft_checkarraystore_node;
+tree soft_monitorenter_node;
+tree soft_monitorexit_node;
+tree soft_lookupinterfacemethod_node;
+tree soft_fmod_node;
+
+/* Build (and pushdecl) a "promoted type" for all standard
+   types shorter than int.  */
+
+static tree
+push_promoted_type (name, actual_type)
+     char *name;
+     tree actual_type;
+{
+  tree type = make_node (TREE_CODE (actual_type));
+#if 1
+  tree in_min = TYPE_MIN_VALUE (int_type_node);
+  tree in_max = TYPE_MAX_VALUE (int_type_node);
+#else
+  tree in_min = TYPE_MIN_VALUE (actual_type);
+  tree in_max = TYPE_MAX_VALUE (actual_type);
+#endif
+  TYPE_MIN_VALUE (type) = build_int_2 (TREE_INT_CST_LOW (in_min),
+                                      TREE_INT_CST_HIGH (in_min));
+  TREE_TYPE (TYPE_MIN_VALUE (type)) = type;
+  TYPE_MAX_VALUE (type) = build_int_2 (TREE_INT_CST_LOW (in_max),
+                                      TREE_INT_CST_HIGH (in_max));
+  TREE_TYPE (TYPE_MAX_VALUE (type)) = type;
+  TYPE_PRECISION (type) = TYPE_PRECISION (int_type_node);
+  layout_type (type);
+  pushdecl (build_decl (TYPE_DECL, get_identifier (name), type));
+  return type;
+}
+
+/* Nodes for integer constants.  */
+tree integer_one_node, integer_two_node, integer_four_node;
+tree integer_negative_one_node;
+
+/* Return a definition for a builtin function named NAME and whose data type
+   is TYPE.  TYPE should be a function type with argument types.
+   FUNCTION_CODE tells later passes how to compile calls to this function.
+   See tree.h for its possible values.
+
+   If LIBRARY_NAME is nonzero, use that for DECL_ASSEMBLER_NAME,
+   the name to be called if we can't opencode the function.  */
+
+tree
+builtin_function (name, type, function_code, library_name)
+     char *name;
+     tree type;
+     enum built_in_function function_code;
+     char *library_name;
+{
+  tree decl = build_decl (FUNCTION_DECL, get_identifier (name), type);
+  DECL_EXTERNAL (decl) = 1;
+  TREE_PUBLIC (decl) = 1;
+  if (library_name)
+    DECL_ASSEMBLER_NAME (decl) = get_identifier (library_name);
+  make_decl_rtl (decl, NULL_PTR, 1);
+  pushdecl (decl);
+  if (function_code != NOT_BUILT_IN)
+    {
+      DECL_BUILT_IN (decl) = 1;
+      DECL_FUNCTION_CODE (decl) = function_code;
+    }
+  return decl;
+}
+
+void
+init_decl_processing ()
+{
+  tree field;
+  tree t;
+
+  current_function_decl = NULL;
+  current_binding_level = NULL_BINDING_LEVEL;
+  free_binding_level = NULL_BINDING_LEVEL;
+  pushlevel (0);       /* make the binding_level structure for global names */
+  global_binding_level = current_binding_level;
+
+  error_mark_node = make_node (ERROR_MARK);
+  TREE_TYPE (error_mark_node) = error_mark_node;
+
+  /* Create sizetype first - needed for other types. */
+  sizetype = make_unsigned_type (POINTER_SIZE);
+  size_zero_node = build_int_2 (0, 0);
+  TREE_TYPE (size_zero_node) = sizetype;
+  size_one_node = build_int_2 (1, 0);
+  TREE_TYPE (size_one_node) = sizetype;
+
+  byte_type_node = make_signed_type (8);
+  pushdecl (build_decl (TYPE_DECL, get_identifier ("byte"), byte_type_node));
+  short_type_node = make_signed_type (16);
+  pushdecl (build_decl (TYPE_DECL, get_identifier ("short"), short_type_node));
+  int_type_node = make_signed_type (32);
+  pushdecl (build_decl (TYPE_DECL, get_identifier ("int"), int_type_node));
+  long_type_node = make_signed_type (64);
+  pushdecl (build_decl (TYPE_DECL, get_identifier ("long"), long_type_node));
+
+  unsigned_byte_type_node = make_unsigned_type (8);
+  pushdecl (build_decl (TYPE_DECL, get_identifier ("unsigned byte"),
+                       unsigned_byte_type_node));
+  unsigned_short_type_node = make_unsigned_type (16);
+  pushdecl (build_decl (TYPE_DECL, get_identifier ("unsigned short"),
+                       unsigned_short_type_node));
+  unsigned_int_type_node = make_unsigned_type (32);
+  pushdecl (build_decl (TYPE_DECL, get_identifier ("unsigned int"),
+                       unsigned_int_type_node));
+  unsigned_long_type_node = make_unsigned_type (64);
+  pushdecl (build_decl (TYPE_DECL, get_identifier ("unsigned long"),
+                       unsigned_long_type_node));
+
+  integer_type_node = int_type_node;
+
+  integer_zero_node = build_int_2 (0, 0);
+  integer_one_node = build_int_2 (1, 0);
+  integer_two_node = build_int_2 (2, 0);
+  integer_four_node = build_int_2 (4, 0);
+  integer_negative_one_node = build_int_2 (-1, 0);
+
+  void_type_node = make_node (VOID_TYPE);
+  pushdecl (build_decl (TYPE_DECL, get_identifier ("void"), void_type_node));
+  layout_type (void_type_node);        /* Uses size_zero_node */
+  ptr_type_node = build_pointer_type (void_type_node);
+  t = make_node (VOID_TYPE);
+  layout_type (t); /* Uses size_zero_node */
+  return_address_type_node = build_pointer_type (t);
+
+  null_pointer_node = build_int_2 (0, 0);
+  TREE_TYPE (null_pointer_node) = ptr_type_node;
+
+#if 0
+  /* Make a type to be the domain of a few array types
+     whose domains don't really matter.
+     200 is small enough that it always fits in size_t
+     and large enough that it can hold most function names for the
+     initializations of __FUNCTION__ and __PRETTY_FUNCTION__.  */
+  short_array_type_node = build_prim_array_type (short_type_node, 200);
+#endif
+  char_type_node = make_node (CHAR_TYPE);
+  TYPE_PRECISION (char_type_node) = 16;
+  fixup_unsigned_type (char_type_node);
+  pushdecl (build_decl (TYPE_DECL, get_identifier ("char"), char_type_node));
+
+  boolean_type_node = make_node (BOOLEAN_TYPE);
+  TYPE_PRECISION (boolean_type_node) = 1;
+  fixup_unsigned_type (boolean_type_node);
+  pushdecl (build_decl (TYPE_DECL, get_identifier ("boolean"),
+                       boolean_type_node));
+  boolean_false_node = TYPE_MIN_VALUE (boolean_type_node);
+  boolean_true_node = TYPE_MAX_VALUE (boolean_type_node);
+
+  promoted_byte_type_node
+    = push_promoted_type ("promoted_byte", byte_type_node);
+  promoted_short_type_node
+    = push_promoted_type ("promoted_short", short_type_node);
+  promoted_char_type_node
+    = push_promoted_type ("promoted_char", char_type_node);
+  promoted_boolean_type_node
+    = push_promoted_type ("promoted_boolean", boolean_type_node);
+
+  float_type_node = make_node (REAL_TYPE);
+  TYPE_PRECISION (float_type_node) = 32;
+  pushdecl (build_decl (TYPE_DECL, get_identifier ("float"),
+                        float_type_node));
+  layout_type (float_type_node);
+
+  double_type_node = make_node (REAL_TYPE);
+  TYPE_PRECISION (double_type_node) = 64;
+  pushdecl (build_decl (TYPE_DECL, get_identifier ("double"),
+                        double_type_node));
+  layout_type (double_type_node);
+
+  object_type_node = lookup_class (get_identifier ("java.lang.Object"));
+  object_ptr_type_node = promote_type (object_type_node);
+  string_type_node = lookup_class (get_identifier ("java.lang.String"));
+  class_type_node = lookup_class (get_identifier ("java.lang.Class"));
+  throwable_type_node = lookup_class (get_identifier ("java.lang.Throwable"));
+
+  methodtable_type = make_node (RECORD_TYPE);
+  layout_type (methodtable_type);
+  pushdecl (build_decl (TYPE_DECL, get_identifier ("methodtable"),
+                       methodtable_type));
+  methodtable_ptr_type = build_pointer_type (methodtable_type);
+
+  TYPE_identifier_node = get_identifier ("TYPE");
+  init_identifier_node = get_identifier ("<init>");
+  clinit_identifier_node = get_identifier ("<clinit>");
+  finalize_identifier_node = get_identifier ("finalize");
+  void_signature_node = get_identifier ("()V");
+  length_identifier_node = get_identifier ("length");
+  this_identifier_node = get_identifier ("this");
+  super_identifier_node = get_identifier ("super");
+
+  /* for lack of a better place to put this stub call */
+  init_expr_processing();
+
+  utf8const_type = make_node (RECORD_TYPE);
+  PUSH_FIELD (utf8const_type, field, "hash", unsigned_short_type_node);
+  PUSH_FIELD (utf8const_type, field, "length", unsigned_short_type_node);
+  FINISH_RECORD (utf8const_type);
+  utf8const_ptr_type = build_pointer_type (utf8const_type);
+
+  constants_type_node = make_node (RECORD_TYPE);
+  PUSH_FIELD (constants_type_node, field, "size", unsigned_int_type_node);
+  PUSH_FIELD (constants_type_node, field, "tags", ptr_type_node);
+  PUSH_FIELD (constants_type_node, field, "data", ptr_type_node);
+  FINISH_RECORD (constants_type_node);
+  pushdecl (build_decl (TYPE_DECL, get_identifier ("constants"),
+                       constants_type_node));
+
+  access_flags_type_node = unsigned_short_type_node;
+
+  dtable_type = make_node (RECORD_TYPE);
+  dtable_ptr_type = build_pointer_type (dtable_type);
+
+  PUSH_FIELD (object_type_node, field, "dtable", dtable_ptr_type);
+  PUSH_FIELD (object_type_node, field, "sync_info", ptr_type_node);
+  for (t = TYPE_FIELDS (object_type_node); t != NULL_TREE; t = TREE_CHAIN (t))
+    FIELD_PRIVATE (t) = 1;
+  FINISH_RECORD (object_type_node);
+
+  class_dtable_decl = build_dtable_decl (class_type_node);
+  TREE_STATIC (class_dtable_decl) = 1;
+  DECL_ARTIFICIAL (class_dtable_decl) = 1;
+  DECL_IGNORED_P (class_dtable_decl) = 1;
+  rest_of_decl_compilation (class_dtable_decl, (char*) 0, 1, 0);
+
+  field_type_node = make_node (RECORD_TYPE);
+  field_ptr_type_node = build_pointer_type (field_type_node);
+  method_type_node = make_node (RECORD_TYPE);
+  method_ptr_type_node = build_pointer_type (method_type_node);
+
+  set_super_info (0, class_type_node, object_type_node, 0);
+  set_super_info (0, string_type_node, object_type_node, 0);
+  class_ptr_type = build_pointer_type (class_type_node);
+
+  PUSH_FIELD (class_type_node, field, "next", class_ptr_type);
+  PUSH_FIELD (class_type_node, field, "name", utf8const_ptr_type);
+  PUSH_FIELD (class_type_node, field, "accflags", access_flags_type_node);
+  PUSH_FIELD (class_type_node, field, "superclass", class_ptr_type);
+  PUSH_FIELD (class_type_node, field, "subclass_head", class_ptr_type);
+  PUSH_FIELD (class_type_node, field, "subclass_next", class_ptr_type);
+  PUSH_FIELD (class_type_node, field, "constants", constants_type_node);
+  PUSH_FIELD (class_type_node, field, "methods", method_ptr_type_node);
+  PUSH_FIELD (class_type_node, field, "nmethods", short_type_node);
+  PUSH_FIELD (class_type_node, field, "msize", short_type_node);
+  PUSH_FIELD (class_type_node, field, "fields", field_ptr_type_node);
+  PUSH_FIELD (class_type_node, field, "bfsize", int_type_node);
+  PUSH_FIELD (class_type_node, field, "nfields", short_type_node);
+  PUSH_FIELD (class_type_node, field, "nsfields", short_type_node);
+  PUSH_FIELD (class_type_node, field, "dtable", dtable_ptr_type);
+  PUSH_FIELD (class_type_node, field, "interfaces",
+             build_pointer_type (class_ptr_type));
+  PUSH_FIELD (class_type_node, field, "loader", ptr_type_node);
+  PUSH_FIELD (class_type_node, field, "interface_len", short_type_node);
+  PUSH_FIELD (class_type_node, field, "state", byte_type_node);
+  PUSH_FIELD (class_type_node, field, "final", byte_type_node);
+  for (t = TYPE_FIELDS (class_type_node);  t != NULL_TREE;  t = TREE_CHAIN (t))
+    FIELD_PRIVATE (t) = 1;
+  push_super_field (class_type_node, object_type_node);
+  FINISH_RECORD (class_type_node);
+  pushdecl (build_decl (TYPE_DECL, get_identifier ("Class"), class_type_node));
+
+  field_info_union_node = make_node (UNION_TYPE);
+  PUSH_FIELD (field_info_union_node, field, "boffset", int_type_node);
+  PUSH_FIELD (field_info_union_node, field, "addr", ptr_type_node);
+#if 0
+  PUSH_FIELD (field_info_union_node, field, "idx", unsigned_short_type_node);
+#endif
+  layout_type (field_info_union_node);
+
+  PUSH_FIELD (field_type_node, field, "name", utf8const_ptr_type);
+  PUSH_FIELD (field_type_node, field, "type", class_ptr_type);
+  PUSH_FIELD (field_type_node, field, "accflags", access_flags_type_node);
+  PUSH_FIELD (field_type_node, field, "bsize", unsigned_short_type_node);
+  PUSH_FIELD (field_type_node, field, "info", field_info_union_node);
+  FINISH_RECORD (field_type_node);
+  CLASS_LOADED_P (field_type_node) = 1;
+  pushdecl (build_decl (TYPE_DECL, get_identifier ("Field"), field_type_node));
+
+  one_elt_array_domain_type = build_index_type (integer_one_node);
+  nativecode_ptr_array_type_node
+    = build_array_type (nativecode_ptr_type_node, one_elt_array_domain_type);
+
+  PUSH_FIELD (dtable_type, field, "class", class_ptr_type);
+  PUSH_FIELD (dtable_type, field, "methods", nativecode_ptr_array_type_node);
+  FINISH_RECORD (dtable_type);
+  pushdecl (build_decl (TYPE_DECL, get_identifier ("dispatchTable"), dtable_type));
+
+#define jint_type int_type_node
+#define jint_ptr_type ptr_type_node
+
+  jexception_type = make_node (RECORD_TYPE);
+  PUSH_FIELD (jexception_type, field, "start_pc", ptr_type_node);
+  PUSH_FIELD (jexception_type, field, "end_pc", ptr_type_node);
+  PUSH_FIELD (jexception_type, field, "handler_pc", ptr_type_node);
+  PUSH_FIELD (jexception_type, field, "catch_type", class_ptr_type);
+  FINISH_RECORD (jexception_type);
+  pushdecl (build_decl (TYPE_DECL, get_identifier ("jexception"), field_type_node));
+  jexception_ptr_type = build_pointer_type (jexception_type);
+
+  lineNumberEntry_type = make_node (RECORD_TYPE);
+  PUSH_FIELD (lineNumberEntry_type, field, "line_nr", unsigned_short_type_node);
+  PUSH_FIELD (lineNumberEntry_type, field, "start_pc", ptr_type_node);
+  FINISH_RECORD (lineNumberEntry_type);
+
+  lineNumbers_type = make_node (RECORD_TYPE);
+  PUSH_FIELD (lineNumbers_type, field, "length", unsigned_int_type_node);
+  FINISH_RECORD (lineNumbers_type);
+
+#define instn_ptr_type_node ptr_type_node      /* XXX JH */
+
+#define lineNumbers_ptr_type_node build_pointer_type(lineNumbers_type)
+
+  PUSH_FIELD (method_type_node, field, "name", utf8const_ptr_type);
+  PUSH_FIELD (method_type_node, field, "signature", utf8const_ptr_type);
+  PUSH_FIELD (method_type_node, field, "accflags", access_flags_type_node);
+  PUSH_FIELD (method_type_node, field, "ncode", nativecode_ptr_type_node);
+  FINISH_RECORD (method_type_node);
+  CLASS_LOADED_P (method_type_node) = 1;
+  pushdecl (build_decl (TYPE_DECL, get_identifier ("Method"), method_type_node));
+
+  t = tree_cons (NULL_TREE, class_ptr_type,
+                build_tree_list (NULL_TREE, int_type_node));
+  alloc_object_node = builtin_function ("_Jv_AllocObject",
+                                       build_function_type (ptr_type_node, t),
+                                       NOT_BUILT_IN, NULL_PTR);
+  soft_initclass_node = builtin_function ("_Jv_InitClass",
+                                         build_function_type (void_type_node,
+                                                              t),
+                                         NOT_BUILT_IN, NULL_PTR);
+  t = build_tree_list (NULL_TREE, void_type_node);
+  throw_node = builtin_function ("_Jv_Throw",
+                                build_function_type (ptr_type_node, t),
+                                NOT_BUILT_IN, NULL_PTR);
+  soft_monitorenter_node 
+      = builtin_function ("_Jv_MonitorEnter",
+                         build_function_type (int_type_node, t),
+                         NOT_BUILT_IN, NULL_PTR);
+  soft_monitorexit_node 
+      = builtin_function ("_Jv_MonitorExit",
+                         build_function_type (int_type_node, t),
+                         NOT_BUILT_IN, NULL_PTR);
+  
+  t = tree_cons (NULL_TREE, int_type_node, 
+                build_tree_list (NULL_TREE, int_type_node));
+  soft_newarray_node
+      = builtin_function ("_Jv_NewArray",
+                         build_function_type(ptr_type_node, t),
+                         NOT_BUILT_IN, NULL_PTR );
+
+  t = tree_cons (NULL_TREE, int_type_node,
+                tree_cons (NULL_TREE, class_ptr_type,
+                           build_tree_list (NULL_TREE, object_ptr_type_node)));
+  soft_anewarray_node
+      = builtin_function ("_Jv_NewObjectArray",
+                         build_function_type (ptr_type_node, t),
+                         NOT_BUILT_IN, NULL_PTR );
+
+  t = tree_cons (NULL_TREE, ptr_type_node,
+                build_tree_list (NULL_TREE, int_type_node));
+  soft_multianewarray_node
+      = builtin_function ("_Jv_NewMultiArray",
+                         build_function_type (ptr_type_node, t),
+                         NOT_BUILT_IN, NULL_PTR );
+
+  t = build_function_type (void_type_node, NULL_TREE);
+  soft_badarrayindex_node
+      = builtin_function ("_Jv_ThrowBadArrayIndex", t, NOT_BUILT_IN, NULL_PTR );
+  TREE_THIS_VOLATILE (soft_badarrayindex_node) = 1;
+  TREE_SIDE_EFFECTS (soft_badarrayindex_node) = 1;
+
+  t = tree_cons (NULL_TREE, class_ptr_type,
+                build_tree_list (NULL_TREE, object_ptr_type_node));
+  soft_checkcast_node
+    = builtin_function ("_Jv_CheckCast",
+                       build_function_type (ptr_type_node, t),
+                       NOT_BUILT_IN, NULL_PTR);
+  t = tree_cons (NULL_TREE, object_ptr_type_node,
+                build_tree_list (NULL_TREE, class_ptr_type));
+  soft_instanceof_node
+    = builtin_function ("_Jv_IsInstanceOf",
+                       build_function_type (promoted_boolean_type_node, t),
+                       NOT_BUILT_IN, NULL_PTR);
+  t = tree_cons (NULL_TREE, object_ptr_type_node,
+                build_tree_list (NULL_TREE, object_ptr_type_node));
+  soft_checkarraystore_node
+    = builtin_function ("_Jv_CheckArrayStore",
+                       build_function_type (void_type_node, t),
+                       NOT_BUILT_IN, NULL_PTR);
+  t = tree_cons (NULL_TREE, ptr_type_node,
+                tree_cons (NULL_TREE, ptr_type_node,
+                           build_tree_list (NULL_TREE, ptr_type_node)));
+  soft_lookupinterfacemethod_node 
+    = builtin_function ("_Jv_LookupInterfaceMethod",
+                       build_function_type(ptr_type_node, t),
+                       NOT_BUILT_IN, NULL_PTR);
+  t = tree_cons (NULL_TREE, double_type_node,
+                build_tree_list (NULL_TREE, double_type_node));
+  soft_fmod_node
+    = builtin_function ("__builtin_fmod",
+                       build_function_type (double_type_node, t),
+                       BUILT_IN_FMOD, "fmod");
+#if 0
+  t = tree_cons (NULL_TREE, float_type_node,
+                build_tree_list (NULL_TREE, float_type_node));
+  soft_fmodf_node
+    = builtin_function ("__builtin_fmodf",
+                       build_function_type (float_type_node, t),
+                       BUILT_IN_FMOD, "fmodf");
+#endif
+    
+  init_class_processing ();
+}
+
+
+/* Look up NAME in the current binding level and its superiors
+   in the namespace of variables, functions and typedefs.
+   Return a ..._DECL node of some kind representing its definition,
+   or return 0 if it is undefined.  */
+
+tree
+lookup_name (name)
+     tree name;
+{
+  register tree val;
+  if (current_binding_level != global_binding_level
+      && IDENTIFIER_LOCAL_VALUE (name))
+    val = IDENTIFIER_LOCAL_VALUE (name);
+  else
+    val = IDENTIFIER_GLOBAL_VALUE (name);
+  return val;
+}
+
+/* Similar to `lookup_name' but look only at current binding level and
+   the previous one if its the parameter level.  */
+
+tree
+lookup_name_current_level (name)
+     tree name;
+{
+  register tree t;
+
+  if (current_binding_level == global_binding_level)
+    return IDENTIFIER_GLOBAL_VALUE (name);
+
+  if (IDENTIFIER_LOCAL_VALUE (name) == 0)
+    return 0;
+
+  for (t = current_binding_level->names; t; t = TREE_CHAIN (t))
+    if (DECL_NAME (t) == name)
+      break;
+
+  return t;
+}
+
+/* Use a binding level to record a labeled block declaration */
+
+void
+push_labeled_block (lb)
+    tree lb;
+{
+  register tree name = DECL_NAME (LABELED_BLOCK_LABEL (lb));
+  register struct binding_level *b = current_binding_level;
+  tree oldlocal = IDENTIFIER_LOCAL_VALUE (name);
+  if (oldlocal != 0)
+      b->shadowed = tree_cons (name, oldlocal, b->shadowed);
+  TREE_CHAIN (lb) = b->names;
+  b->names = lb;
+  IDENTIFIER_LOCAL_VALUE (name) = lb;
+}
+
+/* Pop the current binding level, reinstalling values for the previous
+   labeled block */
+
+void
+pop_labeled_block ()
+{
+  struct binding_level *b = current_binding_level;
+  tree label =  b->names;
+  IDENTIFIER_LOCAL_VALUE (DECL_NAME (LABELED_BLOCK_LABEL (label))) = 
+    NULL_TREE;
+  if (b->shadowed)
+    IDENTIFIER_LOCAL_VALUE (TREE_PURPOSE (b->shadowed)) = 
+      TREE_VALUE (b->shadowed);
+
+  /* Pop the current level, and free the structure for reuse.  */
+  current_binding_level = current_binding_level->level_chain;
+  b->level_chain = free_binding_level;
+  free_binding_level = b;
+}
+
+/* Record a decl-node X as belonging to the current lexical scope.
+   Check for errors (such as an incompatible declaration for the same
+   name already seen in the same scope).
+
+   Returns either X or an old decl for the same name.
+   If an old decl is returned, it may have been smashed
+   to agree with what X says.  */
+
+tree
+pushdecl (x)
+     tree x;
+{
+  register tree t;
+  register tree name = DECL_NAME (x);
+  register struct binding_level *b = current_binding_level;
+
+  DECL_CONTEXT (x) = current_function_decl;
+  if (name)
+    {
+      char *file;
+      int line;
+      int different_binding_level = 0;
+
+      t = lookup_name_current_level (name);
+      if (t != 0 && t == error_mark_node)
+       /* error_mark_node is 0 for a while during initialization!  */
+       {
+         t = 0;
+         error_with_decl (x, "`%s' used prior to declaration");
+       }
+
+      if (t != 0)
+       {
+         file = DECL_SOURCE_FILE (t);
+         line = DECL_SOURCE_LINE (t);
+       }
+
+      /* If we're naming a hitherto-unnamed type, set its TYPE_NAME
+        to point to the TYPE_DECL.
+        Since Java does not have typedefs, a type can only have
+        one (true) name, given by a class, interface, or builtin. */
+      if (TREE_CODE (x) == TYPE_DECL
+         && TYPE_NAME (TREE_TYPE (x)) == 0
+         && TREE_TYPE (x) != error_mark_node)
+       {
+         TYPE_NAME (TREE_TYPE (x)) = x;
+         TYPE_STUB_DECL (TREE_TYPE (x)) = x;
+       }
+
+      /* This name is new in its binding level.
+        Install the new declaration and return it.  */
+      if (b == global_binding_level)
+       {
+         /* Install a global value.  */
+         
+         IDENTIFIER_GLOBAL_VALUE (name) = x;
+       }
+      else
+       {
+         /* Here to install a non-global value.  */
+         tree oldlocal = IDENTIFIER_LOCAL_VALUE (name);
+         tree oldglobal = IDENTIFIER_GLOBAL_VALUE (name);
+         IDENTIFIER_LOCAL_VALUE (name) = x;
+
+#if 0
+         /* Warn if shadowing an argument at the top level of the body.  */
+         if (oldlocal != 0 && !DECL_EXTERNAL (x)
+             /* This warning doesn't apply to the parms of a nested fcn.  */
+             && ! current_binding_level->parm_flag
+             /* Check that this is one level down from the parms.  */
+             && current_binding_level->level_chain->parm_flag
+             /* Check that the decl being shadowed
+                comes from the parm level, one level up.  */
+             && chain_member (oldlocal, current_binding_level->level_chain->names))
+           {
+             if (TREE_CODE (oldlocal) == PARM_DECL)
+               pedwarn ("declaration of `%s' shadows a parameter",
+                        IDENTIFIER_POINTER (name));
+             else
+               pedwarn ("declaration of `%s' shadows a symbol from the parameter list",
+                        IDENTIFIER_POINTER (name));
+           }
+
+         /* Maybe warn if shadowing something else.  */
+         else if (warn_shadow && !DECL_EXTERNAL (x)
+                  /* No shadow warnings for internally generated vars.  */
+                  && DECL_SOURCE_LINE (x) != 0
+                  /* No shadow warnings for vars made for inlining.  */
+                  && ! DECL_FROM_INLINE (x))
+           {
+             char *warnstring = 0;
+
+             if (TREE_CODE (x) == PARM_DECL
+                 && current_binding_level->level_chain->parm_flag)
+               /* Don't warn about the parm names in function declarator
+                  within a function declarator.
+                  It would be nice to avoid warning in any function
+                  declarator in a declaration, as opposed to a definition,
+                  but there is no way to tell it's not a definition.  */
+               ;
+             else if (oldlocal != 0 && TREE_CODE (oldlocal) == PARM_DECL)
+               warnstring = "declaration of `%s' shadows a parameter";
+             else if (oldlocal != 0)
+               warnstring = "declaration of `%s' shadows previous local";
+             else if (IDENTIFIER_GLOBAL_VALUE (name) != 0
+                      && IDENTIFIER_GLOBAL_VALUE (name) != error_mark_node)
+               warnstring = "declaration of `%s' shadows global declaration";
+
+             if (warnstring)
+               warning (warnstring, IDENTIFIER_POINTER (name));
+           }
+#endif
+
+         /* If storing a local value, there may already be one (inherited).
+            If so, record it for restoration when this binding level ends.  */
+         if (oldlocal != 0)
+           b->shadowed = tree_cons (name, oldlocal, b->shadowed);
+       }
+    }
+
+  /* Put decls on list in reverse order.
+     We will reverse them later if necessary.  */
+  TREE_CHAIN (x) = b->names;
+  b->names = x;
+
+  return x;
+}
+void
+pushdecl_force_head (x)
+     tree x;
+{
+  current_binding_level->names = x;
+}
+
+/* Like pushdecl, only it places X in GLOBAL_BINDING_LEVEL, if appropriate.  */
+
+tree
+pushdecl_top_level (x)
+     tree x;
+{
+  register tree t;
+  register struct binding_level *b = current_binding_level;
+
+  current_binding_level = global_binding_level;
+  t = pushdecl (x);
+  current_binding_level = b;
+  return t;
+}
+
+/* Nonzero if we are currently in the global binding level.  */
+
+int
+global_bindings_p ()
+{
+  return current_binding_level == global_binding_level;
+}
+
+/* Return the list of declarations of the current level.
+   Note that this list is in reverse order unless/until
+   you nreverse it; and when you do nreverse it, you must
+   store the result back using `storedecls' or you will lose.  */
+
+tree
+getdecls ()
+{
+  return current_binding_level->names;
+}
+
+/* Create a new `struct binding_level'.  */
+
+static
+struct binding_level *
+make_binding_level ()
+{
+  /* NOSTRICT */
+  return (struct binding_level *) xmalloc (sizeof (struct binding_level));
+}
+
+void
+pushlevel (unused)
+     int unused;
+{
+  register struct binding_level *newlevel = NULL_BINDING_LEVEL;
+
+#if 0
+  /* If this is the top level of a function,
+     just make sure that NAMED_LABELS is 0.  */
+
+  if (current_binding_level == global_binding_level)
+    named_labels = 0;
+#endif
+
+  /* Reuse or create a struct for this binding level.  */
+
+  if (free_binding_level)
+    {
+      newlevel = free_binding_level;
+      free_binding_level = free_binding_level->level_chain;
+    }
+  else
+    {
+      newlevel = make_binding_level ();
+    }
+
+  /* Add this level to the front of the chain (stack) of levels that
+     are active.  */
+
+  *newlevel = clear_binding_level;
+  newlevel->level_chain = current_binding_level;
+  current_binding_level = newlevel;
+  newlevel->keep = keep_next_level_flag;
+  keep_next_level_flag = 0;
+  newlevel->keep_if_subblocks = keep_next_if_subblocks;
+  keep_next_if_subblocks = 0;
+}
+
+/* Exit a binding level.
+   Pop the level off, and restore the state of the identifier-decl mappings
+   that were in effect when this level was entered.
+
+   If KEEP is nonzero, this level had explicit declarations, so
+   and create a "block" (a BLOCK node) for the level
+   to record its declarations and subblocks for symbol table output.
+
+   If FUNCTIONBODY is nonzero, this level is the body of a function,
+   so create a block as if KEEP were set and also clear out all
+   label names.
+
+   If REVERSE is nonzero, reverse the order of decls before putting
+   them into the BLOCK.  */
+
+tree
+poplevel (keep, reverse, functionbody)
+     int keep;
+     int reverse;
+     int functionbody;
+{
+  register tree link;
+  /* The chain of decls was accumulated in reverse order.
+     Put it into forward order, just for cleanliness.  */
+  tree decls;
+  tree subblocks = current_binding_level->blocks;
+  tree block = 0;
+  tree decl;
+  int block_previously_created;
+
+  keep |= current_binding_level->keep;
+
+  /* Get the decls in the order they were written.
+     Usually current_binding_level->names is in reverse order.
+     But parameter decls were previously put in forward order.  */
+
+  if (reverse)
+    current_binding_level->names
+      = decls = nreverse (current_binding_level->names);
+  else
+    decls = current_binding_level->names;
+
+  /* Output any nested inline functions within this block
+     if they weren't already output.  */
+
+  for (decl = decls; decl; decl = TREE_CHAIN (decl))
+    if (TREE_CODE (decl) == FUNCTION_DECL
+       && ! TREE_ASM_WRITTEN (decl)
+       && DECL_INITIAL (decl) != 0
+       && TREE_ADDRESSABLE (decl))
+      {
+       /* If this decl was copied from a file-scope decl
+          on account of a block-scope extern decl,
+          propagate TREE_ADDRESSABLE to the file-scope decl.
+
+          DECL_ABSTRACT_ORIGIN can be set to itself if warn_return_type is
+          true, since then the decl goes through save_for_inline_copying.  */
+       if (DECL_ABSTRACT_ORIGIN (decl) != 0
+           && DECL_ABSTRACT_ORIGIN (decl) != decl)
+         TREE_ADDRESSABLE (DECL_ABSTRACT_ORIGIN (decl)) = 1;
+       else
+         {
+           push_function_context ();
+           output_inline_function (decl);
+           pop_function_context ();
+         }
+      }
+
+  /* If there were any declarations in that level,
+     or if this level is a function body,
+     create a BLOCK to record them for the life of this function.  */
+
+  block = 0;
+  block_previously_created = (current_binding_level->this_block != 0);
+  if (block_previously_created)
+    block = current_binding_level->this_block;
+  else if (keep || functionbody
+          || (current_binding_level->keep_if_subblocks && subblocks != 0))
+    block = make_node (BLOCK);
+  if (block != 0)
+    {
+      BLOCK_VARS (block) = decls;
+      BLOCK_TYPE_TAGS (block) = NULL_TREE;
+      BLOCK_SUBBLOCKS (block) = subblocks;
+      remember_end_note (block);
+    }
+
+  /* In each subblock, record that this is its superior.  */
+
+  for (link = subblocks; link; link = TREE_CHAIN (link))
+    BLOCK_SUPERCONTEXT (link) = block;
+
+  /* Clear out the meanings of the local variables of this level.  */
+
+  for (link = decls; link; link = TREE_CHAIN (link))
+    {
+      tree name = DECL_NAME (link);
+      if (name != 0 && IDENTIFIER_LOCAL_VALUE (name) == link)
+       {
+         /* If the ident. was used or addressed via a local extern decl,
+            don't forget that fact.  */
+         if (DECL_EXTERNAL (link))
+           {
+             if (TREE_USED (link))
+               TREE_USED (name) = 1;
+             if (TREE_ADDRESSABLE (link))
+               TREE_ADDRESSABLE (DECL_ASSEMBLER_NAME (link)) = 1;
+           }
+         IDENTIFIER_LOCAL_VALUE (name) = 0;
+       }
+    }
+
+  /* Restore all name-meanings of the outer levels
+     that were shadowed by this level.  */
+
+  for (link = current_binding_level->shadowed; link; link = TREE_CHAIN (link))
+    IDENTIFIER_LOCAL_VALUE (TREE_PURPOSE (link)) = TREE_VALUE (link);
+
+  /* If the level being exited is the top level of a function,
+     check over all the labels, and clear out the current
+     (function local) meanings of their names.  */
+
+  if (functionbody)
+    {
+      /* If this is the top level block of a function,
+        the vars are the function's parameters.
+        Don't leave them in the BLOCK because they are
+        found in the FUNCTION_DECL instead.  */
+
+      BLOCK_VARS (block) = 0;
+
+      /* Clear out the definitions of all label names,
+        since their scopes end here,
+        and add them to BLOCK_VARS.  */
+
+#if 0
+      for (link = named_labels; link; link = TREE_CHAIN (link))
+       {
+         register tree label = TREE_VALUE (link);
+
+         if (DECL_INITIAL (label) == 0)
+           {
+             error_with_decl (label, "label `%s' used but not defined");
+             /* Avoid crashing later.  */
+             define_label (input_filename, lineno,
+                           DECL_NAME (label));
+           }
+         else if (warn_unused && !TREE_USED (label))
+           warning_with_decl (label, "label `%s' defined but not used");
+         IDENTIFIER_LABEL_VALUE (DECL_NAME (label)) = 0;
+
+         /* Put the labels into the "variables" of the
+            top-level block, so debugger can see them.  */
+         TREE_CHAIN (label) = BLOCK_VARS (block);
+         BLOCK_VARS (block) = label;
+       }
+#endif
+    }
+
+  /* Pop the current level, and free the structure for reuse.  */
+
+  {
+    register struct binding_level *level = current_binding_level;
+    current_binding_level = current_binding_level->level_chain;
+
+    level->level_chain = free_binding_level;
+    free_binding_level = level;
+  }
+
+  /* Dispose of the block that we just made inside some higher level.  */
+  if (functionbody)
+    DECL_INITIAL (current_function_decl) = block;
+  else if (block)
+    {
+      if (!block_previously_created)
+        current_binding_level->blocks
+          = chainon (current_binding_level->blocks, block);
+    }
+  /* If we did not make a block for the level just exited,
+     any blocks made for inner levels
+     (since they cannot be recorded as subblocks in that level)
+     must be carried forward so they will later become subblocks
+     of something else.  */
+  else if (subblocks)
+    current_binding_level->blocks
+      = chainon (current_binding_level->blocks, subblocks);
+
+  /* Set the TYPE_CONTEXTs for all of the tagged types belonging to this
+     binding contour so that they point to the appropriate construct, i.e.
+     either to the current FUNCTION_DECL node, or else to the BLOCK node
+     we just constructed.
+
+     Note that for tagged types whose scope is just the formal parameter
+     list for some function type specification, we can't properly set
+     their TYPE_CONTEXTs here, because we don't have a pointer to the
+     appropriate FUNCTION_TYPE node readily available to us.  For those
+     cases, the TYPE_CONTEXTs of the relevant tagged type nodes get set
+     in `grokdeclarator' as soon as we have created the FUNCTION_TYPE
+     node which will represent the "scope" for these "parameter list local"
+     tagged types.
+  */
+
+  if (block)
+    TREE_USED (block) = 1;
+  return block;
+}
+
+void
+maybe_pushlevels (pc)
+     int pc;
+{
+  while (pending_local_decls != NULL_TREE &&
+        DECL_LOCAL_START_PC (pending_local_decls) <= pc)
+    {
+      tree *ptr = &pending_local_decls;
+      tree decl = *ptr;
+      int end_pc = DECL_LOCAL_END_PC (decl);
+
+      while (*ptr != NULL_TREE
+            && DECL_LOCAL_START_PC (*ptr) <= pc
+            && DECL_LOCAL_END_PC (*ptr) == end_pc)
+       ptr = &TREE_CHAIN (*ptr);
+      pending_local_decls = *ptr;
+      *ptr = NULL_TREE;
+
+      /* Force non-nested range to be nested in current range. */
+      if (end_pc > current_binding_level->end_pc)
+       end_pc = current_binding_level->end_pc;
+
+      pushlevel (1);
+      expand_start_bindings (0);
+      current_binding_level->end_pc = end_pc;
+      
+      current_binding_level->names = decl;
+      for ( ; decl != NULL_TREE;  decl = TREE_CHAIN (decl))
+       {
+         push_jvm_slot (DECL_LOCAL_SLOT_NUMBER (decl), decl);
+       }
+    }
+}
+
+void
+maybe_poplevels (pc)
+     int pc;
+{
+  while (current_binding_level->end_pc <= pc)
+    {
+      expand_end_bindings (getdecls (), 1, 0);
+      poplevel (1, 0, 0);
+    }
+}
+
+/* Insert BLOCK at the end of the list of subblocks of the
+   current binding level.  This is used when a BIND_EXPR is expanded,
+   to handle the BLOCK node inside the BIND_EXPR.  */
+
+void
+insert_block (block)
+     tree block;
+{
+  TREE_USED (block) = 1;
+  abort ();
+  current_binding_level->blocks
+    = chainon (current_binding_level->blocks, block);
+}
+
+/* Set the BLOCK node for the innermost scope
+   (the one we are currently in).  */
+
+void
+set_block (block)
+     register tree block;
+{
+  current_binding_level->this_block = block;
+}
+
+/* integrate_decl_tree calls this function. */
+
+void
+copy_lang_decl (node)
+     tree node;
+{
+  int lang_decl_size
+    = TREE_CODE (node) == VAR_DECL ? sizeof (struct lang_decl_var)
+    : sizeof (struct lang_decl);
+  struct lang_decl *x = (struct lang_decl *) oballoc (lang_decl_size);
+  bcopy (DECL_LANG_SPECIFIC (node), x, lang_decl_size);
+  DECL_LANG_SPECIFIC (node) = x;
+}
+
+/* If DECL has a cleanup, build and return that cleanup here.
+   This is a callback called by expand_expr.  */
+
+tree
+maybe_build_cleanup (decl)
+     tree decl;
+{
+  /* There are no cleanups in Java (I think).  */
+  return NULL_TREE;
+}
+
+void
+give_name_to_locals (jcf)
+     JCF *jcf;
+{
+  int i, n = DECL_LOCALVARIABLES_OFFSET (current_function_decl);
+  tree parm;
+  pending_local_decls = NULL_TREE;
+  if (n == 0)
+    return;
+  JCF_SEEK (jcf, n);
+  n = JCF_readu2 (jcf);
+  for (i = 0; i < n; i++)
+    {
+      int start_pc = JCF_readu2 (jcf);
+      int length = JCF_readu2 (jcf);
+      int name_index = JCF_readu2 (jcf);
+      int signature_index = JCF_readu2 (jcf);
+      int slot = JCF_readu2 (jcf);
+      tree name = get_name_constant (jcf, name_index);
+      tree type = promote_type (parse_signature (jcf, signature_index));
+      if (slot < DECL_ARG_SLOT_COUNT (current_function_decl)
+         && start_pc == 0
+         && length == DECL_CODE_LENGTH (current_function_decl))
+       {
+         tree decl = TREE_VEC_ELT (decl_map, slot);
+         DECL_NAME (decl) = name;
+         DECL_ASSEMBLER_NAME (decl) = name;
+         if (TREE_CODE (decl) != PARM_DECL || TREE_TYPE (decl) != type)
+           warning ("bad type in parameter debug info");
+       }
+      else
+       {
+         tree *ptr;
+         int end_pc = start_pc + length;
+         tree decl = build_decl (VAR_DECL, name, type);
+         if (end_pc > DECL_CODE_LENGTH (current_function_decl))
+           {
+             warning_with_decl (decl,
+                        "bad PC range for debug info for local `%s'");
+             end_pc = DECL_CODE_LENGTH (current_function_decl);
+           }
+         DECL_LANG_SPECIFIC (decl)
+           = (struct lang_decl *) permalloc (sizeof (struct lang_decl_var));
+         DECL_LOCAL_SLOT_NUMBER (decl) = slot;
+         DECL_LOCAL_START_PC (decl) = start_pc;
+         DECL_LOCAL_END_PC (decl) = end_pc;
+
+         /* Now insert the new decl in the proper place in
+            pending_local_decls.  We are essentially doing an insertion sort,
+            which works fine, since the list input will normally already
+            be sorted. */
+         ptr = &pending_local_decls;
+         while (*ptr != NULL_TREE
+                && (DECL_LOCAL_START_PC (*ptr) > start_pc
+                    || (DECL_LOCAL_START_PC (*ptr) == start_pc
+                        && DECL_LOCAL_END_PC (*ptr) < end_pc)))
+           ptr = &TREE_CHAIN (*ptr);
+         TREE_CHAIN (decl) = *ptr;
+         *ptr = decl;
+       }
+    }
+
+  pending_local_decls = nreverse (pending_local_decls);
+
+  /* Fill in default names for the parameters. */ 
+  for (parm = DECL_ARGUMENTS (current_function_decl), i = 0;
+       parm != NULL_TREE;  parm = TREE_CHAIN (parm), i++)
+    {
+      if (DECL_NAME (parm) == NULL_TREE)
+       {
+         int arg_i = METHOD_STATIC (current_function_decl) ? i+1 : i;
+         if (arg_i == 0)
+           DECL_NAME (parm) = get_identifier ("this");
+         else
+           {
+             char buffer[12];
+             sprintf (buffer, "ARG_%d", arg_i);
+             DECL_NAME (parm) = get_identifier (buffer);
+           }
+         DECL_ASSEMBLER_NAME (parm) = DECL_NAME (parm);
+       }
+    }
+}
+
+void
+complete_start_java_method (fndecl)
+  tree fndecl;
+{
+
+  DECL_RESULT (fndecl) = build_decl (RESULT_DECL, NULL_TREE,
+                                    TREE_TYPE (TREE_TYPE (fndecl)));
+
+  if (! flag_emit_class_files)
+    {
+      /* Initialize the RTL code for the function.  */
+      init_function_start (fndecl, input_filename, lineno);
+
+      /* Set up parameters and prepare for return, for the function.  */
+      expand_function_start (fndecl, 0);
+    }
+
+  /* Allocate further tree nodes temporarily during compilation
+     of this function only.  */
+  temporary_allocation ();
+
+#if 0
+      /* If this fcn was already referenced via a block-scope `extern' decl (or
+         an implicit decl), propagate certain information about the usage. */
+      if (TREE_ADDRESSABLE (DECL_ASSEMBLER_NAME (current_function_decl)))
+        TREE_ADDRESSABLE (current_function_decl) = 1;
+
+#endif
+
+  if (METHOD_SYNCHRONIZED (fndecl))
+    {
+      /* FIXME: surround the function body by a try/finally set.  */
+    }
+
+  /* Push local variables. Function compiled from source code are
+     using a different local variables management, and for them,
+     pushlevel shouldn't be called from here.  */
+  if (!CLASS_FROM_SOURCE_P (DECL_CONTEXT (fndecl)))
+
+    {
+      pushlevel (2);
+      if (! flag_emit_class_files)
+       expand_start_bindings (1);
+    }
+}
+
+void
+start_java_method (fndecl)
+     tree fndecl;
+{
+  tree tem, *ptr;
+  int i;
+
+  current_function_decl = fndecl;
+  announce_function (fndecl);
+
+  i = DECL_MAX_LOCALS(fndecl) + DECL_MAX_STACK(fndecl);
+  decl_map = make_tree_vec (i);
+  type_map = (tree *) oballoc (i * sizeof (tree));
+
+  pushlevel (1);  /* Push parameters. */
+
+  ptr = &DECL_ARGUMENTS (fndecl);
+  for (tem = TYPE_ARG_TYPES (TREE_TYPE (fndecl)), i = 0;
+       tem != NULL_TREE; tem = TREE_CHAIN (tem), i++)
+    {
+      tree parm_name = NULL_TREE, parm_decl;
+      if (i >= DECL_MAX_LOCALS(fndecl))
+       fatal ("function has more parameters than local slots");
+
+      parm_decl = build_decl (PARM_DECL, parm_name, TREE_VALUE (tem));
+      DECL_CONTEXT (parm_decl) = fndecl;
+      DECL_ARG_TYPE (parm_decl) = TREE_TYPE (parm_decl);
+
+      *ptr = parm_decl;
+      ptr = &TREE_CHAIN (parm_decl);
+
+      /* Add parm_decl to the decl_map. */
+      push_jvm_slot (i, parm_decl);
+
+      type_map[i] = TREE_TYPE (parm_decl);
+      if (TYPE_IS_WIDE (TREE_TYPE (parm_decl)))
+       {
+         i++;
+         type_map[i] = void_type_node;
+       }
+    }
+  *ptr = NULL_TREE;
+  DECL_ARG_SLOT_COUNT (current_function_decl) = i;
+
+  while (i < DECL_MAX_LOCALS(fndecl))
+    type_map[i++] = NULL_TREE;
+
+  complete_start_java_method (fndecl);
+}
+
+void
+end_java_method ()
+{
+  tree fndecl = current_function_decl;
+
+  expand_end_bindings (getdecls (), 1, 0);
+  /* pop out of function */
+  poplevel (1, 1, 0);
+
+  /* pop out of its parameters */
+  poplevel (1, 0, 1);
+
+  BLOCK_SUPERCONTEXT (DECL_INITIAL (fndecl)) = fndecl;
+
+  emit_handlers ();
+
+  /* Generate rtl for function exit.  */
+  expand_function_end (input_filename, lineno, 0);
+
+  /* Run the optimizers and output assembler code for this function. */
+  rest_of_compilation (fndecl);
+
+  current_function_decl = NULL_TREE;
+  permanent_allocation (1);
+}
+
+tree 
+build_decl_no_layout (code, name, type)
+     enum tree_code code;
+     tree name, type;
+{
+  tree decl = build_decl (TYPE_DECL, name, type);
+  TREE_SET_CODE (decl, code);
+  return decl;
+}
diff --git a/gcc/java/except.c b/gcc/java/except.c
new file mode 100644 (file)
index 0000000..46c0f7c
--- /dev/null
@@ -0,0 +1,278 @@
+/* Handle exceptions for GNU compiler for the Java(TM) language.
+   Copyright (C) 1997 Free Software Foundation, Inc.
+
+This file is part of GNU CC.
+
+GNU CC is free software; you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation; either version 2, or (at your option)
+any later version.
+
+GNU CC is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with GNU CC; see the file COPYING.  If not, write to
+the Free Software Foundation, 59 Temple Place - Suite 330,
+Boston, MA 02111-1307, USA.
+
+Java and all Java-based marks are trademarks or registered trademarks
+of Sun Microsystems, Inc. in the United States and other countries.
+The Free Software Foundation is independent of Sun Microsystems, Inc.  */
+
+#include <stdio.h>
+#include "config.h"
+#include "tree.h"
+#include "real.h"
+#include "rtl.h"
+#include "java-tree.h"
+#include "javaop.h"
+#include "java-opcodes.h"
+#include "jcf.h"
+#include "except.h"
+#include "java-except.h"
+#include "eh-common.h"
+
+extern struct obstack permanent_obstack;
+
+struct eh_range *current_method_handlers;
+
+struct eh_range *current_try_block = NULL;
+
+struct eh_range *eh_range_freelist = NULL;
+
+/* These variables are used to speed up find_handler. */
+
+static int cache_range_start, cache_range_end;
+static struct eh_range *cache_range;
+static struct eh_range *cache_next_child;
+
+/* A dummy range that represents the entire method. */
+
+struct eh_range whole_range;
+
+/* Search for the most specific eh_range containing PC.
+   Assume PC is within RANGE.
+   CHILD is a list of children of RANGE such that any
+   previous children have end_pc values that are too low. */
+
+static struct eh_range *
+find_handler_in_range (pc, range, child)
+     int pc;
+     struct eh_range *range;
+     register struct eh_range *child;
+{
+  for (; child != NULL;  child = child->next_sibling)
+    {
+      if (pc < child->start_pc)
+       break;
+      if (pc <= child->end_pc)
+       return find_handler_in_range (pc, child, child->first_child);
+    }
+  cache_range = range;
+  cache_range_start = pc;
+  cache_next_child = child;
+  cache_range_end = child == NULL ? range->end_pc : child->start_pc;
+  return range;
+}
+
+/* Find the inner-most handler that contains PC. */
+
+struct eh_range *
+find_handler (pc)
+     int pc;
+{
+  struct eh_range *h;
+  if (pc >= cache_range_start)
+    {
+      h = cache_range;
+      if (pc < cache_range_end)
+       return h;
+      while (pc >= h->end_pc)
+       {
+         cache_next_child = h->next_sibling;
+         h = h->outer;
+       }
+    }
+  else
+    {
+      h = &whole_range;
+      cache_next_child = h->first_child;
+    }
+  return find_handler_in_range (pc, h, cache_next_child);
+}
+
+#if 0
+first_child;
+next_sibling;
+outer;
+#endif
+
+/* Recursive helper routine for add_handler. */
+
+static int
+link_handler (start_pc, end_pc, handler, type, outer)
+     int start_pc, end_pc;
+     tree handler;
+     tree type;
+     struct eh_range *outer;
+{
+  struct eh_range **ptr;
+  if (start_pc < outer->start_pc || end_pc > outer->end_pc)
+    return 0;  /* invalid or non-nested exception range */
+  if (start_pc == outer->start_pc && end_pc == outer->end_pc)
+    {
+      outer->handlers = tree_cons (type, handler, outer->handlers);
+      return 1;
+    }
+  ptr = &outer->first_child;
+  for (;; ptr = &(*ptr)->next_sibling)
+    {
+      if (*ptr == NULL || end_pc <= (*ptr)->start_pc)
+       {
+         struct eh_range *h = (struct eh_range *)
+           oballoc (sizeof (struct eh_range));
+         h->start_pc = start_pc;
+         h->end_pc = end_pc;
+         h->next_sibling = *ptr;
+         h->first_child = NULL;
+         h->outer = outer;
+         h->handlers = build_tree_list (type, handler);
+         *ptr = h;
+         return 1;
+       }
+      else if (start_pc < (*ptr)->end_pc)
+       return link_handler (start_pc, end_pc, handler, type, *ptr);
+      /* end_pc > (*ptr)->start_pc && start_pc >= (*ptr)->end_pc. */
+    }
+}
+
+/* Called to re-initialize the exception machinery for a new method. */
+
+void
+method_init_exceptions ()
+{
+  whole_range.start_pc = 0;
+  whole_range.end_pc = DECL_CODE_LENGTH (current_function_decl) + 1;
+  whole_range.outer = NULL;
+  whole_range.first_child = NULL;
+  whole_range.next_sibling = NULL;
+  cache_range_start = 0xFFFFFF;
+  set_exception_lang_code (EH_LANG_Java);
+  set_exception_version_code (1);
+}
+
+int
+add_handler (start_pc, end_pc, handler, type)
+     int start_pc, end_pc;
+     tree handler;
+     tree type;
+{
+  return link_handler (start_pc, end_pc, handler, type, &whole_range);
+}
+
+
+/* if there are any handlers for this range, issue start of region */
+void
+expand_start_java_handler (range)
+     struct eh_range *range;
+{
+  expand_eh_region_start ();
+}
+
+/* if there are any handlers for this range, isssue end of range,
+   and then all handler blocks */
+void
+expand_end_java_handler (range)
+     struct eh_range *range;
+{
+  tree handler = range->handlers;
+  expand_start_all_catch ();
+  for ( ; handler != NULL_TREE; handler = TREE_CHAIN (handler))
+    {
+      tree type = TREE_PURPOSE (handler);
+      tree exp;
+      /* The "type" (metch_info) in a (Java) exception table is one:
+       * a) NULL - meaning match any type in a try-finally.
+       * b) a pointer to a (ccmpiled) class (low-order bit 0).
+       * c) a pointer to the Utf8Const name of the class, plus one
+       * (which yields a value with low-order bit 1). */
+      push_obstacks (&permanent_obstack, &permanent_obstack);
+      if (type == NULL_TREE)
+       exp = null_pointer_node;
+      else if (is_compiled_class (type))
+       exp = build_class_ref (type);
+      else
+       exp = fold (build (PLUS_EXPR, ptr_type_node,
+                          build_utf8_ref (build_internal_class_name (type)),
+                          size_one_node));
+      pop_obstacks ();
+      start_catch_handler (exp);
+      expand_goto (TREE_VALUE (handler));
+    }
+  expand_end_all_catch ();
+}
+
+/* Recursive helper routine for maybe_start_handlers. */
+
+static void
+check_start_handlers (range, pc)
+     struct eh_range *range;
+     int pc;
+{
+  if (range != NULL_EH_RANGE && range->start_pc == pc)
+    {
+      check_start_handlers (range->outer, pc);
+      expand_start_java_handler (range);
+    }
+}
+
+struct eh_range *current_range;
+
+/* Emit any start-of-try-range start at PC. */
+
+void
+maybe_start_try (pc)
+     int pc;
+{
+  if (! doing_eh (1))
+    return;
+
+  current_range = find_handler (pc);
+  check_start_handlers (current_range, pc);
+}
+
+/* Emit any end-of-try-range end at PC. */
+
+void
+maybe_end_try (pc)
+     int pc;
+{
+  if (! doing_eh (1))
+    return;
+
+  while (current_range != NULL_EH_RANGE && current_range->end_pc <= pc)
+    {
+      expand_end_java_handler (current_range);
+      current_range = current_range->outer;
+    }
+}
+
+/* Emit the handler labels and their code */
+
+void
+emit_handlers ()
+{
+  if (catch_clauses)
+    {
+      rtx funcend = gen_label_rtx ();
+      emit_jump (funcend);
+
+      emit_insns (catch_clauses);
+      expand_leftover_cleanups ();
+
+      emit_label (funcend);
+    }
+}
diff --git a/gcc/java/expr.c b/gcc/java/expr.c
new file mode 100644 (file)
index 0000000..2c204da
--- /dev/null
@@ -0,0 +1,2283 @@
+/* Process expressions for the GNU compiler for the Java(TM) language.
+   Copyright (C) 1996, 1997, 1998 Free Software Foundation, Inc.
+
+This file is part of GNU CC.
+
+GNU CC is free software; you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation; either version 2, or (at your option)
+any later version.
+
+GNU CC is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with GNU CC; see the file COPYING.  If not, write to
+the Free Software Foundation, 59 Temple Place - Suite 330,
+Boston, MA 02111-1307, USA.  
+
+Java and all Java-based marks are trademarks or registered trademarks
+of Sun Microsystems, Inc. in the United States and other countries.
+The Free Software Foundation is independent of Sun Microsystems, Inc.  */
+
+/* Hacked by Per Bothner <bothner@cygnus.com> February 1996. */
+
+#include <stdio.h>
+#include "config.h"
+#include "tree.h"
+#include "real.h"
+#include "rtl.h"
+#include "expr.h"
+#include "java-tree.h"
+#include "javaop.h"
+#include "java-opcodes.h"
+#include "jcf.h"
+#include "java-except.h"
+#include "parse.h"
+
+static tree operand_type[59];
+extern struct obstack permanent_obstack;
+
+void
+init_expr_processing()
+{
+  operand_type[21] = operand_type[54] = int_type_node;
+  operand_type[22] = operand_type[55] = long_type_node;
+  operand_type[23] = operand_type[56] = float_type_node;
+  operand_type[24] = operand_type[57] = double_type_node;
+  operand_type[25] = operand_type[58] = ptr_type_node;
+}
+
+/* We store the stack state in two places:
+   Within a basic block, we use the quick_stack, which is a
+   pushdown list (TREE_LISTs) of expression nodes.
+   This is the top part of the stack;  below that we use find_stack_slot.
+   At the end of a basic block, the quick_stack must be flushed
+   to the stack slot array (as handled by find_stack_slot).
+   Using quick_stack generates better code (especially when
+   compiled without optimization), because we do not have to
+   explicitly store and load trees to temporary variables.
+
+   If a variable is on the quick stack, it means the value of variable
+   when the quick stack was last flushed.  Conceptually, flush_quick_stack
+   saves all the the quick_stack elements in parellel.  However, that is
+   complicated, so it actually saves them (i.e. copies each stack value
+   to is home virtual register) from low indexes.  This allows a quick_stack
+   element at index i (counting from the bottom of stack the) to references
+   slot virtuals for register that are >= i, but not those that are deeper.
+   This convention makes most operations easier.  For example iadd works
+   even when the stack contains (reg[0], reg[1]):  It results in the
+   stack containing (reg[0]+reg[1]), which is OK.  However, some stack
+   operations are more complicated.  For example dup given a stack
+   containing (reg[0]) would yield (reg[0], reg[0]), which would violate
+   the convention, since stack value 1 would refer to a register with
+   lower index (reg[0]), which flush_quick_stack does not safely handle.
+   So dup cannot just add an extra element to the quick_stack, but iadd can.
+*/
+
+tree quick_stack = NULL_TREE;
+
+/* A free-list of unused permamnet TREE_LIST nodes. */
+tree tree_list_free_list = NULL_TREE;
+
+/* The stack pointer of the Java virtual machine.
+   This does include the size of the quick_stack. */
+
+int stack_pointer;
+
+unsigned char *linenumber_table;
+int linenumber_count;
+
+tree
+truthvalue_conversion (expr)
+     tree expr;
+{
+  /* It is simpler and generates better code to have only TRUTH_*_EXPR
+     or comparison expressions as truth values at this level.
+
+     This function should normally be identity for Java.  */
+
+  switch (TREE_CODE (expr))
+    {
+    case EQ_EXPR:
+    case NE_EXPR: case LE_EXPR: case GE_EXPR: case LT_EXPR: case GT_EXPR:
+    case TRUTH_ANDIF_EXPR:
+    case TRUTH_ORIF_EXPR:
+    case TRUTH_AND_EXPR:
+    case TRUTH_OR_EXPR:
+    case ERROR_MARK:
+      return expr;
+
+    case INTEGER_CST:
+      return integer_zerop (expr) ? boolean_false_node : boolean_true_node;
+
+    case REAL_CST:
+      return real_zerop (expr) ? boolean_false_node : boolean_true_node;
+
+    /* are these legal? XXX JH */
+    case NEGATE_EXPR:
+    case ABS_EXPR:
+    case FLOAT_EXPR:
+    case FFS_EXPR:
+      /* These don't change whether an object is non-zero or zero.  */
+      return truthvalue_conversion (TREE_OPERAND (expr, 0));
+
+    case COND_EXPR:
+      /* Distribute the conversion into the arms of a COND_EXPR.  */
+      return fold (build (COND_EXPR, boolean_type_node, TREE_OPERAND (expr, 0),
+                          truthvalue_conversion (TREE_OPERAND (expr, 1)),
+                          truthvalue_conversion (TREE_OPERAND (expr, 2))));
+
+    case NOP_EXPR:
+      /* If this is widening the argument, we can ignore it.  */
+      if (TYPE_PRECISION (TREE_TYPE (expr))
+          >= TYPE_PRECISION (TREE_TYPE (TREE_OPERAND (expr, 0))))
+        return truthvalue_conversion (TREE_OPERAND (expr, 0));
+      /* fall through to default */
+
+    default:
+      return fold (build (NE_EXPR, boolean_type_node, expr, boolean_false_node));
+    }
+}
+
+#ifdef JAVA_USE_HANDLES
+/* Given a pointer to a handle, get a pointer to an object. */
+
+tree
+unhand_expr (expr)
+     tree expr;
+{
+  tree field, handle_type;
+  expr = build1 (INDIRECT_REF, TREE_TYPE (TREE_TYPE (expr)), expr);
+  handle_type = TREE_TYPE (expr);
+  field = TYPE_FIELDS (handle_type);
+  expr = build (COMPONENT_REF, TREE_TYPE (field), expr, field);
+  return expr;
+}
+#endif
+
+/* Save any stack slots that happen to be in the quick_stack into their
+   home virtual register slots.
+
+   The copy order is from low stack index to high, to support the invariant
+   that the expression for a slot may contain decls for stack slots with
+   higher (or the same) index, but not lower. */
+
+void
+flush_quick_stack ()
+{
+  int stack_index = stack_pointer;
+  register tree prev, cur, next;
+
+  /* First reverse the quick_stack, and count the number of slots it has. */
+  for (cur = quick_stack, prev = NULL_TREE; cur != NULL_TREE; cur = next)
+    {
+      next = TREE_CHAIN (cur);
+      TREE_CHAIN (cur) = prev;
+      prev = cur;
+      stack_index -= 1 + TYPE_IS_WIDE (TREE_TYPE (TREE_VALUE (cur)));
+    }
+  quick_stack = prev;
+
+  while (quick_stack != NULL_TREE)
+    {
+      tree decl;
+      tree node = quick_stack, type;
+      quick_stack = TREE_CHAIN (node);
+      TREE_CHAIN (node) = tree_list_free_list;
+      tree_list_free_list = node;
+      node = TREE_VALUE (node);
+      type = TREE_TYPE (node);
+
+      decl = find_stack_slot (stack_index, type);
+      if (decl != node)
+         expand_assignment (decl, node, 0, 0);
+      stack_index += 1 + TYPE_IS_WIDE (type);
+    }
+}
+
+void
+push_type (type)
+     tree type;
+{
+  int n_words;
+  type = promote_type (type);
+  n_words = 1 + TYPE_IS_WIDE (type);
+  if (stack_pointer + n_words > DECL_MAX_STACK (current_function_decl))
+    fatal ("stack overflow");
+  stack_type_map[stack_pointer++] = type;
+  n_words--;
+  while (--n_words >= 0)
+    stack_type_map[stack_pointer++] = TYPE_SECOND;
+}
+
+void
+push_value (value)
+     tree value;
+{
+  tree type = TREE_TYPE (value);
+  if (TYPE_PRECISION (type) < 32 && INTEGRAL_TYPE_P (type))
+    {
+      type = promote_type (type);
+      value = convert (type, value);
+    }
+  push_type (type);
+  if (tree_list_free_list == NULL_TREE)
+    quick_stack = perm_tree_cons (NULL_TREE, value, quick_stack);
+  else
+    {
+      tree node = tree_list_free_list;
+      tree_list_free_list = TREE_CHAIN (tree_list_free_list);
+      TREE_VALUE (node) = value;
+      TREE_CHAIN (node) = quick_stack;
+      quick_stack = node;
+    }
+}
+
+tree
+pop_type (type)
+     tree type;
+{
+  int n_words;
+  int i;
+  tree t;
+  if (TREE_CODE (type) == RECORD_TYPE)
+    type = promote_type (type);
+  n_words = 1 + TYPE_IS_WIDE (type);
+  if (stack_pointer < n_words)
+    fatal ("stack underflow");
+  while (--n_words > 0)
+    {
+      if (stack_type_map[--stack_pointer] != void_type_node)
+       fatal ("Invalid multi-word value on type stack");
+    }
+  t = stack_type_map[--stack_pointer];
+  if (type == NULL_TREE || t == type)
+    return t;
+  if (INTEGRAL_TYPE_P (type) && INTEGRAL_TYPE_P (t)
+      && TYPE_PRECISION (type) <= 32 && TYPE_PRECISION (t) <= 32)
+      return t;
+  if (TREE_CODE (type) == POINTER_TYPE && TREE_CODE (t) == POINTER_TYPE)
+    {
+      if (type == ptr_type_node || type == object_ptr_type_node)
+       return t;
+      else if (t == ptr_type_node)  /* Special case for null reference. */
+       return type;
+      else if (can_widen_reference_to (t, type))
+       return t;
+    }
+  error ("unexpected type on stack");
+  return t;
+}
+
+/* Return 1f if SOURCE_TYPE can be safely widened to TARGET_TYPE.
+   Handles array types and interfaces.  */
+
+int
+can_widen_reference_to (source_type, target_type)
+     tree source_type, target_type;
+{
+  if (source_type == ptr_type_node || target_type == object_ptr_type_node)
+    return 1;
+
+  /* Get rid of pointers  */
+  if (TREE_CODE (source_type) == POINTER_TYPE)
+    source_type = TREE_TYPE (source_type);
+  if (TREE_CODE (target_type) == POINTER_TYPE)
+    target_type = TREE_TYPE (target_type);
+
+  if (source_type == target_type)
+    return 1;
+  else
+    {
+      source_type = HANDLE_TO_CLASS_TYPE (source_type);
+      target_type = HANDLE_TO_CLASS_TYPE (target_type);
+      if (TYPE_ARRAY_P (source_type) || TYPE_ARRAY_P (target_type))
+       {
+         HOST_WIDE_INT source_length, target_length;
+         if (TYPE_ARRAY_P (source_type) != TYPE_ARRAY_P (target_type))
+           return 0;
+         target_length = java_array_type_length (target_type);
+         if (target_length >= 0)
+           {
+             source_length = java_array_type_length (source_type);
+             if (source_length != target_length)
+               return 0;
+           }
+         source_type = TYPE_ARRAY_ELEMENT (source_type);
+         target_type = TYPE_ARRAY_ELEMENT (target_type);
+         if (source_type == target_type)
+           return 1;
+         if (TREE_CODE (source_type) != POINTER_TYPE
+             || TREE_CODE (target_type) != POINTER_TYPE)
+           return 0;
+         return can_widen_reference_to (source_type, target_type);
+       }
+      else
+       {
+         int source_depth = class_depth (source_type);
+         int target_depth = class_depth (target_type);
+
+         if (CLASS_INTERFACE (TYPE_NAME (target_type)))
+           {
+             /* target_type is OK if source_type or source_type ancestors
+                implement target_type. We handle multiple sub-interfaces  */
+
+             tree basetype_vec = TYPE_BINFO_BASETYPES (source_type);
+             int n = TREE_VEC_LENGTH (basetype_vec), i;
+             for (i=0 ; i < n; i++)
+               if (can_widen_reference_to 
+                   (TREE_TYPE (TREE_VEC_ELT (basetype_vec, i)),
+                    target_type))
+                 return 1;
+               if (n == 0)
+                 return 0;
+           }
+
+         for ( ; source_depth > target_depth;  source_depth--) 
+           {
+             source_type = TYPE_BINFO_BASETYPE (source_type, 0); 
+           }
+         return source_type == target_type;
+       }
+    }
+}
+
+tree
+pop_value (type)
+     tree type;
+{
+  int n_words = 1 + TYPE_IS_WIDE (type);
+  int i;
+  type = pop_type (type);
+  if (quick_stack)
+    {
+      tree node = quick_stack;
+      quick_stack = TREE_CHAIN (quick_stack);
+      TREE_CHAIN (node) = tree_list_free_list;
+      tree_list_free_list = node;
+      node = TREE_VALUE (node);
+      return node;
+    }
+  else
+    return find_stack_slot (stack_pointer, promote_type (type));
+}
+
+
+/* Pop and discrad the top COUNT stack slots. */
+
+void
+java_stack_pop (count)
+     int count;
+{
+  while (count > 0)
+    {
+      tree type, val;
+      if (stack_pointer == 0)
+       fatal ("stack underflow");
+      type = stack_type_map[stack_pointer - 1];
+      if (type == TYPE_SECOND)
+       {
+         count--;
+         if (stack_pointer == 1 || count <= 0)
+           fatal ("stack underflow");
+         type = stack_type_map[stack_pointer - 2];
+       }
+      val = pop_value (type);
+      count--;
+    }
+}
+
+/* Implement the 'swap' operator (to swap two top stack slots). */
+
+void
+java_stack_swap ()
+{
+  tree type1, type2;
+  rtx temp;
+  tree decl1, decl2;
+
+  if (stack_pointer < 2
+      || (type1 = stack_type_map[stack_pointer - 1]) == TYPE_UNKNOWN
+      || (type2 = stack_type_map[stack_pointer - 2]) == TYPE_UNKNOWN
+      || type1 == TYPE_SECOND || type2 == TYPE_SECOND
+      || TYPE_IS_WIDE (type1) || TYPE_IS_WIDE (type2))
+    fatal ("bad stack swap");
+
+  flush_quick_stack ();
+  decl1 = find_stack_slot (stack_pointer - 1, type1);
+  decl2 = find_stack_slot (stack_pointer - 2, type2);
+  temp = copy_to_reg (DECL_RTL (decl1));
+  emit_move_insn (DECL_RTL (decl1), DECL_RTL (decl2));
+  emit_move_insn (DECL_RTL (decl2), temp);
+  stack_type_map[stack_pointer - 1] = type2;
+  stack_type_map[stack_pointer - 2] = type1;
+}
+
+void
+java_stack_dup (size, offset)
+     int size, offset;
+{
+  int low_index = stack_pointer - size - offset;
+  int dst_index;
+  if (low_index < 0)
+    error ("stack underflow - dup* operation");
+
+  flush_quick_stack ();
+
+  stack_pointer += size;
+  dst_index = stack_pointer;
+
+  for (dst_index = stack_pointer;  --dst_index >= low_index; )
+    {
+      tree type;
+      int src_index = dst_index - size;
+      if (src_index < low_index)
+       src_index = dst_index + size + offset;
+      type = stack_type_map [src_index];
+      if (type == TYPE_SECOND)
+       {
+         if (src_index <= low_index)
+           fatal ("dup operation splits 64-bit number");
+         stack_type_map[dst_index] = type;
+         src_index--;  dst_index--;
+         type = stack_type_map[src_index];
+         if (! TYPE_IS_WIDE (type))
+            fatal ("internal error - dup operation");
+       }
+      else if (TYPE_IS_WIDE (type))
+       fatal ("internal error - dup operation");
+      if (src_index != dst_index)
+       {
+         tree src_decl = find_stack_slot (src_index, type);
+         tree dst_decl = find_stack_slot (dst_index, type);
+         emit_move_insn (DECL_RTL (dst_decl), DECL_RTL (src_decl));
+         stack_type_map[dst_index] = type;
+       }
+    }
+}
+
+/* Calls soft_athrow.  Discard the contents of the value stack. */
+
+tree
+build_java_athrow (node)
+    tree node;
+{
+  tree call;
+
+  call = build (CALL_EXPR,
+               void_type_node,
+               build_address_of (throw_node),
+               build_tree_list (NULL_TREE, node),
+               NULL_TREE);
+  TREE_SIDE_EFFECTS (call) = 1;
+  expand_expr_stmt (call);
+  java_stack_pop (stack_pointer);
+}
+
+/* Implementation for jsr/ret */
+
+void
+build_java_jsr (where, ret)
+    tree where;
+    tree ret;
+{
+  tree ret_label = fold (build1 (ADDR_EXPR, return_address_type_node, ret));
+  push_value (ret_label);
+  flush_quick_stack ();
+  expand_goto (where);
+  expand_label (ret);
+}
+
+void
+build_java_ret (location)
+  tree location;
+{
+  expand_computed_goto (location);
+}
+/* Implementation of operations on array: new, load, store, length */
+
+/* Array core info access macros */
+
+#define JAVA_ARRAY_LENGTH_OFFSET(A)                                       \
+  size_binop (CEIL_DIV_EXPR,                                              \
+             (DECL_FIELD_BITPOS                                           \
+                 (TREE_CHAIN (TYPE_FIELDS (TREE_TYPE (TREE_TYPE (A)))))), \
+              size_int (BITS_PER_UNIT))
+
+tree
+decode_newarray_type  (int atype)
+{
+  switch (atype)
+    {
+    case 4:  return boolean_type_node;
+    case 5:  return char_type_node;
+    case 6:  return float_type_node;
+    case 7:  return double_type_node;
+    case 8:  return byte_type_node;
+    case 9:  return short_type_node;
+    case 10: return int_type_node;
+    case 11: return long_type_node;
+    default: return NULL_TREE;
+    }
+}
+
+/* Build a call to soft_badarrayindex(), the ArrayIndexOfBoundsException
+   exception handler.  */
+
+static tree
+build_java_throw_out_of_bounds_exception ()
+{
+  tree node = build (CALL_EXPR, int_type_node,
+                    build_address_of (soft_badarrayindex_node), 
+                    NULL_TREE, NULL_TREE );
+  TREE_SIDE_EFFECTS (node) = 1;        /* Allows expansion within ANDIF */
+  return (node);
+}
+
+/* Return the length of an array. Doesn't perform any checking on the nature
+   or value of the array NODE. May be used to implement some bytecodes.  */
+
+tree
+build_java_array_length_access (node)
+    tree node;
+{
+  tree type = TREE_TYPE (node);
+  HOST_WIDE_INT length;
+  if (!is_array_type_p (type))
+    fatal ("array length on a non-array reference");
+  length = java_array_type_length (type);
+  if (length >= 0)
+    return build_int_2 (length, 0);
+  return fold (build1 (INDIRECT_REF,
+                      int_type_node,
+                      fold (build (PLUS_EXPR, ptr_type_node,
+                                   node, 
+                                   JAVA_ARRAY_LENGTH_OFFSET(node)))));
+}
+
+/* Optionally checks an array against the NULL pointer, eventually throwing a
+   NullPointerException. It could replace signal handling, but tied to NULL.
+   ARG1: the pointer to check, ARG2: the expression to use if
+   the pointer is non-null and ARG3 the type that should be returned.   */
+
+tree
+build_java_arraynull_check (node, expr, type)
+    tree node;                 
+    tree expr;                 
+    tree type;                 
+{
+#if 0
+  static int java_array_access_throws_null_exception = 0;
+  node = ???;
+  if (java_array_access_throws_null_exception)
+      return (build (COND_EXPR, 
+                    type,
+                    build (EQ_EXPR, int_type_node, node, null_pointer_node),
+                    build_java_athrow (node), expr ));
+  else
+#endif
+      return (expr);
+}
+
+static tree
+java_array_data_offset (array)
+     tree array;
+{
+  tree array_type = TREE_TYPE (TREE_TYPE (array));
+  tree data_fld = TREE_CHAIN (TREE_CHAIN (TYPE_FIELDS (array_type)));
+  if (data_fld == NULL_TREE)
+    return size_in_bytes (array_type);
+  else
+    return build_int_2 (TREE_INT_CST_LOW (DECL_FIELD_BITPOS (data_fld))
+                       / BITS_PER_UNIT, 0);
+}
+
+/* Implement array indexing (either as l-value or r-value).
+   Returns a tree for ARRAY[INDEX], assume TYPE is the element type.
+   Optionally performs bounds checking and/or test to NULL.
+   At this point, ARRAY should have been verified as an array.  */
+
+tree
+build_java_arrayaccess (array, type, index)
+    tree array, type, index;
+{
+  tree arith, node, throw = NULL_TREE;
+
+  arith = fold (build (PLUS_EXPR, int_type_node,
+                      java_array_data_offset (array),
+                      fold (build (MULT_EXPR, int_type_node,
+                                   index, size_in_bytes(type)))));
+
+  if (flag_bounds_check)
+    {
+      /* Generate:
+       * (unsigned jint) INDEX >= (unsigned jint) LEN
+       *    && throw ArrayIndexOutOfBoundsException.
+       * Note this is equivalent to and more efficient than:
+       * INDEX < 0 || INDEX >= LEN && throw ... */
+      tree test;
+      tree len = build_java_array_length_access (array);
+      TREE_TYPE (len) = unsigned_int_type_node;
+      test = fold (build (GE_EXPR, boolean_type_node, 
+                              convert (unsigned_int_type_node, index),
+                              len));
+      if (! integer_zerop (test))
+       {
+         throw = build (TRUTH_ANDIF_EXPR, int_type_node, test,
+                        build_java_throw_out_of_bounds_exception ());
+         /* allows expansion within COMPOUND */
+         TREE_SIDE_EFFECTS( throw ) = 1;
+       }
+    }
+
+  node = build1 (INDIRECT_REF, type, 
+                fold (build (PLUS_EXPR, ptr_type_node, 
+                             array, 
+                             (throw ? build (COMPOUND_EXPR, int_type_node, 
+                                             throw, arith )
+                                    : arith))));
+
+  return (fold (build_java_arraynull_check (array, node, type)));
+}
+
+/* Makes sure that INDEXED_TYPE is appropriate. If not, make it from
+   ARRAY_NODE. This function is used to retrieve something less vague than
+   a pointer type when indexing the first dimension of something like [[<t>.
+   May return a corrected type, if necessary, otherwise INDEXED_TYPE is
+   return unchanged.
+   As a side effect, it also makes sure that ARRAY_NODE is an array.  */
+
+static tree
+build_java_check_indexed_type (array_node, indexed_type)
+    tree array_node;
+    tree indexed_type;
+{
+  tree elt_type;
+
+  if (!is_array_type_p (TREE_TYPE (array_node)))
+    fatal ("array indexing on a non-array reference");
+
+  elt_type = (TYPE_ARRAY_ELEMENT (TREE_TYPE (TREE_TYPE (array_node))));
+
+  if (indexed_type == ptr_type_node )
+      return promote_type (elt_type);
+
+  /* BYTE/BOOLEAN store and load are used for both type */
+  if (indexed_type == byte_type_node && elt_type == boolean_type_node )
+    return boolean_type_node;
+
+  if (indexed_type != elt_type )
+    fatal ("type array element mismatch");
+  else
+    return indexed_type;
+}
+
+/* newarray triggers a call to soft_newarray. This function should be called
+   with an integer code (the type of array to create) and get from the stack
+   the size of the dimmension.  */
+
+tree
+build_newarray (atype_value, length)
+     int atype_value;
+     tree length;
+{
+  tree type = build_java_array_type (decode_newarray_type (atype_value),
+                                    TREE_CODE (length) == INTEGER_CST
+                                    ? TREE_INT_CST_LOW (length)
+                                    : -1);
+  return build (CALL_EXPR, promote_type (type),
+               build_address_of (soft_newarray_node),
+               tree_cons (NULL_TREE, 
+                          build_int_2 (atype_value, 0),
+                          build_tree_list (NULL_TREE, length)),
+               NULL_TREE);
+}
+
+/* Generates anewarray from a given CLASS_TYPE. Gets from the stack the size
+   of the dimension. */
+/* Merge with build_newarray.  FIXME. */
+tree
+build_anewarray (class_type, length)
+    tree class_type;
+    tree length;
+{
+  tree type = build_java_array_type (promote_type (class_type),
+                                    TREE_CODE (length) == INTEGER_CST
+                                    ? TREE_INT_CST_LOW (length)
+                                    : -1);
+  return build (CALL_EXPR, promote_type (type),
+               build_address_of (soft_anewarray_node),
+               tree_cons (NULL_TREE, length,
+                          tree_cons (NULL_TREE, build_class_ref (class_type),
+                                     build_tree_list (NULL_TREE,
+                                                      null_pointer_node))),
+               NULL_TREE);
+}
+
+/* Generates a call to multianewarray. multianewarray expects a class pointer,
+   a number of dimensions and the matching number of dimensions. The argument
+   list is NULL terminated.  */
+
+void
+expand_java_multianewarray (class_type, ndim)
+    tree class_type;
+    int  ndim;
+{
+  int i;
+  tree args = build_tree_list( NULL_TREE, null_pointer_node );
+
+  for( i = 0; i < ndim; i++ )
+    args = tree_cons (NULL_TREE, pop_value (int_type_node), args);
+
+  push_value (build (CALL_EXPR,
+                    promote_type (class_type),
+                    build_address_of (soft_multianewarray_node),
+                    tree_cons (NULL_TREE, build_class_ref (class_type),
+                               tree_cons (NULL_TREE, 
+                                          build_int_2 (ndim, 0), args )),
+                    NULL_TREE));
+}
+
+/*  ARRAY[INDEX] <- RHS. build_java_check_indexed_type makes sure that
+    ARRAY is an array type. May expand some bound checking and NULL
+    pointer checking. RHS_TYPE_NODE we are going to store. In the case
+    of the CHAR/BYTE/BOOLEAN SHORT, the type popped of the stack is an
+    INT. In those cases, we make the convertion.
+
+    if ARRAy is a reference type, the assignment is checked at run-time
+    to make sure that the RHS can be assigned to the array element
+    type. It is not necessary to generate this code if ARRAY is final.  */
+
+void
+expand_java_arraystore (rhs_type_node)
+     tree rhs_type_node;
+{
+  tree rhs_node    = pop_value ((INTEGRAL_TYPE_P (rhs_type_node) 
+                                && TYPE_PRECISION (rhs_type_node) <= 32) ? 
+                                int_type_node : rhs_type_node);
+  tree index = pop_value (int_type_node);
+  tree array = pop_value (ptr_type_node);
+
+  rhs_type_node    = build_java_check_indexed_type (array, rhs_type_node);
+
+  flush_quick_stack ();
+
+  index = save_expr (index);
+  array = save_expr (array);
+
+  if (TREE_CODE (rhs_type_node) == POINTER_TYPE
+      && !CLASS_FINAL (TYPE_NAME (TREE_TYPE (rhs_type_node))))
+    {
+      tree check = build (CALL_EXPR, void_type_node,
+                         build_address_of (soft_checkarraystore_node),
+                         tree_cons (NULL_TREE, array,
+                                    build_tree_list (NULL_TREE, rhs_node)),
+                         NULL_TREE);
+      TREE_SIDE_EFFECTS (check) = 1;
+      expand_expr_stmt (check);
+    }
+  
+  expand_assignment (build_java_arrayaccess (array,
+                                            rhs_type_node,
+                                            index),
+                    rhs_node, 0, 0);
+}
+
+/* Expand the evaluation of ARRAY[INDEX]. build_java_check_indexed_type makes 
+   sure that LHS is an array type. May expand some bound checking and NULL
+   pointer checking.  
+   LHS_TYPE_NODE is the type of ARRAY[INDEX]. But in the case of CHAR/BYTE/
+   BOOLEAN/SHORT, we push a promoted type back to the stack.
+*/
+
+void
+expand_java_arrayload (lhs_type_node )
+    tree lhs_type_node;
+{
+  tree load_node;
+  int convert;
+  tree index_node = pop_value (int_type_node);
+  tree array_node = pop_value (ptr_type_node);
+
+  index_node = save_expr (index_node);
+  array_node = save_expr (array_node);
+  lhs_type_node   = build_java_check_indexed_type (array_node, lhs_type_node);
+
+  load_node = build_java_arrayaccess (array_node,
+                                     lhs_type_node,
+                                     index_node);
+
+  if (INTEGRAL_TYPE_P (lhs_type_node) && TYPE_PRECISION (lhs_type_node) <= 32)
+    load_node = fold (build1 (NOP_EXPR, int_type_node, load_node));
+  push_value (load_node);
+}
+
+/* Expands .length. Makes sure that we deal with and array and may expand
+   a NULL check on the array object.  */
+
+void
+expand_java_array_length ()
+{
+  tree array  = pop_value (ptr_type_node);
+  tree length = build_java_array_length_access (array);
+
+  push_value (build_java_arraynull_check (array, length, int_type_node));
+}
+
+/* Emit code for the call to soft_monitor{enter,exit}. CALL can be either
+   soft_monitorenter_node or soft_monitorexit_node.  */
+
+tree
+build_java_monitor (call, object)
+    tree call;
+    tree object;
+{
+  return (build (CALL_EXPR,
+                void_type_node,
+                build_address_of (call),
+                build_tree_list (NULL_TREE, object),
+                NULL_TREE));
+}
+
+/* Emit code for one of the PUSHC instructions. */
+
+void
+expand_java_pushc (ival, type)
+     int ival;
+     tree type;
+{
+  tree value;
+  if (type == ptr_type_node && ival == 0)
+    value = null_pointer_node;
+  else if (type == int_type_node || type == long_type_node)
+    {
+      value = build_int_2 (ival, ival < 0 ? -1 : 0);
+      TREE_TYPE (value) = type;
+    }
+  else if (type == float_type_node || type == double_type_node)
+    {
+      REAL_VALUE_TYPE x;
+#ifdef REAL_ARITHMETIC
+      REAL_VALUE_FROM_INT (x, ival, 0, TYPE_MODE (type));
+#else
+      x = ival;
+#endif
+      value = build_real (type, x);
+    }
+  else
+    fatal ("internal error in expand_java_pushc");
+  push_value (value);
+}
+
+void
+expand_java_return (type)
+     tree type;
+{
+  if (type == void_type_node)
+    expand_null_return ();
+  else
+    {
+      tree retval = pop_value (type);
+      tree res = DECL_RESULT (current_function_decl);
+      retval = build (MODIFY_EXPR, TREE_TYPE (res), res, retval);
+      TREE_SIDE_EFFECTS (retval) = 1;
+      expand_return (retval);
+    }
+}
+
+tree
+build_address_of (value)
+     tree value;
+{
+  return build1 (ADDR_EXPR, build_pointer_type (TREE_TYPE (value)), value);
+}
+
+void
+expand_java_NEW (type)
+     tree type;
+{
+  if (! CLASS_LOADED_P (type))
+    load_class (type, 1);
+  push_value (build (CALL_EXPR, promote_type (type),
+                    build_address_of (alloc_object_node),
+                    tree_cons (NULL_TREE, build_class_ref (type),
+                               build_tree_list (NULL_TREE,
+                                                size_in_bytes (type))),
+                    NULL_TREE));
+}
+
+void
+expand_java_INSTANCEOF (type)
+     tree type;
+{
+  tree value = pop_value (object_ptr_type_node);
+  value = build (CALL_EXPR, TREE_TYPE (TREE_TYPE (soft_instanceof_node)),
+                build_address_of (soft_instanceof_node),
+                tree_cons (NULL_TREE, value,
+                           build_tree_list (NULL_TREE,
+                                            build_class_ref (type))),
+                NULL_TREE);
+  push_value (value);
+}
+
+void
+expand_java_CHECKCAST (type)
+     tree type;
+{
+  tree value = pop_value (ptr_type_node);
+  value = build (CALL_EXPR, promote_type (type),
+                build_address_of (soft_checkcast_node),
+                tree_cons (NULL_TREE, build_class_ref (type),
+                           build_tree_list (NULL_TREE, value)),
+                NULL_TREE);
+  push_value (value);
+}
+
+void
+expand_iinc (unsigned int local_var_index, int ival, int pc)
+{
+    tree local_var, res;
+    tree constant_value;
+
+    flush_quick_stack ();
+    local_var = find_local_variable (local_var_index, int_type_node, pc);
+    constant_value = build_int_2 (ival, ival < 0 ? -1 : 0);
+    res = fold (build (PLUS_EXPR, int_type_node, local_var, constant_value));
+    expand_assignment (local_var, res, 0, 0);
+}
+
+tree
+build_java_binop (op, type, arg1, arg2)
+     enum tree_code op;
+     tree type, arg1, arg2;
+{
+  tree mask;
+  switch (op)
+    {
+    case URSHIFT_EXPR:
+      {
+       tree u_type = unsigned_type (type);
+       arg1 = convert (u_type, arg1);
+       arg1 = build_java_binop (RSHIFT_EXPR, u_type, arg1, arg2);
+       return convert (type, arg1);
+      }
+    case LSHIFT_EXPR:
+    case RSHIFT_EXPR:
+      mask = build_int_2 (TYPE_PRECISION (TREE_TYPE (arg1)) - 1, 0);
+      arg2 = fold (build (BIT_AND_EXPR, int_type_node, arg2, mask));
+      break;
+
+    case COMPARE_L_EXPR:  /* arg1 > arg2 ?  1 : arg1 == arg2 ? 0 : -1 */
+    case COMPARE_G_EXPR:  /* arg1 < arg2 ? -1 : arg1 == arg2 ? 0 :  1 */
+      arg1 = save_expr (arg1);  arg2 = save_expr (arg2);
+      {
+       tree ifexp1 = fold ( build (op == COMPARE_L_EXPR ? GT_EXPR : LT_EXPR,
+                                   boolean_type_node, arg1, arg2));
+       tree ifexp2 = fold ( build (EQ_EXPR, boolean_type_node, arg1, arg2));
+       tree second_compare = fold (build (COND_EXPR, int_type_node,
+                                          ifexp2, integer_zero_node,
+                                          op == COMPARE_L_EXPR
+                                          ? integer_negative_one_node
+                                          : integer_one_node));
+       return fold (build (COND_EXPR, int_type_node, ifexp1,
+                           op == COMPARE_L_EXPR ? integer_one_node
+                           : integer_negative_one_node,
+                           second_compare));
+      }
+    case COMPARE_EXPR:
+      arg1 = save_expr (arg1);  arg2 = save_expr (arg2);
+      {
+       tree ifexp1 = fold ( build (LT_EXPR, boolean_type_node, arg1, arg2));
+       tree ifexp2 = fold ( build (GT_EXPR, boolean_type_node, arg1, arg2));
+       tree second_compare = fold ( build (COND_EXPR, int_type_node,
+                                           ifexp2, integer_one_node,
+                                           integer_zero_node));
+       return fold (build (COND_EXPR, int_type_node,
+                           ifexp1, integer_negative_one_node, second_compare));
+      }
+
+    case TRUNC_MOD_EXPR:
+      if (TREE_CODE (type) == REAL_TYPE)
+       {
+         tree call;
+         if (type != double_type_node)
+           {
+             arg1 = convert (double_type_node, arg1);
+             arg2 = convert (double_type_node, arg2);
+           }
+         call = build (CALL_EXPR, double_type_node,
+                       build_address_of (soft_fmod_node),
+                       tree_cons (NULL_TREE, arg1,
+                                  build_tree_list (NULL_TREE, arg2)),
+                       NULL_TREE);
+         if (type != double_type_node)
+           call = convert (type, call);
+         return call;
+       }
+      break;
+
+#if 0  /* not required */
+    case PLUS_EXPR:
+    case MULT_EXPR:
+    case MINUS_EXPR:
+    case TRUNC_DIV_EXPR:
+    case RDIV_EXPR:
+/*    case REM_EXPR: */
+    case BIT_AND_EXPR:
+    case BIT_IOR_EXPR:
+    case BIT_XOR_EXPR:
+      break;
+    default:
+      error ("unknown opcode");
+      return error_mark_node;
+#endif
+
+    }
+  return fold (build (op, type, arg1, arg2));
+}
+
+void
+expand_java_binop (type, op)
+     tree type;  enum tree_code op;
+{
+  tree larg, rarg;
+  tree ltype = type;
+  tree rtype = type;
+  switch (op)
+    {
+    case LSHIFT_EXPR:
+    case RSHIFT_EXPR:
+    case URSHIFT_EXPR:
+      rtype = int_type_node;
+      rarg = pop_value (rtype);
+      break;
+    default:
+      rarg = pop_value (rtype);
+    }
+  larg = pop_value (ltype);
+  push_value (build_java_binop (op, type, larg, rarg));
+}
+
+/* Lookup the field named NAME in *TYPEP or its super classes.
+   If not found, return NULL_TREE.
+   (If the *TYPEP is not found, return error_mark_node.)
+   If found, return the FIELD_DECL, and set *TYPEP to the
+   class containing the field. */
+
+tree
+lookup_field (typep, name)
+     tree *typep;
+     tree name;
+{
+  if (CLASS_P (*typep) && !CLASS_LOADED_P (*typep))
+    {
+      load_class (*typep, 1);
+      if (TREE_CODE (TYPE_SIZE (*typep)) == ERROR_MARK)
+       return error_mark_node;
+    }
+  do
+    {
+      tree field;
+      for (field = TYPE_FIELDS (*typep);  field;  field = TREE_CHAIN (field))
+       {
+         if (DECL_NAME (field) == name)
+           return field;
+       }
+      *typep = CLASSTYPE_SUPER (*typep);
+    } while (*typep);
+  return NULL_TREE;
+}
+
+/* Look up the field named NAME in object SELF_VALUE,
+   which has class SELF_CLASS (a non-handle RECORD_TYPE).
+   SELF_VALUE is NULL_TREE if looking for a static field. */
+
+tree
+build_field_ref (self_value, self_class, name)
+     tree self_value, self_class, name;
+{
+  tree base_class = self_class;
+  tree field_decl = lookup_field (&base_class, name);
+  if (field_decl == NULL_TREE)
+    {
+      error ("field `%s' not found", IDENTIFIER_POINTER (name));
+      return error_mark_node;
+    }
+  if (self_value == NULL_TREE)
+    {
+      return build_static_field_ref (field_decl);
+    }
+  else
+    {
+      tree base_handle_type = promote_type (base_class);
+      if (base_handle_type != TREE_TYPE (self_value))
+       self_value = fold (build1 (NOP_EXPR, base_handle_type, self_value));
+#ifdef JAVA_USE_HANDLES
+      self_value = unhand_expr (self_value);
+#endif
+      self_value = build1 (INDIRECT_REF, TREE_TYPE (TREE_TYPE (self_value)),
+                          self_value);
+      return fold (build (COMPONENT_REF, TREE_TYPE (field_decl),
+                         self_value, field_decl));
+    }
+}
+
+tree
+lookup_label (pc)
+     int pc;
+{
+  tree name;
+  char buf[20];
+  sprintf (buf, "LJpc=%d", pc);
+  name = get_identifier (buf);
+  if (IDENTIFIER_LOCAL_VALUE (name))
+    return IDENTIFIER_LOCAL_VALUE (name);
+  else
+    {
+      /* The type of the address of a label is return_address_type_node. */
+      tree decl = create_label_decl (name);
+      LABEL_PC (decl) = pc;
+      label_rtx (decl);
+      return pushdecl (decl);
+    }
+}
+
+tree
+create_label_decl (name)
+     tree name;
+{
+  tree decl;
+  push_obstacks (&permanent_obstack, &permanent_obstack);
+  decl = build_decl (LABEL_DECL, name, 
+                    TREE_TYPE (return_address_type_node));
+  pop_obstacks ();
+  DECL_CONTEXT (decl) = current_function_decl;
+  DECL_IGNORED_P (decl) = 1;
+  return decl;
+}
+
+/* This maps a bytecode offset (PC) to various flags. */
+char *instruction_bits;
+
+void
+note_label (current_pc, target_pc)
+     int current_pc, target_pc;
+{
+  lookup_label (target_pc);
+  instruction_bits [target_pc] |= BCODE_JUMP_TARGET;
+}
+
+/* Emit code to jump to TARGET_PC if VALUE1 CONDITION VALUE2,
+   where CONDITION is one of one the compare operators. */
+
+
+void
+expand_compare (condition, value1, value2, target_pc)
+     enum tree_code condition;
+     tree value1, value2;
+     int target_pc;
+{
+  tree target = lookup_label (target_pc);
+  tree cond = fold (build (condition, boolean_type_node, value1, value2));
+  expand_start_cond (truthvalue_conversion (cond), 0);
+  expand_goto (target);
+  expand_end_cond ();
+}
+
+/* Emit code for a TEST-type opcode. */
+
+void
+expand_test (condition, type, target_pc)
+     enum tree_code condition;
+     tree type;
+     int target_pc;
+{
+  tree value1, value2;
+  flush_quick_stack ();
+  value1 = pop_value (type);
+  value2 = (type == ptr_type_node) ? null_pointer_node : integer_zero_node;
+  expand_compare (condition, value1, value2, target_pc);
+}
+
+/* Emit code for a COND-type opcode. */
+
+void
+expand_cond (condition, type, target_pc)
+     enum tree_code condition;
+     tree type;
+     int target_pc;
+{
+  tree value1, value2;
+  flush_quick_stack ();
+  /* note: pop values in opposite order */
+  value2 = pop_value (type);
+  value1 = pop_value (type);
+  /* Maybe should check value1 and value2 for type compatibility ??? */
+  expand_compare (condition, value1, value2, target_pc);
+}
+
+void
+expand_java_goto (target_pc)
+     int target_pc;
+{
+  tree target_label = lookup_label (target_pc);
+  flush_quick_stack ();
+  expand_goto (target_label);
+}
+
+void
+expand_java_call (target_pc, return_address)
+     int target_pc, return_address;
+{
+  tree target_label = lookup_label (target_pc);
+  tree value = build_int_2 (return_address, return_address < 0 ? -1 : 0);
+  push_value (value);
+  flush_quick_stack ();
+  expand_goto (target_label);
+}
+
+void
+expand_java_ret (return_address)
+     tree return_address;
+{
+  warning ("ret instruction not implemented");
+#if 0
+  tree target_label = lookup_label (target_pc);
+  flush_quick_stack ();
+  expand_goto (target_label);
+#endif
+}
+
+/* Recursive helper function to pop argument types during verifiation. */
+
+void
+pop_argument_types (arg_types)
+     tree arg_types;
+{
+  if (arg_types == NULL_TREE)
+    return;
+  if (TREE_CODE (arg_types) == TREE_LIST)
+    {
+      pop_argument_types (TREE_CHAIN (arg_types));
+      pop_type (TREE_VALUE (arg_types));
+      return;
+    }
+  abort ();
+}
+
+tree
+pop_arguments (arg_types)
+     tree arg_types;
+{
+  if (arg_types == NULL_TREE)
+    return NULL_TREE;
+  if (TREE_CODE (arg_types) == TREE_LIST)
+    {
+      tree tail = pop_arguments (TREE_CHAIN (arg_types));
+      return tree_cons (NULL_TREE, pop_value (TREE_VALUE (arg_types)), tail);
+    }
+  abort ();
+}
+
+/* Build an expression to initialize the class CLAS.
+   if EXPR is non-NULL, returns an expression to first call the initializer
+   (if it is needed) and then calls EXPR. */
+
+tree
+build_class_init (clas, expr)
+     tree clas, expr;
+{
+  tree init;
+  if (inherits_from_p (current_class, clas))
+    return expr;
+  init = build (CALL_EXPR, void_type_node,
+               build_address_of (soft_initclass_node),
+               build_tree_list (NULL_TREE, build_class_ref (clas)),
+               NULL_TREE);
+  TREE_SIDE_EFFECTS (init) = 1;
+  if (expr != NULL_TREE)
+    {
+      expr = build (COMPOUND_EXPR, TREE_TYPE (expr), init, expr);
+      TREE_SIDE_EFFECTS (expr) = 1;
+      return expr;
+    }
+  return init;
+}
+
+static tree methods_ident = NULL_TREE;
+static tree ncode_ident = NULL_TREE;
+tree dtable_ident = NULL_TREE;
+
+tree
+build_known_method_ref (method, method_type, self_type, method_signature, arg_list)
+     tree method, method_type, self_type, method_signature, arg_list;
+{
+  tree func;
+  if (flag_emit_class_files)
+    return method;
+  else if (is_compiled_class (self_type))
+    {
+      make_decl_rtl (method, NULL, 1);
+      func = build1 (ADDR_EXPR, method_ptr_type_node, method);
+    }
+  else
+    {
+      /* We don't know whether the method has been (statically) compiled.
+        Compile this code to get a reference to the method's code:
+        
+        SELF_TYPE->methods[METHOD_INDEX].ncode
+        
+        This is guaranteed to work (assuming SELF_TYPE has
+        been initialized), since if the method is not compiled yet,
+        its ncode points to a trampoline that forces compilation. */
+      
+      int method_index = 0;
+      tree meth;
+      tree ref = build_class_ref (self_type);
+      ref = build1 (INDIRECT_REF, class_type_node, ref);
+      if (ncode_ident == NULL_TREE)
+       ncode_ident = get_identifier ("ncode");
+      if (methods_ident == NULL_TREE)
+       methods_ident = get_identifier ("methods");
+      ref = build (COMPONENT_REF, method_ptr_type_node, ref,
+                  lookup_field (&class_type_node, methods_ident));
+      for (meth = TYPE_METHODS (CLASS_TO_HANDLE_TYPE (self_type));
+          ; meth = TREE_CHAIN (meth))
+       {
+         if (method == meth)
+           break;
+         if (meth == NULL_TREE)
+           fatal ("method '%s' not found in class",
+                  IDENTIFIER_POINTER (DECL_NAME (method)));
+         method_index++;
+       }
+      method_index *= int_size_in_bytes (method_type_node);
+      ref = fold (build (PLUS_EXPR, method_ptr_type_node,
+                        ref, build_int_2 (method_index, 0)));
+      ref = build1 (INDIRECT_REF, method_type_node, ref);
+      func = build (COMPONENT_REF, nativecode_ptr_type_node,
+                   ref,
+                   lookup_field (&method_type_node, ncode_ident));
+    }
+  return func;
+}
+
+tree
+invoke_build_dtable (is_invoke_interface, arg_list)
+     int is_invoke_interface;
+     tree arg_list;
+{
+  tree dtable, objectref;
+
+  TREE_VALUE (arg_list) = save_expr (TREE_VALUE (arg_list));
+
+  /* If we're dealing with interfaces and if the objectref
+     argument is an array then get the dispatch table of the class
+     Object rather than the one from the objectref.  */
+  objectref = (is_invoke_interface 
+              && is_array_type_p (TREE_TYPE (TREE_VALUE (arg_list))) ?
+              object_type_node : TREE_VALUE (arg_list));
+  
+  if (dtable_ident == NULL_TREE)
+    dtable_ident = get_identifier ("dtable");
+  dtable = build1 (INDIRECT_REF, object_type_node, objectref );
+  dtable = build (COMPONENT_REF, dtable_ptr_type, dtable,
+                 lookup_field (&object_type_node, dtable_ident));
+
+  return dtable;
+}
+
+tree 
+build_invokevirtual (dtable, method)
+     tree dtable, method;
+{
+  tree func;
+  tree nativecode_ptr_ptr_type_node
+    = build_pointer_type (nativecode_ptr_type_node);
+  int method_index = TREE_INT_CST_LOW (DECL_VINDEX (method));
+  /* Add one to skip "class" field of dtable, and one to skip unused
+     vtable entry (for C++ compatibility). */
+  method_index += 2;
+  method_index
+    *= int_size_in_bytes (nativecode_ptr_ptr_type_node);
+  func = fold (build (PLUS_EXPR, nativecode_ptr_ptr_type_node,
+                     dtable, build_int_2 (method_index, 0)));
+  func = build1 (INDIRECT_REF, nativecode_ptr_type_node, func);
+
+  return func;
+}
+
+/* Expand one of the invoke_* opcodes.
+   OCPODE is the specific opcode.
+   METHOD_REF_INDEX is an index into the constant pool.
+   NARGS is the number of arguments, or -1 if not specified. */
+
+void
+expand_invoke (opcode, method_ref_index, nargs)
+     int opcode;
+     int method_ref_index;
+     int nargs;
+{
+  tree method_signature = COMPONENT_REF_SIGNATURE(&current_jcf->cpool, method_ref_index);
+  tree method_name = COMPONENT_REF_NAME (&current_jcf->cpool, method_ref_index);
+  tree self_type = get_class_constant
+    (current_jcf, COMPONENT_REF_CLASS_INDEX(&current_jcf->cpool, method_ref_index));
+  char *self_name = IDENTIFIER_POINTER (DECL_NAME (TYPE_NAME (self_type)));
+  tree call, func, method, arg_list, method_type;
+
+  static tree class_ident = NULL_TREE;
+
+  if (! CLASS_LOADED_P (self_type))
+    {
+      load_class (self_type, 1);
+      if (TREE_CODE (TYPE_SIZE (self_type)) == ERROR_MARK)
+       fatal ("failed to find class '%s'", self_name);
+    }
+
+  if (method_name == init_identifier_node)
+    method = lookup_java_constructor (CLASS_TO_HANDLE_TYPE (self_type),
+                                     method_signature);
+  else
+    method = lookup_java_method (CLASS_TO_HANDLE_TYPE (self_type),
+                                method_name, method_signature);
+  if (method == NULL_TREE)
+    {
+      error ("Class '%s' has no method named '%s' matching signature '%s'",
+            self_name,
+            IDENTIFIER_POINTER (method_name),
+            IDENTIFIER_POINTER (method_signature));
+    }
+  /* Invoke static can't invoke static/abstract method */
+  else if (opcode == OPCODE_invokestatic)
+    {
+      if (!METHOD_STATIC (method))
+       {
+         error ("invokestatic on non static method");
+         method = NULL_TREE;
+       }
+      else if (METHOD_ABSTRACT (method))
+       {
+         error ("invokestatic on abstract method");
+         method = NULL_TREE;
+       }
+    }
+  else
+    {
+      if (METHOD_STATIC (method))
+       {
+         error ("invoke[non-static] on static method");
+         method = NULL_TREE;
+       }
+    }
+
+  if (method == NULL_TREE)
+    {
+      method_type = get_type_from_signature (method_signature);
+      pop_arguments (TYPE_ARG_TYPES (method_type));
+      if (opcode != OPCODE_invokestatic) 
+       pop_type (self_type);
+      method_type = promote_type (TREE_TYPE (method_type));
+      push_value (convert (method_type, integer_zero_node));
+      return;
+    }
+
+  method_type = TREE_TYPE (method);
+  arg_list = pop_arguments (TYPE_ARG_TYPES (method_type));
+  flush_quick_stack ();
+
+  if (opcode == OPCODE_invokestatic || opcode == OPCODE_invokespecial
+      && ! inherits_from_p (current_class, self_type))
+    { /* FIXME probably not needed for invokespecial if done by NEW. */
+      /* Ensure self_type is initialized. */ 
+      func = build (CALL_EXPR, void_type_node, soft_initclass_node,
+                   build_tree_list (NULL_TREE,
+                                    build_class_ref (self_type)),
+                   NULL_TREE);
+      expand_expr_stmt (func);
+    }
+
+  func = NULL_TREE;
+  if (opcode == OPCODE_invokestatic || opcode == OPCODE_invokespecial
+      || (opcode == OPCODE_invokevirtual
+         && (METHOD_FINAL (method) || CLASS_FINAL (TYPE_NAME (self_type)))))
+    func = build_known_method_ref (method, method_type, self_type,
+                                  method_signature, arg_list);
+  else
+    {
+      tree dtable = invoke_build_dtable (opcode == OPCODE_invokeinterface, 
+                                        arg_list);
+      if (opcode == OPCODE_invokevirtual)
+       func = build_invokevirtual (dtable, method);
+      else
+       {
+         /* We expand invokeinterface here. soft_lookupinterfacemethod () will
+            ensure that the selected method exists, is public and not abstract
+            nor static.  */
+           
+         tree lookup_arg;
+
+         if (class_ident == NULL_TREE)
+           class_ident = get_identifier ("class");
+
+         dtable = build1 (INDIRECT_REF, dtable_type, dtable);
+         dtable = build (COMPONENT_REF, class_ptr_type, dtable,
+                         lookup_field (&dtable_type, class_ident));
+         lookup_arg = build_tree_list (NULL_TREE,
+                                       build_utf8_ref (method_signature));
+         lookup_arg = tree_cons (NULL_TREE, dtable,
+                                 tree_cons (NULL_TREE,
+                                            build_utf8_ref (method_name),
+                                            lookup_arg));
+         func = build (CALL_EXPR,
+                       ptr_type_node,
+                       build_address_of (soft_lookupinterfacemethod_node),
+                       lookup_arg, NULL_TREE);
+       }
+    }
+  func = build1 (NOP_EXPR, build_pointer_type (method_type), func);
+  call = build (CALL_EXPR, TREE_TYPE (method_type), func, arg_list, NULL_TREE);
+  TREE_SIDE_EFFECTS (call) = 1;
+
+  if (opcode == OPCODE_invokestatic || opcode == OPCODE_invokespecial)
+    { /* FIXME probably not needed for invokespecial if done by NEW. */
+      /* Ensure self_type is initialized. */ 
+      call = build_class_init (self_type, call);
+    }
+
+  if (TREE_CODE (TREE_TYPE (method_type)) == VOID_TYPE)
+    expand_expr_stmt (call);
+  else
+    {
+      push_value (call);
+      flush_quick_stack ();
+    }
+}
+
+
+/* Expand an operation to extract from or store into a field.
+   IS_STATIC is 1 iff the field is static.
+   IS_PUTTING is 1 for putting into a field;  0 for getting from the field.
+   FIELD_REF_INDEX is an index into the constant pool.  */
+
+void
+expand_java_field_op (is_static, is_putting, field_ref_index)
+     int is_static;
+     int is_putting;
+     int field_ref_index;
+{
+  tree self_type = get_class_constant
+    (current_jcf, COMPONENT_REF_CLASS_INDEX (&current_jcf->cpool, field_ref_index));
+  char *self_name = IDENTIFIER_POINTER (DECL_NAME (TYPE_NAME (self_type)));
+  tree field_name = COMPONENT_REF_NAME (&current_jcf->cpool, field_ref_index);
+  tree field_signature = COMPONENT_REF_SIGNATURE (&current_jcf->cpool, field_ref_index);
+  tree field_type = get_type_from_signature (field_signature);
+  tree new_value = is_putting ? pop_value (field_type) : NULL_TREE;
+  tree field_ref;
+  int is_error = 0;
+  tree field_decl = lookup_field (&self_type, field_name);
+  if (field_decl == error_mark_node)
+    {
+      is_error = 1;
+    }
+  else if (field_decl == NULL_TREE)
+    {
+      error ("Missing field '%s' in '%s'",
+            IDENTIFIER_POINTER (field_name), self_name);
+      is_error = 1;
+    }
+  else if (build_java_signature (TREE_TYPE (field_decl)) != field_signature)
+    {
+      error ("Mismatching signature for field '%s' in '%s'",
+            IDENTIFIER_POINTER (field_name), self_name);
+      is_error = 1;
+    }
+  field_ref = is_static ? NULL_TREE : pop_value (self_type);
+  if (is_error)
+    {
+      if (! is_putting)
+       push_value (convert (promote_type (field_type), integer_zero_node));
+      flush_quick_stack ();
+      return;
+    }
+
+  /* Inline references to java.lang.PRIMTYPE.TYPE.
+     In addition to being a useful (minor) optimization,
+     this is also needed to avoid circularities in the implementation
+     of these fields in libjava. */
+  if (field_name == TYPE_identifier_node && ! is_putting
+      && field_type == class_type_node
+      && strncmp (self_name, "java.lang.", 10) == 0)
+    {
+      char *class_name = self_name+10;
+      tree typ;
+      if (strcmp(class_name, "Byte") == 0)
+       typ = byte_type_node;
+      else if (strcmp(class_name, "Short") == 0)
+       typ = short_type_node;
+      else if (strcmp(class_name, "Integer") == 0)
+       typ = int_type_node;
+      else if (strcmp(class_name, "Long") == 0)
+       typ = long_type_node;
+      else if (strcmp(class_name, "Float") == 0)
+       typ = float_type_node;
+      else if (strcmp(class_name, "Boolean") == 0)
+       typ = boolean_type_node;
+      else if (strcmp(class_name, "Char") == 0)
+       typ = char_type_node;
+      else if (strcmp(class_name, "Void") == 0)
+       typ = void_type_node;
+      else
+       typ = NULL_TREE;
+      if (typ != NULL_TREE)
+       {
+         push_value (build_class_ref (typ));
+         return;
+       }
+    }
+
+  field_ref = build_field_ref (field_ref, self_type, field_name);
+  if (is_static)
+    field_ref = build_class_init (self_type, field_ref);
+  if (is_putting)
+    {
+      flush_quick_stack ();
+      if (FIELD_FINAL (field_decl))
+       {
+         if (DECL_CONTEXT (field_decl) != current_class)
+           error_with_decl (field_decl,
+                    "assignment to final field `%s' not in field's class");
+         else if (FIELD_STATIC (field_decl))
+           {
+             if (DECL_NAME (current_function_decl) != clinit_identifier_node)
+               error_with_decl (field_decl, 
+             "assignment to final static field `%s' not in class initializer");
+           }
+         else
+           {
+             if (! DECL_CONSTRUCTOR_P (current_function_decl))
+               error_with_decl (field_decl, 
+                           "assignment to final field `%s' not in constructor");
+           }
+       }
+      expand_assignment (field_ref, new_value, 0, 0);
+    }
+  else
+    push_value (field_ref);
+}
+
+void
+load_type_state (label)
+     tree label;
+{
+  int i;
+  tree vec = LABEL_TYPE_STATE (label);
+  int cur_length = TREE_VEC_LENGTH (vec);
+  stack_pointer = cur_length - DECL_MAX_LOCALS(current_function_decl);
+  for (i = 0; i < cur_length; i++)
+    type_map [i] = TREE_VEC_ELT (vec, i);
+}
+
+struct rtx_def *
+java_lang_expand_expr (exp, target, tmode, modifier)
+     register tree exp;
+     rtx target;
+     enum machine_mode tmode;
+     enum expand_modifier modifier;
+{
+  register rtx op0;
+  tree type = TREE_TYPE (exp);
+  register enum machine_mode mode = TYPE_MODE (type);
+  int unsignedp = TREE_UNSIGNED (type);
+
+  switch (TREE_CODE (exp))
+    {
+    case BLOCK:
+      if (BLOCK_EXPR_BODY (exp))
+       {
+         tree local;
+         struct rtx_def *to_return;
+         pushlevel (2);        /* 2 and above */
+         expand_start_bindings (0);
+         local = BLOCK_EXPR_DECLS (exp);
+         while (local)
+           {
+             tree next = TREE_CHAIN (local);
+             layout_decl (local, 0);
+             expand_decl (pushdecl (local));
+             local = next;
+           }
+         to_return =
+           expand_expr (BLOCK_EXPR_BODY (exp), target, tmode, modifier);
+         poplevel (1, 1, 0);
+         expand_end_bindings (getdecls (), 1, 0);
+         return to_return;
+       }
+      break;
+
+    default:
+      fatal ("Can't expand '%s' tree - java_lang_expand_expr",
+            tree_code_name [TREE_CODE (exp)]);
+    }
+}
+
+void
+expand_byte_code (jcf, method)
+     JCF *jcf;
+     tree method;
+{
+  int PC;
+  int i;
+  int saw_index;
+  unsigned char *linenumber_pointer;
+  struct eh_range *prev_eh_ranges = NULL_EH_RANGE;
+  struct eh_range *eh_ranges;
+
+#undef RET /* Defined by config/i386/i386.h */
+#undef AND /* Causes problems with opcodes for iand and land. */
+#undef PTR
+#define BCODE byte_ops
+#define BYTE_type_node byte_type_node
+#define SHORT_type_node short_type_node
+#define INT_type_node int_type_node
+#define LONG_type_node long_type_node
+#define CHAR_type_node char_type_node
+#define PTR_type_node ptr_type_node
+#define FLOAT_type_node float_type_node
+#define DOUBLE_type_node double_type_node
+#define VOID_type_node void_type_node
+  jint INT_temp;
+  unsigned char* byte_ops;
+  long length = DECL_CODE_LENGTH (method);
+
+  stack_pointer = 0;
+  JCF_SEEK (jcf, DECL_CODE_OFFSET (method));
+  byte_ops = jcf->read_ptr;
+
+#define CONST_INDEX_1 (saw_index = 1, IMMEDIATE_u1)
+#define CONST_INDEX_2 (saw_index = 1, IMMEDIATE_u2)
+#define VAR_INDEX_1 (saw_index = 1, IMMEDIATE_u1)
+#define VAR_INDEX_2 (saw_index = 1, IMMEDIATE_u2)
+
+#define CHECK_PC_IN_RANGE(PC) 1 /* Already handled by verifier. */
+
+  instruction_bits = oballoc (length + 1);
+  bzero (instruction_bits, length + 1);
+
+  /* We make an initial pass of the line number table, to note
+     which instructions have associated line number entries. */
+  linenumber_pointer = linenumber_table;
+  for (i = 0; i < linenumber_count; i++)
+    {
+      int pc = GET_u2 (linenumber_pointer);
+      linenumber_pointer += 4;
+      if (pc >= length)
+       warning ("invalid PC in line number table");
+      else
+       {
+         if ((instruction_bits[pc] & BCODE_HAS_LINENUMBER) != 0)
+           instruction_bits[pc] |= BCODE_HAS_MULTI_LINENUMBERS;
+         instruction_bits[pc] |= BCODE_HAS_LINENUMBER;
+       }
+    }  
+
+  /* Do a preliminary pass.
+   * This figures out which PC can be the targets of jumps. */
+  for (PC = 0; PC < length;)
+    {
+      int oldpc = PC; /* PC at instruction start. */
+      instruction_bits [PC] |=  BCODE_INSTRUCTION_START;
+      switch (byte_ops[PC++])
+       {
+#define JAVAOP(OPNAME, OPCODE, OPKIND, OPERAND_TYPE, OPERAND_VALUE) \
+        case OPCODE: \
+         PRE_##OPKIND(OPERAND_TYPE, OPERAND_VALUE); \
+         break;
+
+#define NOTE_LABEL(PC) note_label(oldpc, PC)
+
+#define PRE_PUSHC(OPERAND_TYPE, OPERAND_VALUE) (void)(OPERAND_VALUE);
+#define PRE_LOAD(OPERAND_TYPE, OPERAND_VALUE) (void)(OPERAND_VALUE);
+#define PRE_STORE(OPERAND_TYPE, OPERAND_VALUE) (void)(OPERAND_VALUE);
+#define PRE_STACK(OPERAND_TYPE, OPERAND_VALUE) /* nothing */
+#define PRE_UNOP(OPERAND_TYPE, OPERAND_VALUE) /* nothing */
+#define PRE_BINOP(OPERAND_TYPE, OPERAND_VALUE) /* nothing */
+#define PRE_CONVERT(OPERAND_TYPE, OPERAND_VALUE) /* nothing */
+#define PRE_CONVERT2(OPERAND_TYPE, OPERAND_VALUE) /* nothing */
+
+#define PRE_SPECIAL(OPERAND_TYPE, INSTRUCTION) \
+  PRE_SPECIAL_##INSTRUCTION(OPERAND_TYPE)
+#define PRE_SPECIAL_IINC(OPERAND_TYPE) \
+  ((void) IMMEDIATE_u1, (void) IMMEDIATE_s1)
+#define PRE_SPECIAL_ENTER(IGNORE) /* nothing */
+#define PRE_SPECIAL_EXIT(IGNORE) /* nothing */
+#define PRE_SPECIAL_THROW(IGNORE) /* nothing */
+#define PRE_SPECIAL_BREAK(IGNORE) /* nothing */
+
+/* two forms of wide instructions */
+#define PRE_SPECIAL_WIDE(IGNORE) \
+  { \
+    int modified_opcode = IMMEDIATE_u1; \
+    if (modified_opcode == OPCODE_iinc)        \
+      { \
+       (void) IMMEDIATE_u2;    /* indexbyte1 and indexbyte2 */ \
+       (void) IMMEDIATE_s2;    /* constbyte1 and constbyte2 */ \
+      } \
+    else \
+      { \
+       (void) IMMEDIATE_u2;    /* indexbyte1 and indexbyte2 */ \
+      } \
+  }
+
+/* nothing */ /* XXX JH */
+
+#define PRE_IMPL(IGNORE1, IGNORE2) /* nothing */
+
+#define PRE_MONITOR(OPERAND_TYPE, OPERAND_VALUE) /* nothing */
+
+#define PRE_RETURN(OPERAND_TYPE, OPERAND_VALUE) /* nothing */
+#define PRE_ARRAY(OPERAND_TYPE, SUBOP) \
+         PRE_ARRAY_##SUBOP(OPERAND_TYPE)
+#define PRE_ARRAY_LOAD(TYPE) /* nothing */
+#define PRE_ARRAY_STORE(TYPE) /* nothing */
+#define PRE_ARRAY_LENGTH(TYPE) /* nothing */
+#define PRE_ARRAY_NEW(TYPE) PRE_ARRAY_NEW_##TYPE
+#define PRE_ARRAY_NEW_NUM ((void) IMMEDIATE_u1)
+#define PRE_ARRAY_NEW_PTR ((void) IMMEDIATE_u2)
+#define PRE_ARRAY_NEW_MULTI ((void) IMMEDIATE_u2, (void) IMMEDIATE_u1)
+
+#define PRE_TEST(OPERAND_TYPE, OPERAND_VALUE) NOTE_LABEL (oldpc+IMMEDIATE_s2)
+#define PRE_COND(OPERAND_TYPE, OPERAND_VALUE) NOTE_LABEL (oldpc+IMMEDIATE_s2)
+#define PRE_BRANCH(OPERAND_TYPE, OPERAND_VALUE) \
+  saw_index = 0;  INT_temp = (OPERAND_VALUE); \
+  if (!saw_index)  NOTE_LABEL(oldpc + INT_temp);
+#define PRE_JSR(OPERAND_TYPE, OPERAND_VALUE) \
+  saw_index = 0;  INT_temp = (OPERAND_VALUE); \
+  if (!saw_index)  NOTE_LABEL(oldpc + INT_temp);
+
+#define PRE_RET(OPERAND_TYPE, OPERAND_VALUE)  (void)(OPERAND_VALUE)
+
+#define PRE_SWITCH(OPERAND_TYPE, TABLE_OR_LOOKUP) \
+  PC = (PC + 3) / 4 * 4; PRE_##TABLE_OR_LOOKUP##_SWITCH
+
+#define PRE_LOOKUP_SWITCH                                              \
+  { jint default_offset = IMMEDIATE_s4;  jint npairs = IMMEDIATE_s4;   \
+    NOTE_LABEL (default_offset+oldpc);                                 \
+    if (npairs >= 0)                                                   \
+      while (--npairs >= 0) {                                          \
+       jint match = IMMEDIATE_s4; jint offset = IMMEDIATE_s4;          \
+       NOTE_LABEL (offset+oldpc); }                                    \
+  }
+
+#define PRE_TABLE_SWITCH                               \
+  { jint default_offset = IMMEDIATE_s4;                        \
+    jint low = IMMEDIATE_s4; jint high = IMMEDIATE_s4; \
+    NOTE_LABEL (default_offset+oldpc);                 \
+    if (low <= high)                                   \
+     while (low++ <= high) {                           \
+       jint offset = IMMEDIATE_s4;                     \
+       NOTE_LABEL (offset+oldpc); }                    \
+  }
+
+#define PRE_FIELD(MAYBE_STATIC, PUT_OR_GET) (void)(IMMEDIATE_u2);
+#define PRE_OBJECT(MAYBE_STATIC, PUT_OR_GET) (void)(IMMEDIATE_u2);
+#define PRE_INVOKE(MAYBE_STATIC, IS_INTERFACE) \
+  (void)(IMMEDIATE_u2); \
+  PC += 2 * IS_INTERFACE /* for invokeinterface */;
+
+#include "javaop.def"
+#undef JAVAOP
+       }
+    } /* for */
+
+  if (! verify_jvm_instructions (jcf, byte_ops, length))
+    return;
+
+  /* Translate bytecodes to rtl instructions. */
+  linenumber_pointer = linenumber_table;
+  for (PC = 0; PC < length;)
+    {
+      if ((instruction_bits [PC] & BCODE_TARGET) != 0 || PC == 0)
+       {
+         tree label = lookup_label (PC);
+          flush_quick_stack ();
+         if ((instruction_bits [PC] & BCODE_TARGET) != 0)
+           expand_label (label);
+         if (LABEL_VERIFIED (label) || PC == 0)
+           load_type_state (label);
+       }
+
+      if (! (instruction_bits [PC] & BCODE_VERIFIED))
+       {
+         /* never executed - skip */
+         warning ("Some bytecode operations (starting at pc %d) can never be executed", PC);
+         while (PC < length
+                && ! (instruction_bits [PC] & BCODE_VERIFIED))
+           PC++;
+         continue;
+       }
+
+
+      /* Handle possible line number entry for this PC.
+
+        This code handles out-of-order and multiple linenumbers per PC,
+        but is optimized for the case of line numbers increasing
+        monotonically with PC. */
+      if ((instruction_bits[PC] & BCODE_HAS_LINENUMBER) != 0)
+       {
+         if ((instruction_bits[PC] & BCODE_HAS_MULTI_LINENUMBERS) != 0
+             || GET_u2 (linenumber_pointer) != PC)
+           linenumber_pointer = linenumber_table;
+         while (linenumber_pointer < linenumber_table + linenumber_count * 4)
+           {
+             int pc = GET_u2 (linenumber_pointer);
+             linenumber_pointer += 4;
+             if (pc == PC)
+               {
+                 lineno = GET_u2 (linenumber_pointer - 2);
+                 emit_line_note (input_filename, lineno);
+                 if (!(instruction_bits[PC] & BCODE_HAS_MULTI_LINENUMBERS))
+                   break;
+               }
+           }
+       }
+      maybe_start_try (PC);
+      maybe_pushlevels (PC);
+
+      PC = process_jvm_instruction (PC, byte_ops, length);
+
+      maybe_poplevels (PC);
+      maybe_end_try (PC);
+    } /* for */
+}
+
+void
+java_push_constant_from_pool (jcf, index)
+     JCF *jcf;
+     int index;
+{
+  tree c;
+  if (JPOOL_TAG (jcf, index) == CONSTANT_String)
+    {
+      tree name;
+      push_obstacks (&permanent_obstack, &permanent_obstack);
+      name = get_name_constant (jcf, JPOOL_USHORT1 (jcf, index));
+      index = alloc_name_constant (CONSTANT_String, name);
+      c = build_ref_from_constant_pool (index);
+      TREE_TYPE (c) = promote_type (string_type_node);
+      pop_obstacks ();
+    }
+  else
+    c = get_constant (jcf, index);
+  push_value (c);
+} 
+
+int
+process_jvm_instruction (PC, byte_ops, length)
+     int PC;
+     unsigned char* byte_ops;
+     long length;
+{ 
+  char *opname; /* Temporary ??? */
+  int oldpc = PC; /* PC at instruction start. */
+  switch (byte_ops[PC++])
+    {
+#define JAVAOP(OPNAME, OPCODE, OPKIND, OPERAND_TYPE, OPERAND_VALUE) \
+    case OPCODE: \
+      opname = #OPNAME; \
+      OPKIND(OPERAND_TYPE, OPERAND_VALUE); \
+      break;
+
+#define RET(OPERAND_TYPE, OPERAND_VALUE)                               \
+  {                                                                    \
+    int saw_index = 0;                                                 \
+    int index     = OPERAND_VALUE;                                     \
+    build_java_ret (find_local_variable (index, ptr_type_node, oldpc));        \
+  }
+
+#define JSR(OPERAND_TYPE, OPERAND_VALUE)               \
+  {                                                    \
+    tree where = lookup_label (oldpc+OPERAND_VALUE);   \
+    tree ret   = lookup_label (PC);                    \
+    build_java_jsr (where, ret);                       \
+  }
+
+/* Push a constant onto the stack. */
+#define PUSHC(OPERAND_TYPE, OPERAND_VALUE) \
+  { int saw_index = 0;  int ival = (OPERAND_VALUE); \
+    if (saw_index) java_push_constant_from_pool (current_jcf, ival); \
+    else expand_java_pushc (ival, OPERAND_TYPE##_type_node); }
+
+/* internal macro added for use by the WIDE case */
+#define LOAD_INTERNAL(OPTYPE, OPVALUE) \
+  push_value (find_local_variable (OPVALUE, type_map[OPVALUE], oldpc));
+
+/* Push local variable onto the opcode stack. */
+#define LOAD(OPERAND_TYPE, OPERAND_VALUE) \
+  { \
+    /* have to do this since OPERAND_VALUE may have side-effects */ \
+    int opvalue = OPERAND_VALUE; \
+    LOAD_INTERNAL(OPERAND_TYPE##_type_node, opvalue); \
+  }
+
+#define RETURN(OPERAND_TYPE, OPERAND_VALUE) \
+  expand_java_return (OPERAND_TYPE##_type_node)
+
+#define REM_EXPR TRUNC_MOD_EXPR
+#define BINOP(OPERAND_TYPE, OPERAND_VALUE) \
+  expand_java_binop (OPERAND_TYPE##_type_node, OPERAND_VALUE##_EXPR)
+
+#define FIELD(IS_STATIC, IS_PUT) \
+  expand_java_field_op (IS_STATIC, IS_PUT, IMMEDIATE_u2)
+
+#define TEST(OPERAND_TYPE, CONDITION) \
+  expand_test (CONDITION##_EXPR, OPERAND_TYPE##_type_node, oldpc+IMMEDIATE_s2)
+
+#define COND(OPERAND_TYPE, CONDITION) \
+  expand_cond (CONDITION##_EXPR, OPERAND_TYPE##_type_node, oldpc+IMMEDIATE_s2)
+
+#define BRANCH(OPERAND_TYPE, OPERAND_VALUE) \
+  BRANCH_##OPERAND_TYPE (OPERAND_VALUE)
+
+#define BRANCH_GOTO(OPERAND_VALUE) \
+  expand_java_goto (oldpc + OPERAND_VALUE)
+
+#define BRANCH_CALL(OPERAND_VALUE) \
+  expand_java_call (oldpc + OPERAND_VALUE, oldpc)
+
+#if 0
+#define BRANCH_RETURN(OPERAND_VALUE) \
+  { \
+    tree type = OPERAND_TYPE##_type_node; \
+    tree value = find_local_variable (OPERAND_VALUE, type, oldpc); \
+    expand_java_ret (value); \
+  }
+#endif
+
+#define NOT_IMPL(OPERAND_TYPE, OPERAND_VALUE) \
+         fprintf (stderr, "%3d: %s ", oldpc, opname); \
+         fprintf (stderr, "(not implemented)\n")
+#define NOT_IMPL1(OPERAND_VALUE) \
+         fprintf (stderr, "%3d: %s ", oldpc, opname); \
+         fprintf (stderr, "(not implemented)\n")
+
+#define BRANCH_RETURN(OPERAND_VALUE) NOT_IMPL1(OPERAND_VALUE)
+
+#define STACK(SUBOP, COUNT) STACK_##SUBOP (COUNT)
+
+#define STACK_POP(COUNT) java_stack_pop (COUNT)
+
+#define STACK_SWAP(COUNT) java_stack_swap()
+
+#define STACK_DUP(COUNT) java_stack_dup (COUNT, 0)
+#define STACK_DUPx1(COUNT) java_stack_dup (COUNT, 1)
+#define STACK_DUPx2(COUNT) java_stack_dup (COUNT, 2)
+
+#define SWITCH(OPERAND_TYPE, TABLE_OR_LOOKUP) \
+  PC = (PC + 3) / 4 * 4; TABLE_OR_LOOKUP##_SWITCH
+
+#define LOOKUP_SWITCH \
+  { jint default_offset = IMMEDIATE_s4;  jint npairs = IMMEDIATE_s4; \
+    tree selector = pop_value (INT_type_node); \
+    tree duplicate, label; \
+    tree type = TREE_TYPE (selector); \
+    flush_quick_stack (); \
+    expand_start_case (0, selector, type, "switch statement");\
+    push_momentary (); \
+    while (--npairs >= 0) \
+      { \
+       jint match = IMMEDIATE_s4; jint offset = IMMEDIATE_s4; \
+       tree value = build_int_2 (match, match < 0 ? -1 : 0); \
+       TREE_TYPE (value) = type; \
+       label =  build_decl (LABEL_DECL, NULL_TREE, NULL_TREE); \
+       pushcase (value, convert, label, &duplicate); \
+       expand_java_goto (oldpc + offset); \
+      } \
+    label =  build_decl (LABEL_DECL, NULL_TREE, NULL_TREE); \
+    pushcase (NULL_TREE, 0, label, &duplicate); \
+    expand_java_goto (oldpc + default_offset); \
+    pop_momentary (); \
+    expand_end_case (selector); \
+  }
+
+#define TABLE_SWITCH \
+  { jint default_offset = IMMEDIATE_s4; \
+    jint low = IMMEDIATE_s4; jint high = IMMEDIATE_s4; \
+    tree selector = pop_value (INT_type_node); \
+    tree duplicate, label; \
+    tree type = TREE_TYPE (selector); \
+    flush_quick_stack (); \
+    expand_start_case (0, selector, type, "switch statement");\
+    push_momentary (); \
+    for (; low <= high; low++) \
+      { \
+        jint offset = IMMEDIATE_s4; \
+        tree value = build_int_2 (low, low < 0 ? -1 : 0); \
+        TREE_TYPE (value) = type; \
+        label =  build_decl (LABEL_DECL, NULL_TREE, NULL_TREE); \
+        pushcase (value, convert, label, &duplicate); \
+        expand_java_goto (oldpc + offset); \
+      } \
+    label =  build_decl (LABEL_DECL, NULL_TREE, NULL_TREE); \
+    pushcase (NULL_TREE, 0, label, &duplicate); \
+    expand_java_goto (oldpc + default_offset); \
+    pop_momentary (); \
+    expand_end_case (selector); \
+  }
+
+#define INVOKE(MAYBE_STATIC, IS_INTERFACE) \
+  { int opcode = byte_ops[PC-1]; \
+    int method_ref_index = IMMEDIATE_u2; \
+    int nargs; \
+    if (IS_INTERFACE) { nargs = IMMEDIATE_u1;  (void) IMMEDIATE_u1; } \
+    else nargs = -1; \
+    expand_invoke (opcode, method_ref_index, nargs); \
+  }
+
+/* Handle new, checkcast, instanceof */
+#define OBJECT(TYPE, OP) \
+  expand_java_##OP (get_class_constant (current_jcf, IMMEDIATE_u2))
+
+#define ARRAY(OPERAND_TYPE, SUBOP) ARRAY_##SUBOP(OPERAND_TYPE)
+
+#define ARRAY_LOAD(OPERAND_TYPE)                       \
+  {                                                    \
+    expand_java_arrayload( OPERAND_TYPE##_type_node ); \
+  }
+
+#define ARRAY_STORE(OPERAND_TYPE)                      \
+  {                                                    \
+    expand_java_arraystore( OPERAND_TYPE##_type_node );        \
+  }
+
+#define ARRAY_LENGTH(OPERAND_TYPE) expand_java_array_length();
+#define ARRAY_NEW(OPERAND_TYPE) ARRAY_NEW_##OPERAND_TYPE()
+#define ARRAY_NEW_PTR()                                                        \
+    push_value (build_anewarray (get_class_constant (current_jcf,      \
+                                                    IMMEDIATE_u2),     \
+                                pop_value (int_type_node)));
+#define ARRAY_NEW_NUM()                                \
+  {                                            \
+    int atype = IMMEDIATE_u1;                  \
+    push_value (build_newarray (atype, pop_value (int_type_node)));\
+  }
+#define ARRAY_NEW_MULTI()                                      \
+  {                                                            \
+    tree class = get_class_constant (current_jcf, IMMEDIATE_u2 );      \
+    int  ndims = IMMEDIATE_u1;                                 \
+    expand_java_multianewarray( class, ndims );                        \
+  }
+
+#define UNOP(OPERAND_TYPE, OPERAND_VALUE) \
+  push_value (fold (build1 (NEGATE_EXPR, OPERAND_TYPE##_type_node, \
+                           pop_value (OPERAND_TYPE##_type_node))));
+
+#define CONVERT2(FROM_TYPE, TO_TYPE)                                    \
+  {                                                                     \
+    push_value (build1 (NOP_EXPR, int_type_node,                        \
+                       (convert (TO_TYPE##_type_node,                   \
+                                 pop_value (FROM_TYPE##_type_node))))); \
+  }
+
+#define CONVERT(FROM_TYPE, TO_TYPE)                            \
+  {                                                            \
+    push_value (convert (TO_TYPE##_type_node,                  \
+                        pop_value (FROM_TYPE##_type_node)));   \
+  }
+
+/* internal macro added for use by the WIDE case 
+   Added TREE_TYPE (decl) assignment, apbianco  */
+#define STORE_INTERNAL(OPTYPE, OPVALUE)                        \
+  {                                                    \
+    tree decl, value;                                  \
+    int var = OPVALUE;                                 \
+    tree type = OPTYPE;                                        \
+    value = pop_value (type);                          \
+    type = TREE_TYPE (value);                          \
+    decl = find_local_variable (var, type, oldpc);     \
+    set_local_type (var, type );                       \
+    expand_assignment (decl, value, 0, 0);             \
+  }
+
+#define STORE(OPERAND_TYPE, OPERAND_VALUE) \
+  { \
+    /* have to do this since OPERAND_VALUE may have side-effects */ \
+    int opvalue = OPERAND_VALUE; \
+    STORE_INTERNAL(OPERAND_TYPE##_type_node, opvalue); \
+  }
+
+#define SPECIAL(OPERAND_TYPE, INSTRUCTION) \
+  SPECIAL_##INSTRUCTION(OPERAND_TYPE)
+
+#define SPECIAL_ENTER(IGNORED) MONITOR_OPERATION (soft_monitorenter_node)
+#define SPECIAL_EXIT(IGNORED)  MONITOR_OPERATION (soft_monitorexit_node)
+
+#define MONITOR_OPERATION(call)                        \
+  {                                            \
+    tree o = pop_value (ptr_type_node);                \
+    tree c;                                    \
+    flush_quick_stack ();                      \
+    c = build_java_monitor (call, o);          \
+    TREE_SIDE_EFFECTS (c) = 1;                 \
+    expand_expr_stmt (c);                      \
+  }
+
+#define SPECIAL_IINC(IGNORED) \
+  { \
+    unsigned int local_var_index = IMMEDIATE_u1; \
+    int ival = IMMEDIATE_s1; \
+    expand_iinc(local_var_index, ival, oldpc); \
+  }
+
+#define SPECIAL_WIDE(IGNORED) \
+  { \
+    int modified_opcode = IMMEDIATE_u1; \
+    unsigned int local_var_index = IMMEDIATE_u2; \
+    switch (modified_opcode) \
+      { \
+      case OPCODE_iinc: \
+       { \
+         int ival = IMMEDIATE_s2; \
+         expand_iinc (local_var_index, ival, oldpc); \
+         break; \
+       } \
+      case OPCODE_iload: \
+      case OPCODE_lload: \
+      case OPCODE_fload: \
+      case OPCODE_dload: \
+      case OPCODE_aload: \
+       { \
+         /* duplicate code from LOAD macro */ \
+         LOAD_INTERNAL(operand_type[modified_opcode], local_var_index); \
+         break; \
+       } \
+      case OPCODE_istore: \
+      case OPCODE_lstore: \
+      case OPCODE_fstore: \
+      case OPCODE_dstore: \
+      case OPCODE_astore: \
+       { \
+         STORE_INTERNAL(operand_type[modified_opcode], local_var_index); \
+         break; \
+       } \
+      default: \
+        error ("unrecogized wide sub-instruction"); \
+      } \
+  }
+
+#define SPECIAL_THROW(IGNORED) \
+  build_java_athrow (pop_value (throwable_type_node))
+
+#define SPECIAL_BREAK NOT_IMPL1
+#define IMPL          NOT_IMPL
+
+#include "javaop.def"
+#undef JAVAOP
+   default:
+    fprintf (stderr, "%3d: unknown(%3d)\n", oldpc, byte_ops[PC]);
+  }
+  return PC;
+}
diff --git a/gcc/java/gjavah.c b/gcc/java/gjavah.c
new file mode 100644 (file)
index 0000000..875e990
--- /dev/null
@@ -0,0 +1,959 @@
+/* Program to write C++-suitable header files from a Java(TM) .class
+   file.  This is similar to SUN's javah.
+
+Copyright (C) 1996, 1998 Free Software Foundation, Inc.
+
+This program is free software; you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation; either version 2, or (at your option)
+any later version.
+
+This program is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with GNU CC; see the file COPYING.  If not, write to
+the Free Software Foundation, 59 Temple Place - Suite 330,
+Boston, MA 02111-1307, USA.  
+
+Java and all Java-based marks are trademarks or registered trademarks
+of Sun Microsystems, Inc. in the United States and other countries.
+The Free Software Foundation is independent of Sun Microsystems, Inc.  */
+
+/* Written by Per Bothner <bothner@cygnus.com>, February 1996. */
+
+#include <stdio.h>
+#include "jcf.h"
+#ifdef __STDC__
+#include <stdlib.h>
+#endif
+#include <math.h>
+
+/* The output file.  */
+FILE *out = NULL;
+
+/* Nonzero on failure.  */
+static int found_error = 0;
+
+/* Directory to place resulting files in. Set by -d option. */
+char *output_directory = "";
+
+char *output_file = NULL;
+
+/* Directory to place temporary file.  Set by -td option.  Currently unused. */
+char *temp_directory = "/tmp";
+
+/* Number of friend functions we have to declare.  */
+static int friend_count;
+
+/* A class can optionally have a `friend' function declared.  If
+   non-NULL, this is that function.  */
+static char **friend_specs = NULL;
+
+/* Number of lines we are prepending before the class.  */
+static int prepend_count;
+
+/* We can prepend extra lines before the class's start. */
+static char **prepend_specs = NULL;
+
+/* Number of lines we are appending at the end of the class.  */
+static int add_count;
+
+/* We can append extra lines just before the class's end. */
+static char **add_specs = NULL;
+
+/* Number of lines we are appending after the class.  */
+static int append_count;
+
+/* We can append extra lines after the class's end. */
+static char **append_specs = NULL;
+
+int verbose = 0;
+
+int stubs = 0;
+
+struct JCF *current_jcf;
+struct JCF *main_jcf;
+
+/* This holds access information for the last field we examined.  They
+   let us generate "private:", "public:", and "protected:" properly.
+   If 0 then we haven't previously examined any field.  */
+static JCF_u2 last_access;
+
+#define ACC_VISIBILITY (ACC_PUBLIC | ACC_PRIVATE | ACC_PROTECTED)
+
+int seen_fields = 0;
+
+static void print_field_info PROTO ((FILE *, JCF*, int, int, JCF_u2));
+static void print_method_info PROTO ((FILE *, JCF*, int, int, JCF_u2));
+static void print_c_decl PROTO ((FILE*, JCF*, int, int, JCF_u2, int));
+
+JCF_u2 current_field_name;
+JCF_u2 current_field_value;
+JCF_u2 current_field_signature;
+JCF_u2 current_field_flags;
+
+#define HANDLE_START_FIELD(ACCESS_FLAGS, NAME, SIGNATURE, ATTRIBUTE_COUNT) \
+( current_field_name = (NAME), current_field_signature = (SIGNATURE), \
+  current_field_flags = (ACCESS_FLAGS), current_field_value = 0)
+
+#define HANDLE_END_FIELD() \
+  print_field_info (out, jcf, current_field_name, current_field_signature, \
+                   current_field_flags);
+
+#define HANDLE_CONSTANTVALUE(VALUEINDEX) current_field_value = (VALUEINDEX)
+
+#define HANDLE_METHOD(ACCESS_FLAGS, NAME, SIGNATURE, ATTRIBUTE_COUNT) \
+  print_method_info (out, jcf, NAME, SIGNATURE, ACCESS_FLAGS)
+
+#include "jcf-reader.c"
+
+/* Some useful constants.  */
+#define F_NAN_MASK 0x7f800000
+#define D_NAN_MASK 0x7ff0000000000000LL
+
+/* Return 1 if F is not Inf or NaN.  */
+static int
+java_float_finite (f)
+     jfloat f;
+{
+  int32 *ip = (int32 *) &f;
+
+  /* We happen to know that F_NAN_MASK will match all NaN values, and
+     also positive and negative infinity.  That's why we only need one
+     test here.  See The Java Language Specification, section 20.9.  */
+  return (*ip & F_NAN_MASK) != F_NAN_MASK;
+}
+
+/* Return 1 if D is not Inf or NaN.  */
+static int
+java_double_finite (d)
+     jdouble d;
+{
+  int64 *ip = (int64 *) &d;
+
+  /* Now check for all NaNs.  */
+  return (*ip & D_NAN_MASK) != D_NAN_MASK;
+}
+
+void
+DEFUN(print_name, (stream, jcf, name_index),
+      FILE* stream AND JCF* jcf AND int name_index)
+{
+  if (JPOOL_TAG (jcf, name_index) != CONSTANT_Utf8)
+    fprintf (stream, "<not a UTF8 constant>");
+  else
+    jcf_print_utf8 (stream, JPOOL_UTF_DATA (jcf, name_index),
+                   JPOOL_UTF_LENGTH (jcf, name_index));
+}
+
+/* Print base name of class.  The base name is everything after the
+   final separator.  */
+
+static void
+print_base_classname (stream, jcf, index)
+     FILE *stream;
+     JCF *jcf;
+     int index;
+{
+  int name_index = JPOOL_USHORT1 (jcf, index);
+  int i, len;
+  unsigned char *s, *p, *limit;
+
+  s = JPOOL_UTF_DATA (jcf, name_index);
+  len = JPOOL_UTF_LENGTH (jcf, name_index);
+  limit = s + len;
+  p = s;
+  while (s < limit)
+    {
+      int c = UTF8_GET (s, limit);
+      if (c == '/')
+       p = s;
+    }
+
+  while (p < limit)
+    {
+      int ch = UTF8_GET (p, limit);
+      if (ch == '/')
+       fputs ("::", stream);
+      else
+       jcf_print_char (stream, ch);
+    }
+}
+
+/* Return 0 if NAME is equal to STR, nonzero otherwise.  */
+
+static int
+utf8_cmp (str, length, name)
+     unsigned char *str;
+     int length;
+     char *name;
+{
+  unsigned char *limit = str + length;
+  int i;
+
+  for (i = 0; name[i]; ++i)
+    {
+      int ch = UTF8_GET (str, limit);
+      if (ch != name[i])
+       return 1;
+    }
+
+  return str != limit;
+}
+
+/* Generate an access control keyword based on FLAGS.  Returns 0 if
+   FLAGS matches the saved access information, nonzero otherwise.  */
+
+static void
+generate_access (stream, flags)
+     FILE *stream;
+     JCF_u2 flags;
+{
+  /* FIXME: Java's "protected" and "no access specifier" modes don't
+     actually map to C++ "protected".  That's how we map them for now,
+     though.  */
+
+  if (! (flags & ACC_VISIBILITY))
+    flags = ACC_PROTECTED;
+
+  if ((flags & ACC_VISIBILITY) == last_access)
+    return;
+  last_access = (flags & ACC_VISIBILITY);
+
+  switch (last_access)
+    {
+    case ACC_PUBLIC:
+      fputs ("public:\n", stream);
+      break;
+    case ACC_PRIVATE:
+      fputs ("private:\n", stream);
+      break;
+    case ACC_PROTECTED:
+      fputs ("protected:\n", stream);
+      break;
+    default:
+      found_error = 1;
+      fprintf (stream, "#error unrecognized visibility %d\n",
+              (flags & ACC_VISIBILITY));
+      break;
+    }
+}
+
+static void
+DEFUN(print_field_info, (stream, jcf, name_index, sig_index, flags),
+      FILE *stream AND JCF* jcf
+      AND int name_index AND int sig_index AND JCF_u2 flags)
+{
+  if (flags & ACC_FINAL)
+    {
+      if (current_field_value > 0)
+       {
+         jlong num;
+         char buffer[25];
+
+         generate_access (stream, flags);
+         switch (JPOOL_TAG (jcf, current_field_value))
+           {
+           case CONSTANT_Integer:
+             fputs ("  static const jint ", out);
+             print_name (out, jcf, name_index);
+             fputs (" = ", out);
+             num = JPOOL_INT (jcf, current_field_value);
+             format_int (buffer, num, 10);
+             fprintf (out, "%sL;\n", buffer);
+             break;
+           case CONSTANT_Long:
+             fputs ("  static const jlong ", out);
+             print_name (out, jcf, name_index);
+             fputs (" = ", out);
+             num = JPOOL_LONG (jcf, current_field_value);
+             format_int (buffer, num, 10);
+             fprintf (out, "%sLL;\n", buffer);
+             break;
+           case CONSTANT_Float:
+             {
+               jfloat fnum = JPOOL_FLOAT (jcf, current_field_value);
+               fputs ("  static const jfloat ", out);
+               print_name (out, jcf, name_index);
+               if (! java_float_finite (fnum))
+                 fputs (";\n", out);
+               else
+                 fprintf (out, " = %.10g;\n",  fnum);
+             }
+             break;
+           case CONSTANT_Double:
+             {
+               jdouble dnum = JPOOL_DOUBLE (jcf, current_field_value);
+               fputs ("  static const jdouble ", out);
+               print_name (out, jcf, name_index);
+               if (! java_double_finite (dnum))
+                 fputs (";\n", out);
+               else
+                 fprintf (out, " = %.17g;\n",  dnum);
+             }
+             break;
+           default:
+             fprintf(out, " <<inappropriate constant type>>\n");
+           }
+
+         return;
+       }
+    }
+
+  generate_access (stream, flags);
+  fputs ("  ", out);
+  if (flags & ACC_STATIC)
+    fputs ("static ", out);
+  print_c_decl (out, jcf, name_index, sig_index, flags, 0);
+  fputs (";\n", out);
+  if (! (flags & ACC_STATIC))
+    seen_fields++;
+}
+
+static void
+DEFUN(print_method_info, (stream, jcf, name_index, sig_index, flags),
+      FILE *stream AND JCF* jcf
+      AND int name_index AND int sig_index AND JCF_u2 flags)
+{
+  unsigned char *str;
+  int length, is_init = 0;
+
+  if (JPOOL_TAG (jcf, name_index) != CONSTANT_Utf8)
+    fprintf (stream, "<not a UTF8 constant>");
+  str = JPOOL_UTF_DATA (jcf, name_index);
+  length = JPOOL_UTF_LENGTH (jcf, name_index);
+  if (str[0] == '<')
+    {
+      /* Ignore internally generated methods like <clinit>.  However,
+        treat <init> as a constructor.  */
+      if (! utf8_cmp (str, length, "<init>"))
+       is_init = 1;
+      else  
+       return;
+    }
+
+  /* We can't generate a method whose name is a C++ reserved word.
+     For now the only problem has been `delete'; add more here as
+     required.  FIXME: we need a better solution than just ignoring
+     the method.  */
+  if (! utf8_cmp (str, length, "delete"))
+    return;
+
+  generate_access (stream, flags);
+
+  fputs ("  ", out);
+  if ((flags & ACC_STATIC))
+    fputs ("static ", out);
+  else if (! (flags & ACC_FINAL) && ! (jcf->access_flags & ACC_FINAL))
+    {
+      /* Don't print `virtual' if we have a constructor.  */
+      if (! is_init)
+       fputs ("virtual ", out);
+    }
+  print_c_decl (out, jcf, name_index, sig_index, flags, is_init);
+
+  /* FIXME: it would be nice to decompile small methods here.  That
+     would allow for inlining.  */
+
+  fprintf(out, ";\n");
+}
+
+static void
+DEFUN(print_c_decl, (stream, jcf, name_index, signature_index, flags, is_init),
+      FILE* stream AND JCF* jcf
+      AND int name_index AND int signature_index AND JCF_u2 flags
+      AND int is_init)
+{
+  if (JPOOL_TAG (jcf, signature_index) != CONSTANT_Utf8)
+    fprintf (stream, "<not a UTF8 constant>");
+  else
+    {
+      int length = JPOOL_UTF_LENGTH (jcf, signature_index);
+      unsigned char *str0 = JPOOL_UTF_DATA (jcf, signature_index);
+      register  unsigned char *str = str0;
+      unsigned char *limit = str + length;
+      int j;
+      char *ctype;
+      int need_space = 0;
+      int is_method = str[0] == '(';
+
+      if (is_method)
+       {
+         /* Skip to the return signature, and print that first.
+            However, don't do this is we are printing a construtcor.
+            */
+         if (is_init)
+           {
+             str = str0 + 1;
+             /* FIXME: Most programmers love Celtic knots because
+                they see their own code in the interconnected loops.
+                That is, this is spaghetti.  */
+             goto have_constructor;
+           }
+         else
+           {
+             while (str < limit)
+               {
+                 int ch = *str++;
+                 if (ch == ')')
+                   break;
+               }
+           }
+       }
+
+    again:
+      while (str < limit)
+       {
+         switch (str[0])
+           {
+           case '[':
+             for (str++; str < limit && *str >= '0' && *str <= '9'; str++)
+               ;
+             switch (*str)
+               {
+               case 'B': ctype = "jbyteArray";  goto printit;
+               case 'C': ctype = "jcharArray";  goto printit;
+               case 'D': ctype = "jdoubleArray";  goto printit;
+               case 'F': ctype = "jfloatArray";  goto printit;
+               case 'I': ctype = "jintArray";  goto printit;
+               case 'S': ctype = "jshortArray";  goto printit;
+               case 'J': ctype = "jlongArray";  goto printit;
+               case 'Z': ctype = "jbooleanArray";  goto printit;
+               case '[': ctype = "jobjectArray"; goto printit;
+               case 'L':
+                 /* We have to generate a reference to JArray here,
+                    so that our output matches what the compiler
+                    does.  */
+                 ++str;
+                 fputs ("JArray<", out);
+                 while (str < limit && *str != ';')
+                   {
+                     int ch = UTF8_GET (str, limit);
+                     if (ch == '/')
+                       fputs ("::", stream);
+                     else
+                       jcf_print_char (stream, ch);
+                   }
+                 fputs (" *> *", out);
+                 need_space = 0;
+                 ++str;
+                 break;
+               default:
+                 fprintf (stderr, "unparseable signature `%s'\n", str0);
+                 found_error = 1;
+                 ctype = "???"; goto printit;
+               }
+             break;
+           case '(':
+             fputc (*str++, stream);
+             continue;
+           case ')':
+             fputc (*str++, stream);
+             /* the return signature was printed in the first pass. */
+             return;
+           case 'B': ctype = "jbyte";  goto printit;
+           case 'C': ctype = "jchar";  goto printit;
+           case 'D': ctype = "jdouble";  goto printit;
+           case 'F': ctype = "jfloat";  goto printit;
+           case 'I': ctype = "jint";  goto printit;
+           case 'J': ctype = "jlong";  goto printit;
+           case 'S': ctype = "jshort";  goto printit;
+           case 'Z': ctype = "jboolean";  goto printit;
+           case 'V': ctype = "void";  goto printit;
+           case 'L':
+             ++str;
+             while (*str && *str != ';')
+               {
+                 int ch = UTF8_GET (str, limit);
+                 if (ch == '/')
+                   fputs ("::", stream);
+                 else
+                   jcf_print_char (stream, ch);
+               }
+             fputs (" *", stream);
+             if (*str == ';')
+               str++;
+             need_space = 0;
+             break;
+           default:
+             need_space = 1;
+             jcf_print_char (stream, *str++);
+             break;
+           printit:
+             str++;
+             need_space = 1;
+             fputs (ctype, stream);
+             break;
+           }
+
+         if (is_method && str < limit && *str != ')')
+           fputs (", ", stream);
+       }
+    have_constructor:
+      if (name_index)
+       {
+         if (need_space)
+           fprintf (stream, " ");
+         /* Declare constructors specially.  */
+         if (is_init)
+           print_base_classname (stream, jcf, jcf->this_class);
+         else
+           print_name (stream, jcf, name_index);
+       }
+      if (is_method)
+       {
+         fputs (" (", stream);
+         /* Go to beginning, skipping '('. */
+         str = str0 + 1;
+         goto again; /* To handle argument signatures. */
+       }
+    }
+}
+
+int
+DEFUN(print_mangled_classname, (stream, jcf, prefix, index),
+      FILE *stream AND JCF *jcf AND char *prefix AND int index)
+{
+  int name_index = JPOOL_USHORT1 (jcf, index);
+  fputs (prefix, stream);
+  jcf_print_utf8_replace (out,
+                         JPOOL_UTF_DATA (jcf, name_index),
+                         JPOOL_UTF_LENGTH (jcf, name_index),
+                         '/', '_');
+}
+
+/* Print PREFIX, then a class name in C++ format.  If the name refers
+   to an array, ignore it and don't print PREFIX.  Returns 1 if
+   something was printed, 0 otherwise.  */
+static int
+print_cxx_classname (stream, prefix, jcf, index)
+     FILE *stream;
+     char *prefix;
+     JCF *jcf;
+     int index;
+{
+  int name_index = JPOOL_USHORT1 (jcf, index);
+  int i, len, c;
+  unsigned char *s, *p, *limit;
+
+  s = JPOOL_UTF_DATA (jcf, name_index);
+  len = JPOOL_UTF_LENGTH (jcf, name_index);
+  limit = s + len;
+
+  /* Explicitly omit arrays here.  */
+  p = s;
+  c = UTF8_GET (p, limit);
+  if (c == '[')
+    return 0;
+
+  fputs (prefix, stream);
+  while (s < limit)
+    {
+      c = UTF8_GET (s, limit);
+      if (c == '/')
+       fputs ("::", stream);
+      else
+       jcf_print_char (stream, c);
+    }
+
+  return 1;
+}
+
+int written_class_count = 0;
+
+/* Return name of superclass.  If LEN is not NULL, fill it with length
+   of name.  */
+static unsigned char *
+super_class_name (derived_jcf, len)
+     JCF *derived_jcf;
+     int *len;
+{
+  int supername_index = JPOOL_USHORT1 (derived_jcf, derived_jcf->super_class);
+  int supername_length = JPOOL_UTF_LENGTH (derived_jcf, supername_index);
+  unsigned char *supername = JPOOL_UTF_DATA (derived_jcf, supername_index);
+
+  if (len)
+    *len = supername_length;
+
+  return supername;
+}
+
+/* Print declarations for all classes required by this class.  FIXME:
+   the current implementation just prints every class name from the
+   constant pool.  This is too much.  We really only need to print a
+   declaration for each class which is the type of a return value, a
+   field, or an argument.  */
+static void
+print_class_decls (out, jcf)
+     FILE *out;
+     JCF *jcf;
+{
+  int i, seen_one = 0;
+
+  for (i = 1; i < JPOOL_SIZE (jcf); ++i)
+    {
+      int kind = JPOOL_TAG (jcf, i);
+      if (kind == CONSTANT_Class)
+       {
+         if (print_cxx_classname (out, "class ", jcf, i))
+           fputs (";\n", out);
+         seen_one = 1;
+       }
+    }
+
+  if (seen_one)
+    fputs ("\n", out);
+}
+
+static void
+DEFUN(process_file, (file, out),
+      JCF *jcf AND FILE *out)
+{
+  int code, i;
+
+  current_jcf = main_jcf = jcf;
+
+  last_access = 0;
+
+  if (jcf_parse_preamble (jcf) != 0)
+    {
+      fprintf (stderr, "Not a valid Java .class file.\n");
+      found_error = 1;
+      return;
+    }
+
+  /* Parse and possibly print constant pool */
+  code = jcf_parse_constant_pool (jcf);
+  if (code != 0)
+    {
+      fprintf (stderr, "error while parsing constant pool\n");
+      found_error = 1;
+      return;
+    }
+  code = verify_constant_pool (jcf);
+  if (code > 0)
+    {
+      fprintf (stderr, "error in constant pool entry #%d\n", code);
+      found_error = 1;
+      return;
+    }
+
+  jcf_parse_class (jcf);
+
+  if (written_class_count++ == 0)
+    fputs ("// DO NOT EDIT THIS FILE - it is machine generated -*- c++ -*-\n\n",
+          out);
+
+  print_mangled_classname (out, jcf, "#ifndef __", jcf->this_class);
+  fprintf (out, "__\n");
+
+  print_mangled_classname (out, jcf, "#define __", jcf->this_class);
+  fprintf (out, "__\n\n");
+
+  if (jcf->super_class)
+    {
+      int super_length;
+      unsigned char *supername = super_class_name (jcf, &super_length);
+
+      fputs ("#include <", out);
+      jcf_print_utf8 (out, supername, super_length);
+      fputs (".h>\n", out);
+
+      /* FIXME: If our superclass is Object, then we include
+        java-array.h.  The right thing to do here is look at all the
+        methods and fields and see if an array is in use.  Only then
+        would we need to include java-array.h.  */
+      if (! utf8_cmp (supername, super_length, "java/lang/Object"))
+       fputs ("#include <java-array.h>\n", out);
+
+      fputs ("\n", out);
+    }
+
+  print_class_decls (out, jcf);
+
+  for (i = 0; i < prepend_count; ++i)
+    fprintf (out, "%s\n", prepend_specs[i]);
+  if (prepend_count > 0)
+    fputc ('\n', out);
+
+  if (! print_cxx_classname (out, "class ", jcf, jcf->this_class))
+    {
+      fprintf (stderr, "class is of array type\n");
+      found_error = 1;
+      return;
+    }
+  if (jcf->super_class)
+    {
+      if (! print_cxx_classname (out, " : public ", jcf, jcf->super_class))
+       {
+         fprintf (stderr, "base class is of array type\n");
+         found_error = 1;
+         return;
+       }
+    }
+  fputs ("\n{\n", out);
+
+  /* We make a single pass over the file, printing methods and fields
+     as we see them.  We have to list the methods in the same order
+     that they appear in the class file, so that the Java and C++
+     vtables have the same layout.  */
+  jcf_parse_fields (jcf);
+  jcf_parse_methods (jcf);
+  jcf_parse_final_attributes (jcf);
+
+  /* Generate friend decl if we still must.  */
+  for (i = 0; i < friend_count; ++i)
+    fprintf (out, "  friend %s\n", friend_specs[i]);
+
+  /* Generate extra declarations.  */
+  if (add_count > 0)
+    fputc ('\n', out);
+  for (i = 0; i < add_count; ++i)
+    fprintf (out, "  %s\n", add_specs[i]);
+
+  fputs ("};\n", out);
+
+  if (append_count > 0)
+    fputc ('\n', out);
+  for (i = 0; i < append_count; ++i)
+    fprintf (out, "%s\n", append_specs[i]);
+
+  print_mangled_classname (out, jcf, "\n#endif /* __", jcf->this_class);
+  fprintf (out, "__ */\n");
+}
+
+static void
+usage ()
+{
+  fprintf (stderr, "gjavah: no classes specified\n");
+  exit (1);
+}
+
+static void
+help ()
+{
+  printf ("Usage: gjavah [OPTION]... CLASS...\n\n");
+  printf ("Generate C++ header files from .class files\n\n");
+  printf ("  --classpath PATH        Set path to find .class files\n");
+  printf ("  -d DIRECTORY            Set output directory name\n");
+  printf ("  --help                  Print this help, then exit\n");
+  printf ("  -o FILE                 Set output file name\n");
+  printf ("  -td DIRECTORY           Set temporary directory name\n");
+  printf ("  -v, --verbose           Print extra information while running\n");
+  printf ("  --version               Print version number, then exit\n");
+  /* FIXME: print bug-report information.  */
+  exit (0);
+}
+
+static void
+no_argument (opt)
+     char *opt;
+{
+  fprintf (stderr, "gjavah: no argument given for option `%s'\n", opt);
+  exit (1);
+}
+
+static void
+version ()
+{
+  /* FIXME: use version.c?  */
+  printf ("gjavah (GNU gcc) 0.0\n\n");
+  printf ("Copyright (C) 1998 Free Software Foundation, Inc.\n");
+  printf ("This is free software; see the source for copying conditions.  There is NO\n");
+  printf ("warranty; not even for MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.\n\n");
+  exit (0);
+}
+
+int
+DEFUN(main, (argc, argv),
+      int argc AND char** argv)
+{
+  JCF jcf;
+  int argi;
+
+  if (argc <= 1)
+    usage ();
+
+  for (argi = 1; argi < argc; argi++)
+    {
+      char *arg = argv[argi];
+
+      if (arg[0] != '-' || ! strcmp (arg, "--"))
+       break;
+
+      /* Just let all arguments be given in either "-" or "--" form.  */
+      if (arg[1] == '-')
+       ++arg;
+
+      if (strcmp (arg, "-o") == 0)
+       {
+         if (argi + 1 < argc)
+           output_file = argv[++argi];
+         else
+           no_argument (argv[argi]);
+       }
+      else if (strcmp (arg, "-d") == 0)
+       {
+         if (argi + 1 < argc)
+           output_directory = argv[++argi];
+         else
+           no_argument (argv[argi]);
+       }
+      else if (strcmp (arg, "-td") == 0)
+       {
+         if (argi + 1 < argc)
+           temp_directory = argv[++argi];
+         else
+           no_argument (argv[argi]);
+       }
+      else if (strcmp (arg, "-prepend") == 0)
+       {
+         if (argi + 1 < argc)
+           {
+             if (prepend_count == 0)
+               prepend_specs = (char**) ALLOC ((argc-argi) * sizeof (char*));
+             prepend_specs[prepend_count++] = argv[++argi];
+           }
+         else
+           no_argument (argv[argi]);
+       }
+      else if (strcmp (arg, "-friend") == 0)
+       {
+         if (argi + 1 < argc)
+           {
+             if (friend_count == 0)
+               friend_specs = (char**) ALLOC ((argc-argi) * sizeof (char*));
+             friend_specs[friend_count++] = argv[++argi];
+           }
+         else
+           no_argument (argv[argi]);
+       }
+      else if (strcmp (arg, "-add") == 0)
+       {
+         if (argi + 1 < argc)
+           {
+             if (add_count == 0)
+               add_specs = (char**) ALLOC ((argc-argi) * sizeof (char*));
+             add_specs[add_count++] = argv[++argi];
+           }
+         else
+           no_argument (argv[argi]);
+       }
+      else if (strcmp (arg, "-append") == 0)
+       {
+         if (argi + 1 < argc)
+           {
+             if (append_count == 0)
+               append_specs = (char**) ALLOC ((argc-argi) * sizeof (char*));
+             append_specs[append_count++] = argv[++argi];
+           }
+         else
+           no_argument (argv[argi]);
+       }
+      else if (strcmp (arg, "-classpath") == 0)
+       {
+         if (argi + 1 < argc)
+           classpath = argv[++argi];
+         else
+           no_argument (argv[argi]);
+       }
+      else if (strcmp (arg, "-verbose") == 0 || strcmp (arg, "-v") == 0)
+       verbose++;
+      else if (strcmp (arg, "-stubs") == 0)
+       stubs++;
+      else if (strcmp (arg, "-help") == 0)
+       help ();
+      else if (strcmp (arg, "-version") == 0)
+       version ();
+      else
+       {
+         fprintf (stderr, "%s: illegal argument\n", argv[argi]);
+         exit (1);
+       }
+    }
+
+  if (argi == argc)
+    usage ();
+
+  if (classpath == NULL)
+    {
+      classpath = (char *) getenv ("CLASSPATH");
+      if (classpath == NULL)
+       classpath = "";
+    }
+
+  for (; argi < argc; argi++)
+    {
+      char *classname = argv[argi];
+      char *classfile_name, *current_output_file;
+
+      if (verbose)
+       fprintf (stderr, "Processing %s\n", classname);
+      classfile_name = find_class (classname, strlen (classname), &jcf, 1);
+      if (classfile_name == NULL)
+       {
+         fprintf (stderr, "%s: no such class\n", classname);
+         exit (1);
+       }
+      if (verbose)
+       fprintf (stderr, "Found in %s\n", classfile_name);
+      if (output_file)
+       {
+         if (strcmp (output_file, "-") == 0)
+           out = stdout;
+         else if (out == NULL)
+           out = fopen (output_file, "w");
+         if (out == NULL)
+           {
+             perror (output_file);
+             exit (1);
+           }
+         current_output_file = output_file;
+       }
+      else
+       {
+         int dir_len = strlen (output_directory);
+         int i, classname_length = strlen (classname);
+         current_output_file = (char*) ALLOC (dir_len + classname_length + 4);
+         strcpy (current_output_file, output_directory);
+         if (dir_len > 0 && output_directory[dir_len-1] != '/')
+           current_output_file[dir_len++] = '/';
+         for (i = 0; classname[i] != '\0'; i++)
+           {
+             char ch = classname[i];
+             if (ch == '.')
+               ch = '/';
+             current_output_file[dir_len++] = ch;
+           }
+         strcpy (current_output_file+dir_len, ".h");
+         out = fopen (current_output_file, "w");
+         if (out == NULL)
+           {
+             perror (current_output_file);
+             exit (1);
+           }
+       }
+      process_file (&jcf, out);
+      JCF_FINISH (&jcf);
+      if (current_output_file != output_file)
+       free (current_output_file);
+    }
+
+  if (out != NULL && out != stdout)
+    fclose (out);
+
+  return found_error;
+}
+
+/* TODO:
+
+ * Do whatever the javah -stubs flag does.
+
+ * Emit "structure forward declarations" when needed.
+
+ * Generate C headers, like javah
+
+ */
diff --git a/gcc/java/java-except.h b/gcc/java/java-except.h
new file mode 100644 (file)
index 0000000..576096f
--- /dev/null
@@ -0,0 +1,67 @@
+/* Definitions for exception handling for use by the GNU compiler
+   for the Java(TM) language compiler.
+   Copyright (C) 1997 Free Software Foundation, Inc.
+
+This file is part of GNU CC.
+
+GNU CC is free software; you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation; either version 2, or (at your option)
+any later version.
+
+GNU CC is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with GNU CC; see the file COPYING.  If not, write to
+the Free Software Foundation, 59 Temple Place - Suite 330,
+Boston, MA 02111-1307, USA.  
+
+Java and all Java-based marks are trademarks or registered trademarks
+of Sun Microsystems, Inc. in the United States and other countries.
+The Free Software Foundation is independent of Sun Microsystems, Inc.  */
+
+struct eh_range
+  {
+    /* The (byte-code PC) range of the handled block. */
+    int start_pc;
+    int end_pc;
+
+    /* A list of handlers.  For each element in the list,
+       the TREE_PURPOSE is the handled class (NULL_EXPR for a finally block),
+       and the TREE_VALUE is the LABEL_DECL of the handler. */
+    tree handlers;
+
+    /* Surrunding handler, if any. */
+    struct eh_range *outer;
+
+    /* The first child range.  It is is nested inside this range
+       (i.e. this.start_pc <= first_child.end_pc
+       && this.end_pc >= first_child.end_pc).
+       The children are linked together using next_sibling, and are sorted
+       by increasing start_pc and end_pc (we do not support non-nested
+       overlapping ranges). */
+    struct eh_range *first_child;
+
+    /* The next child of outer, in address order. */
+    struct eh_range *next_sibling;
+
+#if 0
+    /* Next handler, sorted by ascending start_pc then descending end_pc. */
+    tree next;
+#endif
+  };
+
+/* A dummy range that represents the entire method. */
+extern struct eh_range whole_range;
+
+#define NULL_EH_RANGE (&whole_range)
+
+extern struct eh_range * find_handler PROTO ((int));
+
+extern void method_init_exceptions PROTO ((void));
+
+extern void emit_handlers PROTO ((void));
+
diff --git a/gcc/java/java-opcodes.h b/gcc/java/java-opcodes.h
new file mode 100644 (file)
index 0000000..8fbe55c
--- /dev/null
@@ -0,0 +1,5 @@
+enum java_opcode {
+#define JAVAOP(NAME, CODE, KIND, TYPE, VALUE) OPCODE_##NAME = CODE,
+#include "javaop.def"
+#undef JAVAOP
+};
diff --git a/gcc/java/java-tree.def b/gcc/java/java-tree.def
new file mode 100644 (file)
index 0000000..a05b92d
--- /dev/null
@@ -0,0 +1,12 @@
+/* Shift right, logical. */
+
+DEFTREECODE (URSHIFT_EXPR, "urshift_expr", "2", 2)
+
+/* Return -1, 0, 1 depending on whether the first argument is
+   less, equal, or greater to the second argument. */
+DEFTREECODE (COMPARE_EXPR, "compare_expr", "2", 2)
+
+/* Same as COMPARE_EXPR, but if either value is NaN, the result is -1. */
+DEFTREECODE (COMPARE_L_EXPR, "compare_l_expr", "2", 2)
+/* Same as COMPARE_EXPR, but if either value is NaN, the result is 1. */
+DEFTREECODE (COMPARE_G_EXPR, "compare_g_expr", "2", 2)
diff --git a/gcc/java/java-tree.h b/gcc/java/java-tree.h
new file mode 100644 (file)
index 0000000..d808499
--- /dev/null
@@ -0,0 +1,720 @@
+/* Definitions for parsing and type checking for the GNU compiler for
+   the Java(TM) language.
+   Copyright (C) 1997, 1998 Free Software Foundation, Inc.
+
+This file is part of GNU CC.
+
+GNU CC is free software; you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation; either version 2, or (at your option)
+any later version.
+
+GNU CC is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with GNU CC; see the file COPYING.  If not, write to
+the Free Software Foundation, 59 Temple Place - Suite 330,
+Boston, MA 02111-1307, USA.  
+
+Java and all Java-based marks are trademarks or registered trademarks
+of Sun Microsystems, Inc. in the United States and other countries.
+The Free Software Foundation is independent of Sun Microsystems, Inc.  */
+
+/* Hacked by Per Bothner <bothner@cygnus.com> February 1996. */
+
+/* Java language-specific tree codes.  */
+#define DEFTREECODE(SYM, NAME, TYPE, LENGTH) SYM,
+enum java_tree_code {
+  __DUMMY = LAST_AND_UNUSED_TREE_CODE,
+#include "java-tree.def"
+  LAST_JAVA_TREE_CODE
+};
+#undef DEFTREECODE
+
+struct JCF;
+
+/* Usage of TREE_LANG_FLAG_?:
+   0: IS_A_SINGLE_IMPORT_CLASSFILE_NAME_P (in IDENTIFIER_NODE)
+      RESOLVE_EXPRESSION_NAME_P (in EXPR_WITH_FILE_LOCATION)
+      IS_FOR_LOOP_P (in LOOP_EXPR)
+   1: CLASS_HAS_SUPER_FLAG (in TREE_VEC).
+      IS_A_CLASSFILE_NAME (in IDENTIFIER_NODE)
+      COMPOUND_ASSIGN_P (in EXPR (binop_*))
+   2: RETURN_MAP_ADJUSTED (in TREE_VEC).
+      QUALIFIED_P (in IDENTIFIER_NODE)
+      PRIMARY_P (in EXPR_WITH_FILE_LOCATION)
+      MODIFY_EXPR_FROM_INITIALIZATION_P (in MODIFY_EXPR)
+   3: IS_AN_IMPORT_ON_DEMAND_P (in IDENTIFIER_NODE)
+      RESOLVE_PACKAGE_NAME_P (in EXPR_WITH_FILE_LOCATION)
+   4: RESOLVE_TYPE_NAME_P (in EXPR_WITH_FILE_LOCATION)
+   5: IS_BREAK_STMT_P (in EXPR_WITH_FILE_LOCATION)
+
+   Usage of TYPE_LANG_FLAG_?:
+   1: TYPE_ARRAY_P (in RECORD_TYPE).
+   2: CLASS_LOADED_P (in RECORD_TYPE).
+   3: CLASS_FROM_SOURCE_P (in RECORD_TYPE).
+   4: CLASS_P (in RECORD_TYPE).
+
+   Usage of DECL_LANG_FLAG_?:
+   1: METHOD_PUBLIC (in FUNCTION_DECL).
+      FIELD_PUBLIC (in FIELD_DECL).
+      CLASS_PUBLIC (in TYPE_DECL).
+   2: METHOD_STATIC (in FUNCTION_DECL).
+      (But note that FIELD_STATIC uses TREE_STATIC!)
+      CLASS_COMPLETE_P (in TYPE_DECL)
+   3: METHOD_FINAL (in FUNCTION_DECL)
+      FIELD_FINAL (in FIELD_DECL)
+      CLASS_FINAL (in TYPE_DECL)
+   4: METHOD_SYNCHRONIZED (in FUNCTION_DECL).
+      LABEL_IN_SUBR (in LABEL_DECL)
+      CLASS_INTERFACE (in TYPE_DECL)
+      FIELD_VOLATILE (int FIELD_DECL)
+   5: METHOD_ABSTRACT (in FUNCTION_DECL).
+      LABEL_IS_SUBR_START (in LABEL_DECL)
+      CLASS_ABSTRACT (in TYPE_DECL)
+      FIELD_TRANSIENT (in FIELD_DECL)
+   6: METHOD_TRANSIENT (in FUNCTION_DECL)
+      LABEL_CHANGED (in LABEL_DECL)
+      CLASS_SUPER (in TYPE_DECL, ACC_SUPER flag)
+      INITIALIZED_P (in FIELD_DECL, VAR_DECL, PARM_DECL)
+   7: DECL_CONSTRUCTOR_P (in FUNCTION_DECL)
+*/
+
+/* True if the class whose TYPE_BINFO this is has a superclass.
+   (True of all classes except Object.) */
+#define CLASS_HAS_SUPER_FLAG(BINFO) TREE_LANG_FLAG_1(BINFO)
+#define CLASS_HAS_SUPER(TYPE) CLASS_HAS_SUPER_FLAG (TYPE_BINFO (TYPE))
+
+/* Return the supertype of class TYPE, or NULL_TREE is it has none. */
+#define CLASSTYPE_SUPER(TYPE) (CLASS_HAS_SUPER (TYPE) ? \
+  BINFO_TYPE (TREE_VEC_ELT (TYPE_BINFO_BASETYPES (TYPE), 0)) : NULL_TREE)
+
+/* True if the class we are compiling is a .java source file;
+   false if it is a .class bytecode file. */
+extern int compiling_from_source;
+
+/* The class defined by the actual (main) file we are compiling. */
+extern tree main_class;
+
+/* The class we are currently processing. */
+extern tree current_class;
+
+/* Nonzero if we want to automatically do array bounds checking;
+   on by default.  Use -fno-bounds-check to disable.  */
+
+extern int flag_bounds_check;
+
+/* Nonzero if we should make is_compiled_class always return 1 for
+   appropriate classes that we're referencing.  */
+
+extern int flag_assume_compiled;
+
+extern int flag_emit_class_files;
+
+/* The Java .class file that provides main_class;  the main input file. */
+extern struct JCF main_jcf[1], *current_jcf;
+
+typedef struct CPool constant_pool;
+
+#define CONSTANT_ResolvedFlag 16
+
+/* The cpool->data[i] for a ResolvedString points to a STRING_CST. */
+#define CONSTANT_ResolvedString    (CONSTANT_String+CONSTANT_ResolvedFlag)
+
+/* The cpool->data[i] for a ResolvedClass points to a RECORD_TYPE. */
+#define CONSTANT_ResolvedClass     (CONSTANT_Class+CONSTANT_ResolvedFlag)
+
+#define CPOOL_UTF(CPOOL, INDEX) ((tree) (CPOOL)->data[INDEX])
+
+/* A NameAndType constant is represented as a TREE_LIST.
+   The type is the signature string (as an IDENTIFIER_NODE).  */
+
+#define NAME_AND_TYPE_NAME(CPOOL, IDX) \
+  CPOOL_UTF(CPOOL, CPOOL_USHORT1(CPOOL, IDX))
+#define NAME_AND_TYPE_SIGNATURE(CPOOL, IDX) \
+  CPOOL_UTF(CPOOL, CPOOL_USHORT2(CPOOL, IDX))
+
+/* A FieldRef, MethodRef or InterfaceMethodRef constant
+   is represented as a TREE_LIST. */
+
+#define COMPONENT_REF_CLASS_INDEX(CPOOL, IDX) CPOOL_USHORT1(CPOOL, IDX)
+#define COMPONENT_REF_NAME_AND_TYPE(CPOOL, IDX) CPOOL_USHORT2(CPOOL, IDX)
+#define COMPONENT_REF_NAME(CPOOL, IDX) \
+  NAME_AND_TYPE_NAME (CPOOL, COMPONENT_REF_NAME_AND_TYPE(CPOOL, IDX))
+#define COMPONENT_REF_SIGNATURE(CPOOL, IDX) \
+  NAME_AND_TYPE_SIGNATURE (CPOOL, COMPONENT_REF_NAME_AND_TYPE(CPOOL, IDX))
+
+/* "Promoted types" that are used for primitive types smaller
+   than int.  We could use int_type_node, but then we would lose
+   type information (such as needed for debugging). */
+extern tree promoted_byte_type_node;
+extern tree promoted_short_type_node;
+extern tree promoted_char_type_node;
+extern tree promoted_boolean_type_node;
+
+extern tree byte_type_node;
+extern tree short_type_node;
+extern tree int_type_node;
+extern tree long_type_node;
+
+extern tree unsigned_byte_type_node;
+extern tree unsigned_short_type_node;
+extern tree unsigned_int_type_node;
+extern tree unsigned_long_type_node;
+
+extern tree boolean_type_node;
+extern tree char_type_node;
+extern tree float_type_node;
+extern tree double_type_node;
+
+extern tree object_type_node;
+extern tree object_ptr_type_node;
+extern tree string_type_node;
+extern tree throwable_type_node;
+
+extern tree byte_array_type_node;
+extern tree short_array_type_node;
+extern tree int_array_type_node;
+extern tree long_array_type_node;
+extern tree boolean_array_type_node;
+extern tree char_array_type_node;
+extern tree double_array_type_node;
+extern tree float_array_type_node;
+extern tree array_array_type_node;
+extern tree object_array_type_node;
+extern tree string_array_type_node;
+extern tree TYPE_identifier_node;      /* "TYPE" */
+extern tree init_identifier_node;      /* "<init>" */
+extern tree clinit_identifier_node;      /* "<clinit>" */
+extern tree void_signature_node;       /* "()V" */
+extern tree finalize_identifier_node;  /* "finalize" */
+extern tree length_identifier_node;  /* "length" */
+extern tree this_identifier_node;  /* "this" */
+extern tree super_identifier_node;  /* "super" */
+extern tree one_elt_array_domain_type;
+extern tree void_type_node;
+extern tree ptr_type_node;
+/* The type of the return address of a subroutine. */
+extern tree return_address_type_node;
+
+/* Nodes for boolean constants TRUE and FALSE. */
+extern tree boolean_true_node, boolean_false_node;
+
+/* Integer constants not declared in tree.h. */
+extern tree integer_negative_one_node;
+extern tree integer_two_node;
+extern tree integer_four_node;
+
+/* The type for struct methodtable. */
+extern tree methodtable_type;
+extern tree methodtable_ptr_type;
+
+extern tree utf8const_type;
+extern tree utf8const_ptr_type;
+
+extern tree class_type_node;
+extern tree class_ptr_type;
+extern tree field_type_node;
+extern tree constants_type_node;
+extern tree dtable_type, dtable_ptr_type;
+extern tree field_ptr_type_node;
+extern tree field_info_union_node;
+extern tree method_type_node;
+extern tree method_ptr_type_node;
+#define nativecode_ptr_type_node ptr_type_node
+
+/* References to internal libjava functions we use. */
+extern tree alloc_object_node;
+extern tree soft_instanceof_node;
+extern tree soft_checkcast_node;
+extern tree soft_initclass_node;
+extern tree soft_newarray_node;
+extern tree soft_anewarray_node;
+extern tree soft_multianewarray_node;
+extern tree soft_badarrayindex_node;
+extern tree throw_node;
+extern tree soft_checkarraystore_node;
+extern tree soft_monitorenter_node;
+extern tree soft_monitorexit_node;
+extern tree soft_lookupinterfacemethod_node;
+extern tree soft_fmod_node;
+
+extern tree access_flags_type_node;
+
+extern tree class_dtable_decl;
+
+/* They need to be reset before processing each class */
+extern struct CPool *outgoing_cpool; 
+extern tree current_constant_pool_data_ref;
+
+
+struct lang_identifier
+{
+  struct tree_identifier ignore;
+  tree global_value, local_value;
+
+  /* If non-NULL:  An ADDR_REF to a VAR_DECL that contains
+   * the Utf8Const representation of the identifier.  */
+  tree utf8_ref;
+};
+
+/* Macros for access to language-specific slots in an identifier.  */
+/* UNless specifide, each of these slots contains a DECL node or null.  */
+
+/* This represents the value which the identifier has in the
+   file-scope namespace.  */
+#define IDENTIFIER_GLOBAL_VALUE(NODE)   \
+  (((struct lang_identifier *)(NODE))->global_value)
+/* This represents the value which the identifier has in the current
+   scope.  */
+#define IDENTIFIER_LOCAL_VALUE(NODE)    \
+  (((struct lang_identifier *)(NODE))->local_value)
+
+/* Given an identifier NODE, get the corresponding (non-handle) class.
+   For get_identifier ("java.lang.Number"), the result is
+   the struct whose DECL_ASSEMBLER_NAME is "Classjava_lang_Number". */
+#define IDENTIFIER_CLASS_VALUE(NODE) IDENTIFIER_GLOBAL_VALUE(NODE)
+
+/* Given an identifier NODE, get the corresponding handle class.
+   For get_identifier ("java.lang.Number"), the result is
+   the struct whose DECL_ASSEMBLER_NAME is "Hjava_lang_Number". */
+#define IDENTIFIER_HANDLECLASS_VALUE(NODE) ???
+
+/* Given a signature of a reference (or array) type, or a method, return the
+   corresponding type (if one has been allocated).
+   Do not use for primitive types, since they may be ambiguous.
+   (E.g. is "I" a signature or a class name?) */
+#define IDENTIFIER_SIGNATURE_TYPE(NODE) IDENTIFIER_GLOBAL_VALUE(NODE)
+
+/* If non-NULL:  An ADDR_REF to a VAR_DECL that contains
+   the Utf8Const representation of the identifier.  */
+#define IDENTIFIER_UTF8_REF(NODE) \
+  (((struct lang_identifier *)(NODE))->utf8_ref)
+
+#define IDENTIFIER_UTF8_DECL(NODE) \
+  TREE_OPERAND((((struct lang_identifier *)(NODE))->utf8_ref), 0)
+
+/* For a FUNCTION_DECL, if we are compiling a .class file, then this is
+   the position in the .class file of the method code.
+   Specifically, this is the code itself, not the code attribute. */
+#define DECL_CODE_OFFSET(DECL) (DECL_LANG_SPECIFIC(DECL)->code_offset)
+/* Similarly, the length of the bytecode. */
+#define DECL_CODE_LENGTH(DECL) (DECL_LANG_SPECIFIC(DECL)->code_length)
+/* Similarly, the position of the LineNumberTable attribute. */
+#define DECL_LINENUMBERS_OFFSET(DECL) \
+  (DECL_LANG_SPECIFIC(DECL)->linenumbers_offset)
+/* Similarly, the position of the LocalVariableTable attribute
+   (following the standard attribute header). */
+#define DECL_LOCALVARIABLES_OFFSET(DECL) \
+  (DECL_LANG_SPECIFIC(DECL)->localvariables_offset)
+
+#define DECL_MAX_LOCALS(DECL) (DECL_LANG_SPECIFIC(DECL)->max_locals)
+#define DECL_MAX_STACK(DECL) (DECL_LANG_SPECIFIC(DECL)->max_stack)
+/* Number of local variable slots needed for the arguments of this function. */
+#define DECL_ARG_SLOT_COUNT(DECL) (DECL_LANG_SPECIFIC(DECL)->arg_slot_count)
+/* Pointer to the function's COMPOUND_EXPR tree */
+#define DECL_FUNCTION_BODY(DECL) (DECL_LANG_SPECIFIC(DECL)->function_decl_body)
+
+/* In a LABEL_DECL, a TREE_VEC that saves the type_map at that point. */
+#define LABEL_TYPE_STATE(NODE) (DECL_INITIAL (NODE))
+
+/* In the label of a subroutine, a dummy label that records the
+   state following a merge of all the ret instructions in this subroutine. */
+#define LABEL_RETURN_LABEL(DECL) DECL_ARGUMENTS(DECL)
+
+/* In the label of a sub-routine, records the type state at return.
+ * A local may be TYPE_UNUSED, which means that the local is not
+ * used (stored to or loaded from) in this subroutine - at least for
+ * code that we have verified so far. */
+#define LABEL_RETURN_TYPE_STATE(NODE) LABEL_TYPE_STATE (LABEL_RETURN_LABEL (NODE))
+
+/* In a TREE_VEC for a LABEL_RETURN_TYPE_STATE, notes that
+   TREE_VEC_LENGTH has been adjust to the correct stack size. */
+#define RETURN_MAP_ADJUSTED(NODE) TREE_LANG_FLAG_2(NODE)
+
+/* In the label of a sub-routine, a chain of the return location labels. */
+#define LABEL_RETURN_LABELS(node) DECL_RESULT (LABEL_RETURN_LABEL(node))
+
+/* In a LABEL_DECL, the next pending label.
+   See pending_blocks in expr.c. */
+#define LABEL_PENDING_CHAIN(NODE) DECL_RESULT(NODE)
+
+/* In a LABEL_DECL, the corresponding bytecode program counter. */
+#define LABEL_PC(NODE) ((NODE)->decl.saved_insns.i)
+
+/* Used during verification to mark the label has "changed". (See JVM Spec). */
+#define LABEL_CHANGED(NODE) DECL_LANG_FLAG_6(NODE)
+
+/* In a LABEL_DECL, true if we have verified instructions starting here. */
+#define LABEL_VERIFIED(NODE) (instruction_bits[LABEL_PC(NODE)]&BCODE_VERIFIED)
+
+/* True if this code is within a subroutine (target of a jsr). */
+#define LABEL_IN_SUBR(NODE) DECL_LANG_FLAG_4(NODE)
+/* True if this code is the start of a subroutine (target of a jsr). */
+#define LABEL_IS_SUBR_START(NODE) DECL_LANG_FLAG_5(NODE)
+
+/* In a LABEL_DECL, if LABEL_IN_SUBR(NODE), points to start of subroutine. */
+#define LABEL_SUBR_START(NODE) DECL_ABSTRACT_ORIGIN(NODE)
+
+/* In a LABEL_DECL that has LABEL_IS_SUBR_START, this points to the start
+   of surrounding subroutine in the case of a nested subroutine,
+   and NULL_TREE otherwise. */
+#define LABEL_SUBR_CONTEXT(NODE) DECL_CONTEXT (LABEL_RETURN_LABEL (NODE))
+
+/* The slot number for this local variable. */
+#define DECL_LOCAL_SLOT_NUMBER(NODE) \
+  (((struct lang_decl_var*)DECL_LANG_SPECIFIC(NODE))->slot_number)
+/* The start (bytecode) pc for the valid range of this local variable. */
+#define DECL_LOCAL_START_PC(NODE) \
+  (((struct lang_decl_var*)DECL_LANG_SPECIFIC(NODE))->start_pc)
+/* The end (bytecode) pc for the valid range of this local variable. */
+#define DECL_LOCAL_END_PC(NODE) \
+  (((struct lang_decl_var*)DECL_LANG_SPECIFIC(NODE))->end_pc)
+/* For a VAR_DECLor PARM_DECL, used to chain decls with the same
+   slot_number in decl_map. */
+#define DECL_LOCAL_SLOT_CHAIN(NODE) \
+  (((struct lang_decl_var*)DECL_LANG_SPECIFIC(NODE))->slot_chain)
+
+/* DECL_LANG_SPECIFIC for FUNCTION_DECLs. */
+struct lang_decl
+{
+  /*  tree chain; not yet used. */
+  long code_offset;
+  int code_length;
+  long linenumbers_offset;
+  long localvariables_offset;
+  int arg_slots;
+  int max_locals, max_stack, arg_slot_count;
+  tree function_decl_body;     /* Hold all function's statements */
+};
+
+/* DECL_LANG_SPECIFIC for VAR_DECL and PARM_DECL. */
+struct lang_decl_var
+{
+  int slot_number;
+  int start_pc;
+  int end_pc;
+  tree slot_chain;
+};
+
+struct lang_type
+{
+  tree signature;
+  struct JCF *jcf;
+};
+
+#ifdef JAVA_USE_HANDLES
+/* TYPE_BINFO_HANDLE points from a handle-class to its corresponding
+   non-handle-class, and vice verse. */
+
+#define BINFO_HANDLE(NODE) TREE_VEC_ELT ((NODE), 6)
+
+/* Given a RECORD_TYPE for a handle type, return the corresponding class. */
+#define HANDLE_TO_CLASS_TYPE(HTYPE) BINFO_HANDLE (TYPE_BINFO (HTYPE))
+
+/* Given a RECORD_TYPE for a class, return the corresponding handle type. */
+#define CLASS_TO_HANDLE_TYPE(TYPE) BINFO_HANDLE (TYPE_BINFO (TYPE))
+#else
+#define HANDLE_TO_CLASS_TYPE(HTYPE) (HTYPE)
+#define CLASS_TO_HANDLE_TYPE(TYPE) (TYPE)
+#endif
+
+#define JCF_u4 unsigned long
+#define JCF_u2 unsigned short
+
+extern tree lookup_class PROTO ((tree));
+extern tree lookup_java_constructor PROTO ((tree, tree));
+extern tree lookup_java_method PROTO ((tree, tree, tree));
+extern tree lookup_argument_method PROTO ((tree, tree, tree));
+extern tree promote_type PROTO ((tree));
+extern tree get_constant PROTO ((struct JCF*, int));
+extern tree get_name_constant PROTO ((struct JCF*, int));
+extern tree get_class_constant PROTO ((struct JCF*, int));
+extern tree parse_signature PROTO ((struct JCF *jcf, int sig_index));
+extern int jcf_parse PROTO ((struct JCF*));
+extern tree add_field PROTO ((tree, tree, tree, int));
+extern tree add_method PROTO ((tree, int, tree, tree));
+extern tree add_method_1 PROTO ((tree, int, tree, tree));
+extern tree make_class ();
+extern tree push_class PROTO ((tree, tree));
+extern tree unmangle_classname PROTO ((const char *name, int name_length));
+extern tree parse_signature_string PROTO ((const unsigned char *, int));
+extern tree get_type_from_signature PROTO ((tree));
+extern void layout_class PROTO ((tree));
+extern tree make_class ();
+extern tree build_class_ref PROTO ((tree));
+extern tree build_dtable_decl PROTO ((tree));
+extern tree build_internal_class_name PROTO ((tree));
+extern tree build_constants_constructor ();
+extern tree build_ref_from_constant_pool PROTO ((int));
+extern tree build_utf8_ref PROTO ((tree));
+extern tree ident_subst PROTO ((const char*, int,
+                               const char*, int, int, const char*));
+extern tree identifier_subst PROTO ((const tree,
+                                    const char *, int, int, const char *));
+extern tree build_java_signature PROTO ((tree));
+extern tree build_java_argument_signature PROTO ((tree));
+extern void set_java_signature PROTO ((tree, tree));
+extern tree build_static_field_ref PROTO ((tree));
+extern tree build_address_of PROTO ((tree));
+extern tree find_local_variable PROTO ((int index, tree type, int pc));
+extern tree find_stack_slot PROTO ((int index, tree type));
+extern tree build_prim_array_type PROTO ((tree, HOST_WIDE_INT));
+extern tree build_java_array_type PROTO ((tree, HOST_WIDE_INT));
+extern int is_compiled_class PROTO ((tree));
+extern tree mangled_classname PROTO ((char*, tree));
+extern tree lookup_label PROTO ((int));
+extern tree pop_type PROTO ((tree));
+extern void pop_argument_types PROTO ((tree));
+extern tree decode_newarray_type PROTO ((int));
+extern tree lookup_field PROTO ((tree*, tree));
+extern int is_array_type_p PROTO ((tree));
+extern HOST_WIDE_INT java_array_type_length PROTO ((tree));
+extern void load_class PROTO ((tree, int));
+
+extern tree lookup_name PROTO ((tree));
+extern tree build_known_method_ref PROTO ((tree, tree, tree, tree, tree));
+extern tree build_class_init PROTO ((tree, tree));
+extern tree build_invokevirtual PROTO ((tree, tree));
+extern tree invoke_build_dtable PROTO ((int, tree));
+extern tree match_java_method PROTO ((tree, tree, tree));
+extern tree build_field_ref PROTO ((tree, tree, tree));
+extern void pushdecl_force_head PROTO ((tree));
+extern tree build_java_binop PROTO ((enum tree_code, tree, tree, tree));
+extern tree binary_numeric_promotion PROTO ((tree, tree, tree *, tree *));
+extern tree build_decl_no_layout PROTO ((enum tree_code, tree, tree));
+extern tree build_java_arrayaccess PROTO ((tree, tree, tree));
+extern tree build_newarray PROTO ((int, tree));
+extern tree build_anewarray PROTO ((tree, tree));
+extern tree build_java_array_length_access PROTO ((tree));
+extern tree build_java_arraynull_check PROTO ((tree, tree, tree));
+extern tree create_label_decl PROTO ((tree));
+extern void push_labeled_block PROTO ((tree));
+
+/* Access flags etc for a method (a FUNCTION_DECL): */
+
+#define METHOD_PUBLIC(DECL) DECL_LANG_FLAG_1 (DECL)
+#define METHOD_PRIVATE(DECL) TREE_PRIVATE (DECL)
+#define METHOD_PROTECTED(DECL) TREE_PROTECTED (DECL)
+#define METHOD_STATIC(DECL) DECL_LANG_FLAG_2 (DECL)
+#define METHOD_FINAL(DECL) DECL_LANG_FLAG_3 (DECL)
+#define METHOD_SYNCHRONIZED(DECL) DECL_LANG_FLAG_4 (DECL)
+#define METHOD_NATIVE(DECL) DECL_EXTERNAL(DECL)
+#define METHOD_ABSTRACT(DECL) DECL_LANG_FLAG_5 (DECL)
+#define METHOD_TRANSIENT(DECL) DECL_LANG_FLAG_6 (DECL)
+
+#define DECL_CONSTRUCTOR_P(DECL) DECL_LANG_FLAG_7(DECL)
+
+/* Access flags etc for a variable/field (a FIELD_DECL): */
+
+#define FIELD_PRIVATE(DECL) TREE_PRIVATE (DECL)
+#define FIELD_PROTECTED(DECL) TREE_PROTECTED (DECL)
+#define FIELD_PUBLIC(DECL) DECL_LANG_FLAG_1 (DECL)
+#define FIELD_STATIC(DECL) TREE_STATIC (DECL)
+#define FIELD_FINAL(DECL) DECL_LANG_FLAG_3 (DECL)
+#define FIELD_VOLATILE(DECL) DECL_LANG_FLAG_4 (DECL)
+#define FIELD_TRANSIENT(DECL) DECL_LANG_FLAG_5 (DECL)
+
+/* Initialized flag on variable/field/parm decl */
+
+#define INITIALIZED_P(DECL) DECL_LANG_FLAG_6 (DECL)
+
+/* Access flags etc for a class (a TYPE_DECL): */
+
+#define CLASS_PUBLIC(DECL) DECL_LANG_FLAG_1 (DECL)
+#define CLASS_FINAL(DECL) DECL_LANG_FLAG_3 (DECL)
+#define CLASS_INTERFACE(DECL) DECL_LANG_FLAG_4 (DECL)
+#define CLASS_ABSTRACT(DECL) DECL_LANG_FLAG_5 (DECL)
+#define CLASS_SUPER(DECL) DECL_LANG_FLAG_6 (DECL)
+
+/* The number of virtual methods in this class's dispatch table.
+ Does not include initial two dummy entries (one points to the
+ Class object, and the other is for G++ -fvtable-thunks compatibility). */
+#define TYPE_NVIRTUALS(TYPE) TYPE_BINFO_VIRTUALS (TYPE)
+
+/* A TREE_VEC (indexed by DECL_VINDEX) containing this class's
+   virtual methods. */
+#define TYPE_VTABLE(TYPE) TYPE_BINFO_VTABLE(TYPE)
+
+/* True of a RECORD_TYPE of a class/interface type (not array type) */
+#define CLASS_P(TYPE) TYPE_LANG_FLAG_4 (TYPE)
+/* Use CLASS_LOADED_P? FIXME */
+#define CLASS_COMPLETE_P(DECL) DECL_LANG_FLAG_2 (DECL) 
+
+/* This maps a bytecode offset (PC) to various flags,
+   listed below (starting with BCODE_). */
+extern char *instruction_bits;
+
+/* True iff the byte is the start of an instruction. */
+#define BCODE_INSTRUCTION_START 1
+
+/* True iff there is a jump to this location. */
+#define BCODE_JUMP_TARGET 2
+
+/* True iff there is a return to this location.
+   (I.e. the preceedng instruction was a call.) */
+#define BCODE_RETURN_TARGET 4
+
+/* True iff this is the start of an exception handler. */
+#define BCODE_EXCEPTION_TARGET 16
+
+/* True iff there is a jump to this location (and it needs a label). */
+#define BCODE_TARGET \
+  (BCODE_JUMP_TARGET|BCODE_RETURN_TARGET \
+   | BCODE_EXCEPTION_TARGET)
+
+/* True iff there is an entry in the linenumber table for this location. */
+#define BCODE_HAS_LINENUMBER 32
+
+/* True iff there is more than one entry in the linenumber table for
+   this location.  (This probably does not make much sense.)  */
+#define BCODE_HAS_MULTI_LINENUMBERS 64
+
+/* True if this instruction has been verified. */
+#define BCODE_VERIFIED 8
+
+/* A pointer to the line number table of the current method. */
+extern unsigned char *linenumber_table;
+/* The length (in items) of the line number table. */
+extern int linenumber_count;
+
+/* In type_map, means that slot is uninitialized or otherwise unusable. */
+#define TYPE_UNKNOWN NULL_TREE
+
+/* In type_map, means the second half of a 64-bit double or long. */
+#define TYPE_SECOND void_type_node
+
+/* In type_map, means the null type (i.e. type of a null reference). */ 
+#define TYPE_NULL ptr_type_node
+
+/* In a type map means the type the address subroutine return address. */
+#define TYPE_RETURN_ADDR return_address_type_node
+
+/* In a subroutine's return type map, indicates that the slot was neither
+   used nor set in the subroutine. */
+#define TYPE_UNUSED error_mark_node
+
+/* A array mapping variable/stack slot index to the type current
+   in that variable/stack slot.
+   TYPE_UNKNOWN, TYPE_SECOND, and TYPE_NULL are special cases. */
+extern tree *type_map;
+
+/* Map a stack index to the type currently in that slot. */
+#define stack_type_map (type_map+DECL_MAX_LOCALS(current_function_decl))
+
+/* True iff TYPE takes two variable/stack slots. */
+#define TYPE_IS_WIDE(TYPE) \
+  ((TYPE) == double_type_node || (TYPE) == long_type_node)
+
+/* True iff TYPE is a Java array type. */
+#define TYPE_ARRAY_P(TYPE) TYPE_LANG_FLAG_1 (TYPE)
+
+/* If FUNCTION_TYPE or METHOD_TYPE: cache for build_java_argument_signature. */
+#define TYPE_ARGUMENT_SIGNATURE(TYPE) TYPE_VFIELD(TYPE)
+
+/* Given an array type, give the type of the elements. */
+/* FIXME this use of TREE_TYPE conflicts with something or other. */
+#define TYPE_ARRAY_ELEMENT(ATYPE) TREE_TYPE(ATYPE)
+
+/* True if class TYPE has been loaded. */
+#define CLASS_LOADED_P(TYPE) TYPE_LANG_FLAG_2 (TYPE)
+
+/* True if class TYPE was defined in Java source code. */
+#define CLASS_FROM_SOURCE_P(TYPE) TYPE_LANG_FLAG_3 (TYPE)
+
+/* True if identifier ID was seen while processing a single type import stmt */
+#define IS_A_SINGLE_IMPORT_CLASSFILE_NAME_P(ID) TREE_LANG_FLAG_0 (ID)
+
+/* True if identifier ID was seen while processing an import statement */
+#define IS_A_CLASSFILE_NAME(ID) TREE_LANG_FLAG_1 (ID)
+
+/* True if ID is a qualified named (contains . or /) */
+#define QUALIFIED_P(ID) TREE_LANG_FLAG_2 (ID)
+
+/* True if ID is an already processed import on demand */
+#define IS_AN_IMPORT_ON_DEMAND_P(ID) TREE_LANG_FLAG_3 (ID)
+
+/* True if EXPR is RHS sub-tree of a compound assign expression */
+#define COMPOUND_ASSIGN_P(EXPR) TREE_LANG_FLAG_1 (EXPR)
+
+/* True if EXPR (a WFL in that case) was created after the
+   reduction of PRIMARY . XXX */
+#define PRIMARY_P(EXPR) TREE_LANG_FLAG_2 (EXPR)
+
+/* True if EXPR (a MODIFY_EXPR in that case) is the result of variable
+   initialization during its declaration */
+#define MODIFY_EXPR_FROM_INITIALIZATION_P(EXPR) TREE_LANG_FLAG_2 (EXPR)
+
+/* True if EXPR (a WFL in that case) resolves into an expression name */
+#define RESOLVE_EXPRESSION_NAME_P(WFL) TREE_LANG_FLAG_0 (WFL)
+
+/* True if EXPR (a LOOP_EXPR in that case) is part of a for statement */
+#define IS_FOR_LOOP_P(EXPR) TREE_LANG_FLAG_0 (EXPR)
+
+/* True if EXPR (a WFL in that case) resolves into a package name */
+#define RESOLVE_PACKAGE_NAME_P(WFL) TREE_LANG_FLAG_3 (WFL)
+
+/* True if EXPR (a WFL in that case) resolves into a type name */
+#define RESOLVE_TYPE_NAME_P(WFL) TREE_LANG_FLAG_4 (WFL)
+
+/* True if STMT (a WFL in that case) holds a BREAK statement */
+#define IS_BREAK_STMT_P(WFL) TREE_LANG_FLAG_5 (WFL)
+
+/* Add a FIELD_DECL to RECORD_TYPE RTYPE.
+   The field has name NAME (a char*), and type FTYPE.
+   Unless this is the first field, FIELD most hold the previous field.
+   FIELD is set to the newly created FIELD_DECL.
+
+   We set DECL_ARTIFICIAL so these fields get skipped by make_class_data
+   if compiling java.lang.Object or java.lang.Class. */
+
+#define PUSH_FIELD(RTYPE, FIELD, NAME, FTYPE) \
+{ tree tmp_field = build_decl (FIELD_DECL, get_identifier(NAME), FTYPE); \
+  if (TYPE_FIELDS (RTYPE) == NULL_TREE) TYPE_FIELDS (RTYPE) = tmp_field; \
+  else TREE_CHAIN(FIELD) = tmp_field; \
+  DECL_CONTEXT (tmp_field) = RTYPE; \
+  DECL_ARTIFICIAL (tmp_field) = 1; \
+  FIELD = tmp_field; }
+
+#define FINISH_RECORD(RTYPE) layout_type (RTYPE)
+
+/* Start building a RECORD_TYPE constructor with a given TYPE in CONS. */
+#define START_RECORD_CONSTRUCTOR(CONS, CTYPE) { \
+  CONS = build (CONSTRUCTOR, CTYPE, NULL_TREE, NULL_TREE);\
+  TREE_CHAIN(CONS) = TYPE_FIELDS (CTYPE); }
+
+/* Append a field initializer to CONS for the dummy field for the inherited
+   fields.  The dummy field has the given VALUE, and the same type as the
+   super-class.   Must be specified before calls to PUSH_FIELD_VALUE. */
+
+#define PUSH_SUPER_VALUE(CONS, VALUE) {\
+  tree field = TREE_CHAIN(CONS);\
+  if (DECL_NAME (field) != NULL_TREE) abort();\
+  CONSTRUCTOR_ELTS(CONS) = tree_cons (field, VALUE, CONSTRUCTOR_ELTS(CONS));\
+  TREE_CHAIN(CONS) = TREE_CHAIN (field); }
+
+/* Append a field initializer to CONS for a field with the given VALUE.
+   NAME is a char* string used for error checking;
+   the initializer must be specified in order. */
+#define PUSH_FIELD_VALUE(CONS, NAME, VALUE) {\
+  tree field = TREE_CHAIN(CONS);\
+  if (strcmp (IDENTIFIER_POINTER (DECL_NAME (field)), NAME) != 0) abort();\
+  CONSTRUCTOR_ELTS(CONS) = tree_cons (field, VALUE, CONSTRUCTOR_ELTS(CONS));\
+  TREE_CHAIN(CONS) = TREE_CHAIN (field); }
+
+/* Finish creating a record CONSTRUCTOR CONS. */
+#define FINISH_RECORD_CONSTRUCTOR(CONS) \
+  CONSTRUCTOR_ELTS(CONS) = nreverse (CONSTRUCTOR_ELTS(CONS))
+
+/* New tree code for expression, so we can expand then individually. */
+#define JAVA_UNARY_PLUS_EXPR ((int)LAST_AND_UNUSED_TREE_CODE + 2)
+#define JAVA_NEW_ARRAY_EXPR  ((int)LAST_AND_UNUSED_TREE_CODE + 3)
+#define JAVA_NEW_CLASS_EXPR  ((int)LAST_AND_UNUSED_TREE_CODE + 4)
+#define JAVA_THIS_EXPR       ((int)LAST_AND_UNUSED_TREE_CODE + 5)
+
+/* Macro(s) using the definitions above */
+#define CALL_CONSTRUCTOR_P(NODE) (TREE_CODE (NODE) == JAVA_NEW_CLASS_EXPR)
+
diff --git a/gcc/java/javaop.def b/gcc/java/javaop.def
new file mode 100644 (file)
index 0000000..a722a8b
--- /dev/null
@@ -0,0 +1,314 @@
+/* Table of opcodes for byte codes defined by the Java(TM) virtual
+   machine specification.      
+   Copyright (C) 1998 Free Software Foundation, Inc.
+
+This file is part of GNU CC.
+
+GNU CC is free software; you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation; either version 2, or (at your option)
+any later version.
+
+GNU CC is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with GNU CC; see the file COPYING.  If not, write to
+the Free Software Foundation, 59 Temple Place - Suite 330,
+Boston, MA 02111-1307, USA.
+
+Java and all Java-based marks are trademarks or registered trademarks
+of Sun Microsystems, Inc. in the United States and other countries.
+The Free Software Foundation is independent of Sun Microsystems, Inc.
+
+Written by Per Bothner <bothner@cygnus.com>, February 1996.
+*/
+
+/* JAVAOP (OPNAME, OPCODE, OPKIND, OPERAND_TYPE, OPERAND_VALUE), where:
+       OPNAME is the name of the opcode.
+       OPCODE is the 1-byte opcode value.
+       OPKIND is the kind of operation.
+       OPERAND_TYPE is the type operands affected.
+       OPERAND_VALUE tells where to get the value.  (Its meaning
+         depends on OPKIND.) */
+
+/* Various macro used in OPERAND_VALUE:
+IMMEDIATE_s1:  An immediate signed 1-byte value in the byte-code stream.
+IMMEDIATE_s2:  An immediate signed 2-byte value in the byte-code stream.
+IMMEDIATE_u1:  An immediate unsigned 1-byte value in the byte-code stream.
+IMMEDIATE_u2:  An immediate unsigned 2-byte value in the byte-code stream.
+
+CONST_INDEX_1:  An immediate unsigned 1-byte index into the constant pool.
+CONST_INDEX_2:  An immediate unsigned 2-byte index into the constant pool.
+*/
+
+/* More specifically, this is the meaning of the various OPKINDs:
+
+BINOP: binary operation
+       OPERAND_TYPE is the argument type.
+       OPERAND_VALUE is PLUS, MINUS, MULT, TRUNC_DIV, RDIV, REM,
+         LSHIFT, RSHIFT, URSHIFT, BIT_AND, BIT_IOR, BIT_XOR,
+         COMPARE, COMPARE_L, or COMPARE_G.
+
+UNOP:  unary operation
+       OPERAND_TYPE is the argument type.
+       OPERAND_VALUE is NEG.
+
+INVOKE: invoke operations
+       OPERAND_TYPE is VIRTUAL, SPECIAL, STATIC, or INTERFACE.
+       OPERAND_VALUE is 1 if interface, 0 otherwise.
+
+OBJECT:        new, checkcast, instanceof
+       OPERAND_TYPE is PTR.
+       OPERAND_VALUE is NEW, CHECKCAST, or INSTANCEOF.
+
+BRANCH: branch operations
+       OPERAND_TYPE is GOTO, CALL, or RETURN.
+       OPERAND_VALUE is IMMEDIATE_s2, VAR_INDEX_1, IMMEDIATE_s4,
+         or VAR_INDEX_2.
+
+STACK:  Various stack operations.
+
+PUSHC:  Push a constant onto the operand stack.
+       OPERAND_VALUE maybe be the value to push,
+       or IMMEDIATE_s1 or IMMEDIATE_s2 if the constant is immediate,
+       or CONST_INDEX_1 or CONST_INDEX_2 for a constant pool index.
+
+LOAD: Push a value from a local variable onto the operand stack.
+       OPERAND_VALUE is the index of the local variable in the current
+       Java frame.  It can be a literal, or IMMEDIATE_i2.
+
+CONVERT: Convert top of stack value from one type to another.
+       OPERAND_TYPE is the argument type.
+       OPERAND_VALUE is the result type.
+
+TEST: Compares an integer (popped from the stack) against zero.
+       If the test (in OPERAND_VALUE) is true, goto a relative
+       offset given by the next two bytes.
+
+COND:  Compares two values (popped from the stack) againt each other.
+       If the test (in OPERAND_VALUE) is true, goto a relative
+       offset given by the next two bytes.
+
+SWITCH:
+       OPERAND_VALUE is either TABLE or LOOKUP.
+
+ARRAY:
+       OPERAND_VALUE is LOAD, STORE, LENGTH, or NEW.
+
+FIELD: Extracts from or stores into a field.
+       OPERAND_TYPE is 1 for a static field, 0 for a regular field.
+       OPERAND_VALUE is 1 for a put operation, 0 for a get operation.
+
+SPECIAL:
+       Random bunch of opcodes.
+
+*/
+
+JAVAOP (nop,             0, STACK,   POP,    0)
+JAVAOP (aconst_null,     1, PUSHC,   PTR,    0)
+JAVAOP (iconst_m1,       2, PUSHC,   INT,    -1)
+JAVAOP (iconst_0,        3, PUSHC,   INT,    0)
+JAVAOP (iconst_1,        4, PUSHC,   INT,    1)
+JAVAOP (iconst_2,        5, PUSHC,   INT,    2)
+JAVAOP (iconst_3,        6, PUSHC,   INT,    3)
+JAVAOP (iconst_4,        7, PUSHC,   INT,    4)
+JAVAOP (iconst_5,        8, PUSHC,   INT,    5)
+JAVAOP (lconst_0,        9, PUSHC,   LONG,   0)
+JAVAOP (lconst_1,       10, PUSHC,   LONG,   1)
+JAVAOP (fconst_0,       11, PUSHC,   FLOAT,  0)
+JAVAOP (fconst_1,       12, PUSHC,   FLOAT,  1)
+JAVAOP (fconst_2,       13, PUSHC,   FLOAT,  2)
+JAVAOP (dconst_0,       14, PUSHC,   DOUBLE, 0)
+JAVAOP (dconst_1,       15, PUSHC,   DOUBLE, 1)
+JAVAOP (bipush,         16, PUSHC,   INT,    IMMEDIATE_s1)
+JAVAOP (sipush,         17, PUSHC,   INT,    IMMEDIATE_s2)
+JAVAOP (ldc,            18, PUSHC,   INT,    CONST_INDEX_1)
+JAVAOP (ldc_w,          19, PUSHC,   INT,    CONST_INDEX_2)
+JAVAOP (ldc2_w,         20, PUSHC,   LONG,   CONST_INDEX_2)
+JAVAOP (iload,          21, LOAD,    INT,    IMMEDIATE_u1)
+JAVAOP (lload,          22, LOAD,    LONG,   IMMEDIATE_u1)
+JAVAOP (fload,          23, LOAD,    FLOAT,  IMMEDIATE_u1)
+JAVAOP (dload,          24, LOAD,    DOUBLE, IMMEDIATE_u1)
+JAVAOP (aload,          25, LOAD,    PTR,    IMMEDIATE_u1)
+JAVAOP (iload_0,        26, LOAD,    INT,    0)
+JAVAOP (iload_1,        27, LOAD,    INT,    1)
+JAVAOP (iload_2,        28, LOAD,    INT,    2)
+JAVAOP (iload_3,        29, LOAD,    INT,    3)
+JAVAOP (lload_0,        30, LOAD,    LONG,   0)
+JAVAOP (lload_1,        31, LOAD,    LONG,   1)
+JAVAOP (lload_2,        32, LOAD,    LONG,   2)
+JAVAOP (lload_3,        33, LOAD,    LONG,   3)
+JAVAOP (fload_0,        34, LOAD,    FLOAT,  0)
+JAVAOP (fload_1,        35, LOAD,    FLOAT,  1)
+JAVAOP (fload_2,        36, LOAD,    FLOAT,  2)
+JAVAOP (fload_3,        37, LOAD,    FLOAT,  3)
+JAVAOP (dload_0,        38, LOAD,    DOUBLE, 0)
+JAVAOP (dload_1,        39, LOAD,    DOUBLE, 1)
+JAVAOP (dload_2,        40, LOAD,    DOUBLE, 2)
+JAVAOP (dload_3,        41, LOAD,    DOUBLE, 3)
+JAVAOP (aload_0,        42, LOAD,    PTR,    0)
+JAVAOP (aload_1,        43, LOAD,    PTR,    1)
+JAVAOP (aload_2,        44, LOAD,    PTR,    2)
+JAVAOP (aload_3,        45, LOAD,    PTR,    3)
+JAVAOP (iaload,         46, ARRAY,   INT,    LOAD)
+JAVAOP (laload,         47, ARRAY,   LONG,   LOAD)
+JAVAOP (faload,         48, ARRAY,   FLOAT,  LOAD)
+JAVAOP (daload,         49, ARRAY,   DOUBLE, LOAD)
+JAVAOP (aaload,         50, ARRAY,   PTR,    LOAD)
+JAVAOP (baload,         51, ARRAY,   BYTE,   LOAD)
+JAVAOP (caload,         52, ARRAY,   CHAR,   LOAD)
+JAVAOP (saload,         53, ARRAY,   SHORT,  LOAD)
+JAVAOP (istore,         54, STORE,   INT,    IMMEDIATE_u1)
+JAVAOP (lstore,         55, STORE,   LONG,   IMMEDIATE_u1)
+JAVAOP (fstore,         56, STORE,   FLOAT,  IMMEDIATE_u1)
+JAVAOP (dstore,         57, STORE,   DOUBLE, IMMEDIATE_u1)
+JAVAOP (astore,         58, STORE,   PTR,    IMMEDIATE_u1)
+JAVAOP (istore_0,       59, STORE,   INT,    0)
+JAVAOP (istore_1,       60, STORE,   INT,    1)
+JAVAOP (istore_2,       61, STORE,   INT,    2)
+JAVAOP (istore_3,       62, STORE,   INT,    3)
+JAVAOP (lstore_0,       63, STORE,   LONG,   0)
+JAVAOP (lstore_1,       64, STORE,   LONG,   1)
+JAVAOP (lstore_2,       65, STORE,   LONG,   2)
+JAVAOP (lstore_3,       66, STORE,   LONG,   3)
+JAVAOP (fstore_0,       67, STORE,   FLOAT,  0)
+JAVAOP (fstore_1,       68, STORE,   FLOAT,  1)
+JAVAOP (fstore_2,       69, STORE,   FLOAT,  2)
+JAVAOP (fstore_3,       70, STORE,   FLOAT,  3)
+JAVAOP (dstore_0,       71, STORE,   DOUBLE, 0)
+JAVAOP (dstore_1,       72, STORE,   DOUBLE, 1)
+JAVAOP (dstore_2,       73, STORE,   DOUBLE, 2)
+JAVAOP (dstore_3,       74, STORE,   DOUBLE, 3)
+JAVAOP (astore_0,       75, STORE,   PTR,    0)
+JAVAOP (astore_1,       76, STORE,   PTR,    1)
+JAVAOP (astore_2,       77, STORE,   PTR,    2)
+JAVAOP (astore_3,       78, STORE,   PTR,    3)
+JAVAOP (iastore,        79, ARRAY,   INT,    STORE)
+JAVAOP (lastore,        80, ARRAY,   LONG,   STORE)
+JAVAOP (fastore,        81, ARRAY,   FLOAT,  STORE)
+JAVAOP (dastore,        82, ARRAY,   DOUBLE, STORE)
+JAVAOP (aastore,        83, ARRAY,   PTR,    STORE)
+JAVAOP (bastore,        84, ARRAY,   BYTE,   STORE)
+JAVAOP (castore,        85, ARRAY,   CHAR,   STORE)
+JAVAOP (sastore,        86, ARRAY,   SHORT,  STORE)
+JAVAOP (pop,            87, STACK,   POP,    1)
+JAVAOP (pop2,           88, STACK,   POP,    2)
+JAVAOP (dup,            89, STACK,   DUP,    1)
+JAVAOP (dup_x1,         90, STACK,   DUPx1,  1)
+JAVAOP (dup_x2,         91, STACK,   DUPx2,  1)
+JAVAOP (dup2,           92, STACK,   DUP,    2)
+JAVAOP (dup2_x1,        93, STACK,   DUPx1,  2)
+JAVAOP (dup2_x2,        94, STACK,   DUPx2,  2)
+JAVAOP (swap,           95, STACK,   SWAP,   0)
+JAVAOP (iadd,           96, BINOP,   INT,    PLUS)
+JAVAOP (ladd,           97, BINOP,   LONG,   PLUS)
+JAVAOP (fadd,           98, BINOP,   FLOAT,  PLUS)
+JAVAOP (dadd,           99, BINOP,   DOUBLE, PLUS)
+JAVAOP (isub,          100, BINOP,   INT,    MINUS)
+JAVAOP (lsub,          101, BINOP,   LONG,   MINUS)
+JAVAOP (fsub,          102, BINOP,   FLOAT,  MINUS)
+JAVAOP (dsub,          103, BINOP,   DOUBLE, MINUS)
+JAVAOP (imul,          104, BINOP,   INT,    MULT)
+JAVAOP (lmul,          105, BINOP,   LONG,   MULT)
+JAVAOP (fmul,          106, BINOP,   FLOAT,  MULT)
+JAVAOP (dmul,          107, BINOP,   DOUBLE, MULT)
+JAVAOP (idiv,          108, BINOP,   INT,    TRUNC_DIV)
+JAVAOP (ldiv,          109, BINOP,   LONG,   TRUNC_DIV)
+JAVAOP (fdiv,          110, BINOP,   FLOAT,  RDIV)
+JAVAOP (ddiv,          111, BINOP,   DOUBLE, RDIV)
+JAVAOP (irem,          112, BINOP,   INT,    REM)
+JAVAOP (lrem,          113, BINOP,   LONG,   REM)
+JAVAOP (frem,          114, BINOP,   FLOAT,  REM)
+JAVAOP (drem,          115, BINOP,   DOUBLE, REM)
+JAVAOP (ineg,          116, UNOP,    INT,    NEG)
+JAVAOP (lneg,          117, UNOP,    LONG,   NEG)
+JAVAOP (fneg,          118, UNOP,    FLOAT,  NEG)
+JAVAOP (dneg,          119, UNOP,    DOUBLE, NEG)
+JAVAOP (ishl,          120, BINOP,   INT,    LSHIFT)
+JAVAOP (lshl,          121, BINOP,   LONG,   LSHIFT)
+JAVAOP (ishr,          122, BINOP,   INT,    RSHIFT)
+JAVAOP (lshr,          123, BINOP,   LONG,   RSHIFT)
+JAVAOP (iushr,         124, BINOP,   INT,    URSHIFT)
+JAVAOP (lushr,         125, BINOP,   LONG,   URSHIFT)
+JAVAOP (iand,          126, BINOP,   INT,    BIT_AND)
+JAVAOP (land,          127, BINOP,   LONG,   BIT_AND)
+JAVAOP (ior,           128, BINOP,   INT,    BIT_IOR)
+JAVAOP (lor,           129, BINOP,   LONG,   BIT_IOR)
+JAVAOP (ixor,          130, BINOP,   INT,    BIT_XOR)
+JAVAOP (lxor,          131, BINOP,   LONG,   BIT_XOR)
+JAVAOP (iinc,          132, SPECIAL, INT,    IINC)
+JAVAOP (i2l,           133, CONVERT, INT,    LONG)
+JAVAOP (i2f,           134, CONVERT, INT,    FLOAT)
+JAVAOP (i2d,           135, CONVERT, INT,    DOUBLE)
+JAVAOP (l2i,           136, CONVERT, LONG,   INT)
+JAVAOP (l2f,           137, CONVERT, LONG,   FLOAT)
+JAVAOP (l2d,           138, CONVERT, LONG,   DOUBLE)
+JAVAOP (f2i,           139, CONVERT, FLOAT,  INT)
+JAVAOP (f2l,           140, CONVERT, FLOAT,  LONG)
+JAVAOP (f2d,           141, CONVERT, FLOAT,  DOUBLE)
+JAVAOP (d2i,           142, CONVERT, DOUBLE, INT)
+JAVAOP (d2l,           143, CONVERT, DOUBLE, LONG)
+JAVAOP (d2f,           144, CONVERT, DOUBLE, FLOAT)
+JAVAOP (i2b,          145, CONVERT2, INT,    BYTE)
+JAVAOP (i2c,           146, CONVERT2, INT,    CHAR)
+JAVAOP (i2s,           147, CONVERT2, INT,    SHORT)
+JAVAOP (lcmp,          148, BINOP,   LONG,   COMPARE)
+JAVAOP (fcmpl,         149, BINOP,   FLOAT,  COMPARE_L)
+JAVAOP (fcmpg,         150, BINOP,   FLOAT,  COMPARE_G)
+JAVAOP (dcmpl,         151, BINOP,   DOUBLE, COMPARE_L)
+JAVAOP (dcmpg,         152, BINOP,   DOUBLE, COMPARE_G)
+JAVAOP (ifeq,          153, TEST,    INT,    EQ)
+JAVAOP (ifne,          154, TEST,    INT,    NE)
+JAVAOP (iflt,          155, TEST,    INT,    LT)
+JAVAOP (ifge,          156, TEST,    INT,    GE)
+JAVAOP (ifgt,          157, TEST,    INT,    GT)
+JAVAOP (ifle,          158, TEST,    INT,    LE)
+JAVAOP (if_icmpeq,     159, COND,    INT,    EQ)
+JAVAOP (if_icmpne,     160, COND,    INT,    NE)
+JAVAOP (if_icmplt,     161, COND,    INT,    LT)
+JAVAOP (if_icmpge,     162, COND,    INT,    GE)
+JAVAOP (if_icmpgt,     163, COND,    INT,    GT)
+JAVAOP (if_icmple,     164, COND,    INT,    LE)
+JAVAOP (if_acmpeq,     165, COND,    PTR,    EQ)
+JAVAOP (if_acmpne,     166, COND,    PTR,    NE)
+JAVAOP (goto,          167, BRANCH,  GOTO,   IMMEDIATE_s2)
+JAVAOP (jsr,           168, JSR,     CALL,   IMMEDIATE_s2)
+JAVAOP (ret,           169, RET,     RETURN, VAR_INDEX_1)
+JAVAOP (tableswitch,   170, SWITCH,  INT,    TABLE)
+JAVAOP (lookupswitch,  171, SWITCH,  INT,    LOOKUP)
+JAVAOP (ireturn,       172, RETURN,  INT,    0)
+JAVAOP (lreturn,       173, RETURN,  LONG,   0)
+JAVAOP (freturn,       174, RETURN,  FLOAT,  0)
+JAVAOP (dreturn,       175, RETURN,  DOUBLE, 0)
+JAVAOP (areturn,       176, RETURN,  PTR,    0)
+JAVAOP (return,        177, RETURN,  VOID,   0)
+JAVAOP (getstatic,     178, FIELD,   1,      0)
+JAVAOP (putstatic,     179, FIELD,   1,      1)
+JAVAOP (getfield,      180, FIELD,   0,      0)
+JAVAOP (putfield,      181, FIELD,   0,      1)
+JAVAOP (invokevirtual, 182, INVOKE,  VIRTUAL,0)
+JAVAOP (invokespecial, 183, INVOKE,  SPECIAL, 0)
+JAVAOP (invokestatic,  184, INVOKE,  STATIC, 0)
+JAVAOP (invokeinterface,185, INVOKE, INTERFACE, 1)
+JAVAOP (new,           187, OBJECT,  PTR,    NEW)
+JAVAOP (newarray,      188, ARRAY,   NUM,    NEW)
+JAVAOP (anewarray,     189, ARRAY,   PTR,    NEW)
+JAVAOP (arraylength,   190, ARRAY,   INT,    LENGTH)
+JAVAOP (athrow,        191, SPECIAL, ANY,    THROW)
+JAVAOP (checkcast,     192, OBJECT,  PTR,    CHECKCAST)
+JAVAOP (instanceof,    193, OBJECT,  PTR,    INSTANCEOF)
+JAVAOP (monitorenter,  194, SPECIAL, MONITOR, ENTER)
+JAVAOP (monitorexit,   195, SPECIAL, MONITOR, EXIT)
+JAVAOP (wide,          196, SPECIAL, ANY,    WIDE)
+JAVAOP (multianewarray,197, ARRAY,   MULTI,  NEW)
+JAVAOP (ifnull,        198, TEST,    PTR,    EQ)
+JAVAOP (ifnonnull,     199, TEST,    PTR,    NE)
+JAVAOP (goto_w,        200, BRANCH,  GOTO,   IMMEDIATE_s4)
+JAVAOP (jsr_w,         201, JSR,     CALL,   IMMEDIATE_s4)
+JAVAOP (breakpoint,    202, SPECIAL, ANY,    BREAK)
+JAVAOP (ret_w,         209, RET,     RETURN, VAR_INDEX_2)
+JAVAOP (impdep1,       254, IMPL,    ANY,    1)
+JAVAOP (impdep2,       255, IMPL,    ANY,    2)
diff --git a/gcc/java/javaop.h b/gcc/java/javaop.h
new file mode 100644 (file)
index 0000000..f4ae050
--- /dev/null
@@ -0,0 +1,142 @@
+/* Utility macros to handle Java(TM) byte codes.
+
+   Copyright (C) 1996  Free Software Foundation, Inc.
+
+This program is free software; you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation; either version 2, or (at your option)
+any later version.
+
+This program is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with GNU CC; see the file COPYING.  If not, write to
+the Free Software Foundation, 59 Temple Place - Suite 330,
+Boston, MA 02111-1307, USA.  
+
+Java and all Java-based marks are trademarks or registered trademarks
+of Sun Microsystems, Inc. in the United States and other countries.
+The Free Software Foundation is independent of Sun Microsystems, Inc.  */
+
+/* Written by Per Bothner <bothner@cygnus.com>, February 1996. */
+
+#ifndef JAVAOP_H
+#define JAVAOP_H
+
+typedef        char            int8;
+typedef        unsigned char   uint8;
+#ifndef int16
+#define int16 short
+#endif
+typedef unsigned int16 uint16;
+
+#ifndef int32
+#define int32 long
+#endif
+typedef unsigned int32 uint32;
+
+/* A signed 64-bit (or more) integral type, suiteable for Java's 'long'. */
+#ifndef int64
+#define int64 long long
+#endif
+/* An unsigned 64-bit (or more) integral type, same length as int64. */
+#ifndef uint64
+#define uint64 unsigned int64
+#endif
+
+typedef uint16                 jchar;
+typedef int8                   jbyte;
+typedef int16                   jshort;
+typedef int32                   jint;
+typedef int64                   jlong;
+typedef void*                   jref;
+
+/* A 32-bit IEEE single-precision float. */
+#ifndef jfloat 
+#define jfloat float
+#endif
+
+/* A 32-bit IEEE double-precision float. */
+#ifndef jdouble
+#define jdouble double
+#endif
+
+union Word {
+  jint i;
+  jfloat f;
+  void *p;
+};
+
+/* A jword is an unsigned integral type big enough for a 32-bit jint
+   or jfloat *or* a pointer.  It is the type appropriate for stack
+   locations and local variables in a Java interpreter. */
+
+
+#ifndef jword
+#define jword uint32
+#endif
+
+#if !defined(inline) && !defined(__GC__) && !defined(__cplusplus)
+#define inline static
+#endif
+
+#ifndef IMMEDIATE_u1
+#define IMMEDIATE_u1 (PC++, CHECK_PC_IN_RANGE(PC), BCODE[PC-1])
+#endif
+#ifndef IMMEDIATE_s1
+#define IMMEDIATE_s1 (PC++, CHECK_PC_IN_RANGE(PC), (signed char)BCODE[PC-1])
+#endif
+#ifndef IMMEDIATE_s2
+#define IMMEDIATE_s2 (PC+=2, CHECK_PC_IN_RANGE(PC), \
+  (signed char) BCODE[PC-2] * 256 + BCODE[PC-1])
+#endif
+#ifndef IMMEDIATE_u2
+#define IMMEDIATE_u2 (PC+=2, CHECK_PC_IN_RANGE(PC),\
+  (BCODE[PC-2] * 256 + BCODE[PC-1]))
+#endif
+#ifndef IMMEDIATE_s4
+#define IMMEDIATE_s4 (PC+=4, CHECK_PC_IN_RANGE(PC), \
+  ((jint)((BCODE[PC-4] << 24) | (BCODE[PC-3] << 16) \
+         | (BCODE[PC-2] << 8) | (BCODE[PC-1]))))
+#endif
+
+inline jfloat
+WORD_TO_FLOAT(jword w)
+{ union Word wu;
+  wu.i = w;
+  return wu.f;
+} 
+
+inline jlong
+WORDS_TO_LONG(jword hi, jword lo)
+{
+  return ((jlong) hi << 32) | ((jlong)lo & (((jlong)1 << 32) -1));
+}
+
+union DWord {
+  jdouble d;
+  jlong l;
+  jword w[2];
+};
+
+inline jdouble
+WORDS_TO_DOUBLE(jword hi, jword lo)
+{ union DWord wu;
+  wu.l = WORDS_TO_LONG(hi, lo);
+  return wu.d;
+} 
+
+/* If PREFIX_CHAR is the first character of the Utf8 encoding of a character,
+   return the number of bytes taken by the encoding.
+   Return -1 for a continuation character.  */
+#define UT8_CHAR_LENGTH(PREFIX_CHAR) \
+  ((unsigned char)(PREFIX_CHAR) < 128 ? 1 \
+   : ((PREFIX_CHAR) & 0x40) == 0 ? -1 \
+   : ((PREFIX_CHAR) & 0x20) == 0 ? 2 \
+   : ((PREFIX_CHAR) & 0x10) == 0 ? 3 \
+   : ((PREFIX_CHAR) & 0x08) == 0 ? 4 : 5)
+
+#endif /* !JAVAOP_H */
diff --git a/gcc/java/jcf-dump.c b/gcc/java/jcf-dump.c
new file mode 100644 (file)
index 0000000..749aa82
--- /dev/null
@@ -0,0 +1,1072 @@
+/* Program to dump out a Java(TM) .class file.
+   Functionally similar to Sun's javap.
+
+   Copyright (C) 1996, 1997, 1998 Free Software Foundation, Inc.
+
+This program is free software; you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation; either version 2, or (at your option)
+any later version.
+
+This program is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with GNU CC; see the file COPYING.  If not, write to
+the Free Software Foundation, 59 Temple Place - Suite 330,
+Boston, MA 02111-1307, USA.  
+
+Java and all Java-based marks are trademarks or registered trademarks
+of Sun Microsystems, Inc. in the United States and other countries.
+The Free Software Foundation is independent of Sun Microsystems, Inc.  */
+
+/* Written by Per Bothner <bothner@cygnus.com>, February 1996. */
+
+/*
+  jcf-dump is a program to print out the contents of class files.
+  Usage:  jcf-dump [FLAGS] CLASS
+  Each CLASS is either:
+  + the name of a class in the CLASSPATH (e.g "java.lang.String"), or
+  + the name of a class *file* (e.g. "/home/me/work/package/Foo.class").
+  + The name of a .zip or .jar file (which prints all the classes in the
+  archive).
+
+  OPTIONS:
+  -c
+       Dis-assemble each method.
+  -classpath PATH
+       Overrides $CLASSPATH.
+  --print-main
+       Print nothing if there is no valid "main" method;
+       otherwise, print only the class name.
+  --javap
+       Print output in the style of Sun's javap program.  VERY UNFINISHED.
+ */
+    
+
+#include <stdio.h>
+#include "jcf.h"
+
+/* Outout file. */
+FILE *out;
+/* Name of output file, if NULL if stdout. */
+char *output_file = NULL;
+
+int verbose = 0;
+
+int flag_disassemble_methods = 0;
+int flag_print_class_info = 1;
+int flag_print_constant_pool = 1;
+int flag_print_fields = 1;
+int flag_print_methods = 1;
+int flag_print_attributes = 1;
+
+/* Print names of classes that have a "main" method. */
+int flag_print_main = 0;
+
+/* Index in constant pool of this class. */
+int this_class_index = 0;
+
+int class_access_flags = 0;
+
+/* Print in format similar to javap.  VERY IMCOMPLETE. */
+int flag_javap_compatible = 0;
+
+static int print_access_flags PROTO ((FILE *, uint16));
+static void print_constant_terse PROTO ((FILE*, JCF*, int, int));
+static void print_constant PROTO ((FILE *, JCF *, int, int));
+static void print_constant_ref PROTO ((FILE *, JCF *, int));
+static void disassemble_method PROTO ((JCF*, unsigned char *, int));
+static void print_name PROTO ((FILE*, JCF*, int));
+static void print_signature PROTO ((FILE*, JCF*, int, int));
+
+#define PRINT_SIGNATURE_RESULT_ONLY 1
+#define PRINT_SIGNATURE_ARGS_ONLY 2
+
+extern char* open_class();
+
+int
+DEFUN(utf8_equal_string, (jcf, index, value),
+      JCF *jcf AND int index AND char * value)
+{
+  if (CPOOL_INDEX_IN_RANGE (&jcf->cpool, index)
+      && JPOOL_TAG (jcf, index) == CONSTANT_Utf8)
+    {
+      int len = strlen (value);
+      if (JPOOL_UTF_LENGTH (jcf, index) == len
+         && memcmp (JPOOL_UTF_DATA (jcf, index), value, len) == 0)
+       return 1;
+    }
+  return 0;
+}
+
+#define HANDLE_MAGIC(MAGIC, MINOR, MAJOR) \
+  this_class_index = 0; \
+  if (flag_print_class_info) \
+    fprintf (out, \
+             "Magic number: 0x%0x, minor_version: %d, major_version: %d.\n", \
+            MAGIC, MINOR, MAJOR)
+
+#define HANDLE_START_CONSTANT_POOL(COUNT) \
+  if (flag_print_constant_pool) \
+    fprintf (out, "\nConstant pool (count: %d):\n", COUNT)
+
+#define HANDLE_SOURCEFILE(INDEX) \
+{ fprintf (out, "Attribute "); \
+  print_constant_terse (out, jcf, attribute_name, CONSTANT_Utf8); \
+  fprintf (out, ", length:%d, #%d=", attribute_length, INDEX); \
+  print_constant_terse (out, jcf, INDEX, CONSTANT_Utf8); fputc ('\n', out); }
+
+#define HANDLE_CLASS_INFO(ACCESS_FLAGS, THIS, SUPER, INTERFACES_COUNT) \
+  this_class_index = THIS; \
+  class_access_flags = ACCESS_FLAGS; \
+  if (flag_print_class_info) \
+    { fprintf (out, "\nAccess flags: 0x%x", ACCESS_FLAGS); \
+      print_access_flags (out, ACCESS_FLAGS); \
+      fputc ('\n', out); \
+      fprintf (out, "This class: "); \
+      if (flag_print_constant_pool) \
+        fprintf (out, "%d=", THIS); \
+      print_constant_terse (out, jcf, THIS, CONSTANT_Class); \
+      if (flag_print_constant_pool || SUPER != 0) \
+        fprintf (out, ", super: "); \
+      if (flag_print_constant_pool) \
+        { \
+          fprintf (out, "%d", SUPER); \
+          if (SUPER != 0) \
+            fputc ('=', out); \
+        } \
+      if (SUPER != 0) \
+        print_constant_terse (out, jcf, SUPER, CONSTANT_Class); \
+      fprintf (out, "\nInterfaces (count: %d):\n", INTERFACES_COUNT); \
+    }
+
+#define IGNORE_ATTRIBUTE(JCF, NAME, NAME_LENGTH) \
+  (flag_print_attributes <= 0)
+
+#define HANDLE_CLASS_INTERFACE(INDEX) \
+  if (flag_print_class_info) \
+    { fprintf (out, "- Implements: %d=", INDEX); \
+      print_constant_terse (out, jcf, INDEX, CONSTANT_Class); \
+      fputc ('\n', out); }
+
+#define HANDLE_START_FIELDS(FIELDS_COUNT) \
+  if (flag_print_fields) \
+    fprintf (out, "\nFields (count: %d):\n", FIELDS_COUNT)
+
+#define HANDLE_START_FIELD(ACCESS_FLAGS, NAME, SIGNATURE, ATTRIBUTE_COUNT) \
+  if (flag_print_fields) \
+    { fprintf (out, "Field name:"); \
+      print_constant_terse (out, jcf, NAME, CONSTANT_Utf8); \
+      print_access_flags (out, ACCESS_FLAGS); \
+      fprintf (out, " Signature: "); \
+      if (flag_print_constant_pool) \
+        fprintf (out, "%d=", SIGNATURE); \
+      print_signature (out, jcf, SIGNATURE, 0); \
+      fputc ('\n', out); } \
+  else \
+    flag_print_attributes--;
+
+#define HANDLE_END_FIELD() \
+  if (! flag_print_fields) \
+    flag_print_attributes++;
+
+#define HANDLE_START_METHODS(METHODS_COUNT) \
+  if (flag_print_methods) \
+    fprintf (out, "\nMethods (count: %d):\n", METHODS_COUNT); \
+  else \
+    flag_print_attributes--;
+
+
+#define HANDLE_END_METHODS() \
+  if (! flag_print_methods) \
+    flag_print_attributes++;
+
+#define HANDLE_METHOD(ACCESS_FLAGS, NAME, SIGNATURE, ATTRIBUTE_COUNT) \
+{ \
+  if (flag_print_methods) \
+    { \
+      if (flag_javap_compatible) \
+        { \
+         fprintf (out, "    "); \
+         print_access_flags (out, ACCESS_FLAGS); \
+         fputc (' ', out); \
+         print_signature (out, jcf, SIGNATURE, PRINT_SIGNATURE_RESULT_ONLY); \
+         fputc (' ', out); \
+         print_constant_terse (out, jcf, NAME, CONSTANT_Utf8); \
+         print_signature (out, jcf, SIGNATURE, PRINT_SIGNATURE_ARGS_ONLY); \
+         fputc ('\n', out); \
+       } \
+      else \
+       { \
+         fprintf (out, "\nMethod name:"); \
+         print_constant_terse (out, jcf, NAME, CONSTANT_Utf8); \
+         print_access_flags (out, ACCESS_FLAGS); \
+         fprintf (out, " Signature: "); \
+         if (flag_print_constant_pool) \
+           fprintf (out, "%d=", SIGNATURE); \
+         print_signature (out, jcf, SIGNATURE, 0); \
+         fputc ('\n', out); \
+       } \
+    } \
+  if (flag_print_main && ACCESS_FLAGS == (ACC_STATIC|ACC_PUBLIC) \
+      && utf8_equal_string (jcf, NAME, "main") \
+      && utf8_equal_string (jcf, SIGNATURE, "([Ljava/lang/String;)V") \
+      && this_class_index > 0 \
+      && (class_access_flags & ACC_PUBLIC)) \
+    { \
+      print_constant_terse(out, jcf, this_class_index, CONSTANT_Class); \
+      fputc  ('\n', out); \
+   } \
+}
+
+#define COMMON_HANDLE_ATTRIBUTE(JCF, INDEX, LENGTH) \
+( fprintf (out, "Attribute "), \
+  print_constant_terse (out, jcf, INDEX, CONSTANT_Utf8), \
+  fprintf (out, ", length:%d", LENGTH) )
+
+#define HANDLE_CONSTANTVALUE(VALUE_INDEX) \
+( COMMON_HANDLE_ATTRIBUTE(JCF, attribute_name, attribute_length), \
+  fprintf (out, ", value: "), \
+  print_constant_ref (out, jcf, VALUE_INDEX), \
+  fprintf (out, "\n") )
+
+#define HANDLE_CODE_ATTRIBUTE(MAX_STACK, MAX_LOCALS, CODE_LENGTH) \
+{ COMMON_HANDLE_ATTRIBUTE(JCF, attribute_name, attribute_length); \
+  fprintf (out, ", max_stack:%d, max_locals:%d, code_length:%d\n", \
+    MAX_STACK, MAX_LOCALS, CODE_LENGTH); \
+  disassemble_method (jcf, jcf->read_ptr, CODE_LENGTH); }
+
+#define HANDLE_EXCEPTIONS_ATTRIBUTE(COUNT) \
+{ int n = (COUNT); int i; \
+  COMMON_HANDLE_ATTRIBUTE(JCF, attribute_name, attribute_length); \
+  fprintf (out, ", count: %d\n", n); \
+  for (i = 0; i < n; i++) {\
+    int ex_index = JCF_readu2 (jcf); \
+    fprintf (out, "%3d: ", i); \
+    print_constant_ref (out, jcf, ex_index); \
+    fputc ('\n', out); } }
+
+#define HANDLE_LOCALVARIABLETABLE_ATTRIBUTE(COUNT) \
+{ int n = (COUNT); int i; \
+  COMMON_HANDLE_ATTRIBUTE(JCF, attribute_name, attribute_length); \
+  fprintf (out, ", count: %d\n", n); \
+  for (i = 0; i < n; i++) {\
+    int start_pc = JCF_readu2 (jcf); \
+    int length = JCF_readu2 (jcf); \
+    int name_index = JCF_readu2 (jcf); \
+    int signature_index = JCF_readu2 (jcf); \
+    int slot = JCF_readu2 (jcf); \
+    fprintf (out, "  slot#%d: name: %d=", slot, name_index); \
+    print_name (out, jcf, name_index); \
+    fprintf (out, ", type: %d=", signature_index); \
+    print_signature (out, jcf, signature_index, 0); \
+    fprintf (out, " (pc: %d length: %d)\n", start_pc, length); }}
+
+#define HANDLE_LINENUMBERTABLE_ATTRIBUTE(COUNT) \
+{ int n = (COUNT); int i; \
+  COMMON_HANDLE_ATTRIBUTE(jcf, attribute_name, attribute_length); \
+  fprintf (out, ", count: %d\n", n); \
+  if (flag_disassemble_methods) \
+    for (i = 0; i < n; i++) {\
+      int start_pc = JCF_readu2 (jcf); \
+      int line_number = JCF_readu2 (jcf); \
+      fprintf (out, "  line: %d at pc: %d\n", line_number, start_pc); }\
+  else \
+    JCF_SKIP (jcf, 4 * n); }
+
+#define PROCESS_OTHER_ATTRIBUTE(JCF, INDEX, LENGTH) \
+{ COMMON_HANDLE_ATTRIBUTE(JCF, INDEX, LENGTH); \
+  fputc ('\n', out); JCF_SKIP (JCF, LENGTH); }
+
+#define START_FINAL_ATTRIBUTES(ATTRIBUTES_COUNT) \
+  if (flag_print_attributes > 0) \
+    fprintf (out, "\nAttributes (count: %d):\n", attributes_count);
+
+#include "javaop.h"
+#include "jcf-reader.c"
+
+static void
+DEFUN(print_constant_ref, (stream, jcf, index),
+      FILE *stream AND JCF *jcf AND int index)
+{
+  fprintf (stream, "#%d=<", index);
+  if (index <= 0 || index >= JPOOL_SIZE(jcf))
+    fprintf (stream, "out of range");
+  else
+    print_constant (stream, jcf, index, 1);
+  fprintf (stream, ">", index);
+}
+
+static int
+DEFUN (print_access_flags, (stream, flags),
+       FILE *stream AND uint16 flags)
+{
+  if (flags & ACC_PUBLIC) fprintf (stream, " public");
+  if (flags & ACC_PRIVATE) fprintf (stream, " private");
+  if (flags & ACC_PROTECTED) fprintf (stream, " protected");
+  if (flags & ACC_STATIC) fprintf (stream, " static");
+  if (flags & ACC_FINAL) fprintf (stream, " final");
+  if (flags & ACC_SYNCHRONIZED) fprintf (stream, " synchronized");
+  if (flags & ACC_VOLATILE) fprintf (stream, " volatile");
+  if (flags & ACC_TRANSIENT) fprintf (stream, " transient");
+  if (flags & ACC_NATIVE) fprintf (stream, " native");
+  if (flags & ACC_INTERFACE) fprintf (stream, " interface");
+  if (flags & ACC_ABSTRACT) fprintf (stream, " abstract");
+}
+
+
+static void
+DEFUN(print_name, (stream, jcf, name_index),
+      FILE* stream AND JCF* jcf AND int name_index)
+{
+  if (JPOOL_TAG (jcf, name_index) != CONSTANT_Utf8)
+    fprintf (stream, "<not a UTF8 constant>");
+  else
+    jcf_print_utf8 (stream, JPOOL_UTF_DATA (jcf,name_index),
+                   JPOOL_UTF_LENGTH (jcf, name_index));
+}
+
+/* If the type of the constant at INDEX matches EXPECTED,
+   print it tersely, otherwise more verbosely. */
+
+void
+DEFUN(print_constant_terse, (out, jcf, index, expected),
+      FILE *out AND JCF *jcf AND int index AND int expected)
+{
+  if (JPOOL_TAG (jcf, index) != expected)
+    {
+      fprintf (out, "<Unexpected constant type ");
+      print_constant (out, jcf, index, 1);
+      fprintf (out, ">");
+    }
+  else
+    print_constant (out, jcf, index, 0);
+}
+
+/* Print the constant at INDEX in JCF's constant pool.
+   If verbosity==0, print very tersely (no extraneous text).
+   If verbosity==1, prefix the type of the constant.
+   If verbosity==2, add more descriptive text. */
+
+static void
+DEFUN(print_constant, (out, jcf, index, verbosity),
+      FILE *out AND JCF *jcf AND int index AND int verbosity)
+{
+  int j, n;
+  jlong num;
+  char *str;
+  int kind = JPOOL_TAG (jcf, index);
+  switch (kind)
+    {
+    case CONSTANT_Class:
+      n = JPOOL_USHORT1 (jcf, index);
+      if (verbosity > 0)
+       fprintf (out, verbosity > 1 ? "Class name: %d=" : "Class ", n);
+      if (! CPOOL_INDEX_IN_RANGE (&jcf->cpool, n))
+       fprintf (out, "<out of range>");
+      else if (verbosity < 2 && JPOOL_TAG (jcf, n) == CONSTANT_Utf8)
+       {
+         int len = JPOOL_UTF_LENGTH (jcf, n);
+         jcf_print_utf8_replace (out, JPOOL_UTF_DATA(jcf,n), len, '/', '.');
+       }
+      else
+       print_constant_terse (out, jcf, n, CONSTANT_Utf8);
+      break;
+    case CONSTANT_Fieldref:
+      str = "Field"; goto field_or_method;
+    case CONSTANT_Methodref:
+      str = "Method"; goto field_or_method;
+    case CONSTANT_InterfaceMethodref:
+      str = "InterfaceMethod"; goto field_or_method;
+    field_or_method:
+      {
+       uint16 tclass = JPOOL_USHORT1 (jcf, index);
+       uint16 name_and_type = JPOOL_USHORT2 (jcf, index);
+       if (verbosity == 2)
+         fprintf (out, "%sref class: %d=", str, tclass);
+       else if (verbosity > 0)
+           fprintf (out, "%s ", str);
+       print_constant_terse (out, jcf, tclass, CONSTANT_Class);
+       fprintf (out, verbosity < 2 ? "." : " name_and_type: %d=<",
+                name_and_type);
+       print_constant_terse (out, jcf, name_and_type, CONSTANT_NameAndType);
+       if (verbosity == 2)
+         fputc ('>', out);
+      }
+      break;
+    case CONSTANT_String:
+      j = JPOOL_USHORT1 (jcf, index);
+      if (verbosity > 0)
+       fprintf (out, verbosity > 1 ? "String %d=" : "String ", j);
+      print_constant_terse (out, jcf, j, CONSTANT_Utf8);
+      break;
+    case CONSTANT_Integer:
+      if (verbosity > 0)
+       fprintf (out, "Integer ");
+      num = JPOOL_INT (jcf, index);
+      goto integer;
+    case CONSTANT_Long:
+      if (verbosity > 0)
+       fprintf (out, "Long ");
+      num = JPOOL_LONG (jcf, index);
+      goto integer;
+    integer:
+      {
+       char buffer[25];
+       format_int (buffer, num, 10);
+       fprintf (out, "%s", buffer);
+       if (verbosity > 1)
+         {
+           format_uint (buffer, (uint64)num, 16);
+           fprintf (out, "=0x%s", buffer);
+         }
+      }
+      break;
+    case CONSTANT_Float:
+      {
+       jfloat fnum = JPOOL_FLOAT (jcf, index);
+       fprintf (out, "%s%.10g", verbosity > 1 ? "Float " : "", (double) fnum);
+       if (verbosity > 1)
+         fprintf (out, ", bits = 0x%08lx", (long) (* (int32 *) &fnum));
+       break;
+      }
+    case CONSTANT_Double:
+      {
+       jdouble dnum = JPOOL_DOUBLE (jcf, index);
+       fprintf (out, "%s%.20g", verbosity > 1 ? "Double " : "", dnum);
+       if (verbosity > 1)
+         {
+           int32 hi, lo;
+           hi = JPOOL_UINT (jcf, index);
+           lo = JPOOL_UINT (jcf, index + 1);
+           fprintf (out, ", bits = 0x%08lx%08lx", (long) hi, (long) lo);
+         }
+       break;
+      }
+    case CONSTANT_NameAndType:
+      {
+       uint16 name = JPOOL_USHORT1 (jcf, index);
+       uint16 sig = JPOOL_USHORT2 (jcf, index);
+       if (verbosity > 0)
+         fprintf (out, verbosity > 1 ? "%s name: %d=" : "%s ",
+                  "NameAndType", name);
+       print_name (out, jcf, name);
+       if (verbosity <= 1)
+         fputc (' ', out);
+       else
+         fprintf (out, ", signature: %d=", sig);
+       print_signature (out, jcf, sig, 0);
+      }
+      break;
+    case CONSTANT_Utf8:
+      {
+       register unsigned char *str = JPOOL_UTF_DATA (jcf, index);
+       int length = JPOOL_UTF_LENGTH (jcf, index);
+       if (verbosity > 0)
+         { /* Print as 8-bit bytes. */
+           fputs ("Utf8: \"", out);
+           while (--length >= 0)
+             jcf_print_char (out, *str++);
+         }
+       else
+         { /* Print as Unicode. */
+           fputc ('\"', out);
+           jcf_print_utf8 (out, str, length);
+         }
+       fputc ('\"', out);
+      }
+      break;
+    default:
+      fprintf (out, "(Unknown constant type %d)", kind);
+    }
+}
+
+void
+DEFUN(print_constant_pool, (jcf),
+      JCF *jcf)
+{
+  int i;
+  for (i = 1; i < JPOOL_SIZE(jcf); i++)
+    {
+      int kind = JPOOL_TAG (jcf, i);
+      fprintf (out, "#%d: ", i);
+      print_constant (out, jcf, i, 2);
+      fprintf (out, "\n");
+      if (kind == CONSTANT_Double || kind == CONSTANT_Long)
+       i++; /* These take up two slots in the constant table */
+    }
+}
+
+static void
+DEFUN(print_signature_type, (stream, ptr, limit),
+     FILE* stream AND const unsigned char **ptr AND const unsigned char *limit)
+{
+  int array_size;
+  if ((*ptr) >= limit)
+    return;
+  switch (*(*ptr))
+    {
+    case '[':
+      array_size = -1;
+      for ((*ptr)++; (*ptr) < limit && isdigit (**ptr); (*ptr)++)
+       {
+         int digit = 
+           array_size = (array_size < 0 ? 0 : 10 * array_size) + *(*ptr) - '0';
+       }
+      print_signature_type (stream, ptr, limit);
+      if (array_size == -1)
+       fprintf (stream, "[]");
+      else
+       fprintf (stream, "[%d]", array_size);
+      break;
+    case '(':
+      {
+       int nargs = 0;
+       fputc (*(*ptr)++, stream);
+       for (; **ptr != ')' && *ptr < limit; nargs++)
+         {
+           if (nargs > 0)
+             fputc (',', stream);
+           print_signature_type (stream, ptr, limit);
+         }
+       if (*ptr < limit)
+         {
+           fputc (*(*ptr)++, stream);
+           print_signature_type (stream, ptr, limit);
+         }
+       else
+         fprintf (stream, "???");
+      }
+    break;
+      
+    case 'B':  fprintf (stream, "byte");  (*ptr)++;  break;
+    case 'C':  fprintf (stream, "char");  (*ptr)++;  break;
+    case 'D':  fprintf (stream, "double");  (*ptr)++;  break;
+    case 'F':  fprintf (stream, "float");  (*ptr)++;  break;
+    case 'S':  fprintf (stream, "short");  (*ptr)++;  break;
+    case 'I':  fprintf (stream, "int");  (*ptr)++;  break;
+    case 'J':  fprintf (stream, "long");  (*ptr)++;  break;
+    case 'Z':  fprintf (stream, "boolean");  (*ptr)++;  break;
+    case 'V':  fprintf (stream, "void");  (*ptr)++;  break;
+
+    case 'L':
+      for ((*ptr)++; (*ptr)<limit && *(*ptr) != ';'; (*ptr)++)
+       jcf_print_char (stream, *(*ptr) == '/' ? '.' : *(*ptr));
+      if (*(*ptr) == ';')
+       (*ptr)++;
+      break;
+    default:
+      jcf_print_char (stream, *(*ptr)++);
+    }
+}
+
+static void
+DEFUN(print_signature, (stream, jcf, signature_index, int options),
+      FILE* stream AND JCF *jcf AND int signature_index AND int options)
+{
+  if (JPOOL_TAG (jcf, signature_index) != CONSTANT_Utf8)
+    print_constant_terse (out, jcf, signature_index, CONSTANT_Utf8);
+  else
+    {
+      int j;
+      const unsigned char *str = JPOOL_UTF_DATA (jcf, signature_index);
+      int length = JPOOL_UTF_LENGTH (jcf, signature_index);
+      const unsigned char *limit;
+      limit = str + length;
+      if (str >= limit)
+       fprintf (stream, "<empty signature string>");
+      else
+       {
+         if (options & PRINT_SIGNATURE_RESULT_ONLY)
+           {
+             while (str < limit && *str++ != ')') ;
+           }
+         if (options & PRINT_SIGNATURE_ARGS_ONLY)
+           {
+             *str++;
+             fputc ('(', stream);
+             while (str < limit && *str != ')')
+               {
+                 print_signature_type (stream, &str, limit);
+                 if (*str != ')')
+                   fputs (", ", stream);
+               }
+             fputc (')', stream);
+           }
+         else
+           {
+             print_signature_type (stream, &str, limit);
+             if (str < limit)
+               {
+                 fprintf (stream, "<junk:");
+                 jcf_print_utf8 (stream, str, limit - str);
+                 fputc ('>', stream);
+               }
+           }
+       }
+    }
+}
+
+int
+DEFUN (usage, (), )
+{
+  fprintf (stderr, "Usage: jcf-dump [-o outputfile] [-c] classname\n");
+  exit(1);
+}
+
+void
+DEFUN(process_class, (jcf),
+      JCF *jcf)
+{
+  int code;
+  if (jcf_parse_preamble (jcf) != 0)
+    fprintf (stderr, "Not a valid Java .class file.\n");    
+
+  /* Parse and possibly print constant pool */
+  code = jcf_parse_constant_pool (jcf);
+  if (code != 0)
+    {
+      fprintf (stderr, "error while parsing constant pool\n");
+      exit (-1);
+    }
+  code = verify_constant_pool (jcf);
+  if (code > 0)
+    {
+      fprintf (stderr, "error in constant pool entry #%d\n", code);
+      exit (-1);
+    }
+  if (flag_print_constant_pool)
+    print_constant_pool (jcf);
+
+  jcf_parse_class (jcf);
+  code = jcf_parse_fields (jcf);
+  if (code != 0)
+    {
+      fprintf (stderr, "error while parsing fields\n");
+      exit (-1);
+    }
+  code = jcf_parse_methods (jcf);
+  if (code != 0)
+    {
+      fprintf (stderr, "error while parsing methods\n");
+      exit (-1);
+    }
+  code = jcf_parse_final_attributes (jcf);
+  if (code != 0)
+    {
+      fprintf (stderr, "error while parsing final attributes\n");
+      exit (-1);
+    }
+  jcf->filename = NULL;
+}
+
+int
+DEFUN(main, (argc, argv),
+      int argc AND char** argv)
+{
+  JCF jcf[1];
+  int argi;
+  if (argc <= 1)
+    usage ();
+
+  for (argi = 1; argi < argc; argi++)
+    {
+      char *arg = argv[argi];
+      if (arg[0] == '-')
+       {
+         if (strcmp (arg, "-o") == 0 && argi + 1 < argc)
+           output_file = argv[++argi];
+         else if (arg[1] == '-')
+           {
+             arg++;
+             if (strcmp (arg, "-classpath") == 0 && argi + 1 < argc)
+               classpath = argv[++argi];
+             else if (strcmp (arg, "-verbose") == 0)
+               verbose++;
+             else if (strcmp (arg, "-print-main") == 0)
+               flag_print_main++;
+             else if (strcmp (arg, "-javap") == 0)
+               {
+                 flag_javap_compatible++;
+                 flag_print_constant_pool = 0;
+               }
+             else if (arg[2] == '\0')
+               break;
+             else
+               {
+                 fprintf (stderr, "%s: illegal argument\n", argv[argi]);
+                 exit (-1);
+               }
+             
+           }
+         else if (strcmp (arg, "-classpath") == 0 && argi + 1 < argc)
+           classpath = argv[++argi];
+         else if (strcmp (arg, "-verbose") == 0)
+           verbose++;
+         else if (strcmp (arg, "-print-main") == 0)
+           flag_print_main++;
+         else if (strcmp (arg, "-c") == 0)
+           flag_disassemble_methods++;
+         else
+           {
+             fprintf (stderr, "%s: illegal argument\n", argv[argi]);
+             exit (-1);
+           }
+       }
+      else
+       break;
+    }
+  if (argi == argc)
+    usage ();
+  if (flag_print_main)
+    {
+      flag_print_fields = 0;
+      flag_print_methods = 0;
+      flag_print_constant_pool = 0;
+      flag_print_attributes = 0;
+      flag_print_class_info = 0;
+    }
+
+  if (classpath == NULL)
+    {
+      classpath = (char *) getenv ("CLASSPATH");
+      if (classpath == NULL)
+       classpath = "";
+    }
+
+  if (output_file)
+    {
+      out = fopen (output_file, "w");
+      if (out)
+       {
+         fprintf (stderr, "Cannot open '%s' for output.\n", output_file);
+         exit (-1);
+       }
+    }
+  else
+    out = stdout;
+
+  if (argi >= argc)
+    {
+      fprintf (out, "Reading .class from <standard input>.\n");
+#if JCF_USE_STDIO
+      open_class ("<stdio>", jcf, stdin);
+#else
+      open_class ("<stdio>", jcf, 0);
+#endif
+      process_class (jcf);
+    }
+  else
+    {
+      for (; argi < argc; argi++)
+       {
+         char *arg = argv[argi];
+         char* class_filename = find_class (arg, strlen (arg), jcf, 1);
+         if (class_filename == NULL)
+           class_filename = find_classfile (arg, jcf);
+         if (class_filename == NULL)
+           {
+             perror ("Could not find class");
+             exit (-1);
+           }
+         JCF_FILL (jcf, 4);
+         if (GET_u4 (jcf->read_ptr) == ZIPMAGIC)
+           {
+             long compressed_size, member_size;
+             int compression_method, filename_length, extra_length;
+             int general_purpose_bits;
+             char *filename;
+             int total_length;
+             if (flag_print_class_info)
+               fprintf (out, "Reading classes from archive %s.\n",
+                        class_filename);
+             for (;;)
+               {
+                 int skip = 0;
+                 jcf_filbuf_t save_filbuf = jcf->filbuf;
+                 long magic = JCF_readu4_le (jcf);
+                 if (magic == 0x02014b50 || magic == 0x06054b50)
+                   break;  /* got to central directory */
+                 if (magic != 0x04034b50) /* ZIPMAGIC (little-endian) */
+                   {
+                     fprintf (stderr, "bad format of .zip archine\n");
+                     exit (-1);
+                   }
+                 JCF_FILL (jcf, 26);
+                 JCF_SKIP (jcf, 2);
+                 general_purpose_bits = JCF_readu2_le (jcf);
+                 compression_method = JCF_readu2_le (jcf);
+                 JCF_SKIP (jcf, 8);
+                 compressed_size = JCF_readu4_le (jcf);
+                 member_size = JCF_readu4_le (jcf);
+                 filename_length = JCF_readu2_le (jcf);
+                 extra_length = JCF_readu2_le (jcf);
+                 total_length = filename_length + extra_length
+                   + compressed_size;
+                 if (jcf->read_end - jcf->read_ptr < total_length)
+                   jcf_trim_old_input (jcf);
+                 JCF_FILL (jcf, total_length);
+                 filename = jcf->read_ptr;
+                 JCF_SKIP (jcf, filename_length);
+                 JCF_SKIP (jcf, extra_length);
+                 if (filename_length > 0
+                     && filename[filename_length-1] == '/')
+                   {
+                     if (flag_print_class_info)
+                       fprintf (out, "[Skipping directory %.*s]\n",
+                                filename_length, filename);
+                     skip = 1;
+                   }
+                 else if (compression_method != 0)
+                   {
+                     if (flag_print_class_info)
+                       fprintf (out, "[Skipping compressed file %.*s]\n",
+                                filename_length, filename);
+                     skip = 1;
+                   }
+                 else if (member_size < 4
+                          || GET_u4 (jcf->read_ptr) != 0xcafebabe)
+                   {
+                     if (flag_print_class_info)
+                       fprintf (out, "[Skipping non-.class member %.*s]\n",
+                                filename_length, filename);
+                     skip = 1;
+                   }
+                 else
+                   {
+                     if (flag_print_class_info)
+                       fprintf (out, "Reading class member: %.*s.\n",
+                                filename_length, filename);
+                   }
+                 if (skip)
+                   {
+                     JCF_SKIP (jcf, compressed_size);
+                   }
+                 else
+                   {
+                     unsigned char *save_end;
+                     jcf->filbuf = jcf_unexpected_eof;
+                     save_end = jcf->read_end;
+                     jcf->read_end = jcf->read_ptr + compressed_size;
+                     process_class (jcf);
+                     jcf->filbuf = save_filbuf;
+                     jcf->read_end = save_end;
+                   }
+               }
+           }
+         else
+           {
+             if (flag_print_class_info)
+               fprintf (out, "Reading .class from %s.\n", class_filename);
+             process_class (jcf);
+           }
+         JCF_FINISH(jcf);
+       }
+    }
+}
+
+static void
+DEFUN(disassemble_method, (jcf, byte_ops, len),
+      JCF* jcf AND unsigned char *byte_ops AND int len)
+{
+#undef AND /* Causes problems with opcodes for iand and land. */
+#undef PTR
+  int PC;
+  int i;
+  if (flag_disassemble_methods == 0)
+    return;
+#define BCODE byte_ops
+  for (PC = 0; PC < len;)
+    {
+      int oldpc = PC;
+      int saw_index;
+      jlong LONG_temp;
+      jint INT_temp;
+      jfloat FLOAT_temp;
+      jdouble DOUBLE_temp;
+      switch (byte_ops[PC++])
+       {
+
+/* This is the actual code emitted for each of opcodes in javaops.def.
+   The actual opcode-specific stuff is handled by the OPKIND macro.
+   I.e. for an opcode whose OPKIND is BINOP, the BINOP will be called.
+   Those macros are defiend below.  The OPKINDs that do not have any
+   inline parameters (such as BINOP) and therefore do mot need anything
+   else to me printed out just use an empty body. */
+
+#define JAVAOP(OPNAME, OPCODE, OPKIND, OPERAND_TYPE, OPERAND_VALUE) \
+        case OPCODE: \
+         fprintf (out, "%3d: %s", oldpc, #OPNAME); \
+         OPKIND(OPERAND_TYPE, OPERAND_VALUE); \
+         fputc ('\n', out); \
+         break;
+
+#define CONST_INDEX_1 (saw_index = 1, IMMEDIATE_u1)
+#define CONST_INDEX_2 (saw_index = 1, IMMEDIATE_u2)
+#define VAR_INDEX_1 (saw_index = 1, IMMEDIATE_u1)
+#define VAR_INDEX_2 (saw_index = 1, IMMEDIATE_u2)
+
+#define CHECK_PC_IN_RANGE(PC) (PC < 0 || PC > len ? \
+  (fprintf(stderr, "Bad byte codes.\n"), exit(-1)) : 1)
+
+/* Print out operand (if not implied by the opcode) for PUSCH opcodes.
+   These all push a constant onto the opcode stack. */
+#define PUSHC(OPERAND_TYPE, OPERAND_VALUE) \
+  saw_index = 0, INT_temp = (OPERAND_VALUE); \
+  if (oldpc+1 == PC) /* nothing */; \
+  else if (saw_index) fprintf (out, " "), print_constant_ref (out, jcf, INT_temp); \
+  else fprintf (out, " %d", INT_temp);
+
+/* Print out operand (a local variable index) for LOAD opcodes.
+   These all push local variable onto the opcode stack. */
+#define LOAD(OPERAND_TYPE, OPERAND_VALUE) \
+  INT_temp = (OPERAND_VALUE); \
+  if (oldpc+1 == PC) /* nothing - local index implied by opcode */; \
+  else fprintf (out, " %d", INT_temp);
+
+/* Handle STORE opcodes same as LOAD opcodes.
+   These all store a value from the opcode stack in a local variable. */
+#define STORE LOAD
+
+/* Handle more kind of opcodes. */
+#define STACK(OPERAND_TYPE, OPERAND_VALUE) /* nothing */
+#define UNOP(OPERAND_TYPE, OPERAND_VALUE) /* nothing */
+#define BINOP(OPERAND_TYPE, OPERAND_VALUE) /* nothing */
+#define CONVERT(OPERAND_TYPE, OPERAND_VALUE) /* nothing */
+#define CONVERT2(OPERAND_TYPE, OPERAND_VALUE) /* nothing */
+#define RETURN(OPERAND_TYPE, OPERAND_VALUE) /* nothing */
+#define UNKNOWN(OPERAND_TYPE, OPERAND_VALUE) /* nothing */
+
+/* Handle putfield and getfield opcodes, with static versions. */
+#define FIELD(MAYBE_STATIC, PUT_OR_GET) \
+  fputc (' ', out); print_constant_ref (out, jcf, IMMEDIATE_u2)
+
+/* Print operand for invoke opcodes. */
+#define INVOKE(OPERAND_TYPE, OPERAND_VALUE) \
+  fputc (' ', out); print_constant_ref (out, jcf, IMMEDIATE_u2);\
+  PC += 2 * OPERAND_VALUE /* for invokeinterface */;
+
+#define OBJECT(OPERAND_TYPE, OPERAND_VALUE) \
+  fputc (' ', out); print_constant_ref (out, jcf, IMMEDIATE_u2);
+
+#define ARRAY(OPERAND_TYPE, SUBOP) \
+  ARRAY_##SUBOP(OPERAND_TYPE)
+/* Handle sub-categories of ARRAY opcodes. */
+#define ARRAY_LOAD(TYPE) /* nothing */
+#define ARRAY_STORE(TYPE) /* nothing */
+#define ARRAY_LENGTH(TYPE) /* nothing */
+#define ARRAY_NEW(TYPE) ARRAY_NEW_##TYPE
+#define ARRAY_NEW_NUM \
+ INT_temp = IMMEDIATE_u1; \
+ { char *str; \
+  switch (INT_temp) {  \
+    case  4: str = "boolean"; break; \
+    case  5: str = "char"; break; \
+    case  6: str = "float"; break; \
+    case  7: str = "double"; break; \
+    case  8: str = "byte"; break; \
+    case  9: str = "short"; break; \
+    case 10: str = "int"; break; \
+    case 11: str = "long"; break; \
+    default: str = "<unknown type code %d>"; break; \
+  } \
+  fputc (' ', out); fprintf (out, str, INT_temp); }
+
+#define ARRAY_NEW_PTR  \
+  fputc (' ', out); print_constant_ref (out, jcf, IMMEDIATE_u2);
+
+#define ARRAY_NEW_MULTI \
+  fputc (' ', out); print_constant_ref (out, jcf, IMMEDIATE_u2); \
+  fprintf (out, " %d", IMMEDIATE_u1); /* number of dimensions */
+
+#define TEST(OPERAND_TYPE, OPERAND_VALUE) \
+  fprintf (out, " %d", oldpc + IMMEDIATE_s2)
+
+#define BRANCH(OPERAND_TYPE, OPERAND_VALUE) \
+  saw_index = 0, INT_temp = (OPERAND_VALUE); \
+  fprintf (out, " %d", saw_index ? INT_temp : oldpc + INT_temp)
+
+#define JSR(OPERAND_TYPE, OPERAND_VALUE) \
+  saw_index = 0, INT_temp = (OPERAND_VALUE); \
+  fprintf (out, " %d", saw_index ? INT_temp : oldpc + INT_temp)
+
+#define RET(OPERAND_TYPE, OPERAND_VALUE) \
+  INT_temp = (OPERAND_VALUE); \
+  fprintf (out, " %d", INT_temp);
+
+#define SWITCH(OPERAND_TYPE, TABLE_OR_LOOKUP) \
+  PC = (PC + 3) / 4 * 4; TABLE_OR_LOOKUP##_SWITCH
+
+#define LOOKUP_SWITCH \
+  { jint default_offset = IMMEDIATE_s4;  jint npairs = IMMEDIATE_s4; \
+    fprintf (out, " npairs=%d, default=%d", npairs, default_offset+oldpc); \
+    while (--npairs >= 0) { \
+     jint match = IMMEDIATE_s4; jint offset = IMMEDIATE_s4; \
+     fprintf (out, "\n%10ld: %ld", (long)match, (long)(offset+oldpc)); } \
+  }
+
+#define TABLE_SWITCH \
+  { jint default_offset = IMMEDIATE_s4; \
+    jint low = IMMEDIATE_s4; jint high = IMMEDIATE_s4; \
+    fprintf (out, " low==%d, high=%ddefault=%d", \
+      low, high, default_offset+oldpc); \
+    for (; low <= high; low++) { \
+     jint offset = IMMEDIATE_s4; \
+     fprintf (out, "\n%10ld: %ld", (long)low, (long)(offset+oldpc)); } \
+  }
+
+#define SPECIAL(OPERAND_TYPE, OPERAND_VALUE) \
+  SPECIAL_##OPERAND_VALUE(OPERAND_TYPE)
+
+#define SPECIAL_IINC(OPERAND_TYPE) \
+  INT_temp = IMMEDIATE_u1; \
+  fprintf (out, "%d %d", INT_temp, IMMEDIATE_s1)
+
+#define SPECIAL_WIDE(OPERAND_TYPE) \
+  INT_temp = IMMEDIATE_u1; fprintf (out, "%d", INT_temp)
+
+#define SPECIAL_EXIT(OPERAND_TYPE) /* nothing */
+#define SPECIAL_ENTER(OPERAND_TYPE) /* nothing */
+#define SPECIAL_BREAK(OPERAND_TYPE) /* nothing */
+#define SPECIAL_THROW(OPERAND_TYPE) /* nothing */
+
+#define IMPL(OPERAND_TYPE, OPERAND_VALUE) \
+  fprintf (out, " %d", IMMEDIATE_u##OPERAND_VALUE)
+
+#define COND(OPERAND_TYPE, OPERAND_VALUE) \
+   TEST(OPERAND_TYPE, OPERAND_VALUE)
+
+#include "javaop.def"
+       default:
+         fprintf (out, "%3d: unknown(%3d)\n", oldpc, byte_ops[PC]);
+       }
+    }
+
+  /* Print exception table. */
+  i = GET_u2(byte_ops+len);
+  if (i > 0)
+    {
+      unsigned char *ptr = byte_ops+len+2;
+      fprintf (out, "Exceptions (count: %d):\n", i);
+      for (; --i >= 0;  ptr+= 8)
+       {
+         int start_pc = GET_u2 (ptr);
+         int end_pc = GET_u2 (ptr+2);
+         int handler_pc = GET_u2 (ptr+4);
+         int catch_type = GET_u2 (ptr+6);
+         fprintf (out, "  start: %d, end: %d, handler: %d, type: %d",
+                  start_pc, end_pc, handler_pc, catch_type);
+         if (catch_type == 0)
+           fputs (" /* finally */", out);
+         else
+           {
+             fputc('=', out);
+             print_constant_terse (out, jcf, catch_type, CONSTANT_Class);
+           }
+         fputc ('\n', out);
+       }
+    }
+}
diff --git a/gcc/java/jcf-io.c b/gcc/java/jcf-io.c
new file mode 100644 (file)
index 0000000..931f34e
--- /dev/null
@@ -0,0 +1,574 @@
+/* Utility routines for finding and reading Java(TM) .class files.
+   Copyright (C) 1996  Free Software Foundation, Inc.
+
+This program is free software; you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation; either version 2, or (at your option)
+any later version.
+
+This program is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with GNU CC; see the file COPYING.  If not, write to
+the Free Software Foundation, 59 Temple Place - Suite 330,
+Boston, MA 02111-1307, USA.  
+
+Java and all Java-based marks are trademarks or registered trademarks
+of Sun Microsystems, Inc. in the United States and other countries.
+The Free Software Foundation is independent of Sun Microsystems, Inc.  */
+
+/* Written by Per Bothner <bothner@cygnus.com>, February 1996. */
+
+#include <stdio.h>
+
+#define ENABLE_UNZIP 1
+
+#include "jcf.h"
+#ifdef __STDC__
+#include <stdlib.h>
+#endif
+#include <sys/types.h>
+#include <sys/stat.h>
+#include <sys/wait.h>
+#include <errno.h>
+#include <fcntl.h>
+
+/* DOS brain-damage */
+#ifndef O_BINARY
+#define O_BINARY 0 /* MS-DOS brain-damage */
+#endif
+
+char *classpath;
+
+int
+DEFUN(jcf_unexpected_eof, (jcf, count),
+      JCF *jcf AND int count)
+{
+  if (jcf->filename)
+    fprintf (stderr, "Premature end of .class file %s.\n", jcf->filename);
+  else
+    fprintf (stderr, "Premature end of .class file <stdin>.\n");
+  exit (-1);
+}
+
+void
+DEFUN(jcf_trim_old_input, (jcf),
+      JCF *jcf)
+{
+  int count = jcf->read_ptr - jcf->buffer;
+  if (count > 0)
+    {
+      memmove (jcf->buffer, jcf->read_ptr, jcf->read_end - jcf->read_ptr);
+      jcf->read_ptr -= count;
+      jcf->read_end -= count;
+    }
+}
+
+int
+DEFUN(jcf_filbuf_from_stdio, (jcf, count),
+      JCF *jcf AND int count)
+{
+  FILE *file = (FILE*) (jcf->read_state);
+  if (count > jcf->buffer_end - jcf->read_ptr)
+    {
+      JCF_u4 old_read_ptr = jcf->read_ptr - jcf->buffer;
+      JCF_u4 old_read_end = jcf->read_end - jcf->buffer;
+      JCF_u4 old_size = jcf->buffer_end - jcf->buffer;
+      JCF_u4 new_size = (old_size == 0 ? 2000 : 2 * old_size) + count;
+      unsigned char *new_buffer = jcf->buffer == NULL ? ALLOC (new_size)
+       : REALLOC (jcf->buffer, new_size);
+      jcf->buffer = new_buffer;
+      jcf->buffer_end = new_buffer + new_size;
+      jcf->read_ptr = new_buffer + old_read_ptr;
+      jcf->read_end = new_buffer + old_read_end;
+    }
+  count -= jcf->read_end - jcf->read_ptr;
+  if (count <= 0)
+    return 0;
+  if (fread (jcf->read_end, 1, count, file) != count)
+    jcf_unexpected_eof (jcf, count);
+  jcf->read_end += count;
+  return 0;
+}
+
+#if ENABLE_UNZIP
+#include "zipfile.h"
+
+struct ZipFileCache *SeenZipFiles = NULL;
+
+int
+DEFUN(open_in_zip, (jcf, 
+zipfile, zipmember),
+      JCF *jcf AND const char *zipfile AND const char *zipmember)
+{
+  struct ZipFileCache* zipf;
+  ZipDirectory *zipd;
+  int i, len;
+  for (zipf = SeenZipFiles; ; zipf = zipf->next)
+    {
+      if (zipf == NULL)
+       {
+         char magic [4];
+         int fd = open (zipfile, O_RDONLY | O_BINARY);
+         if (read (fd, magic, 4) != 4 || GET_u4 (magic) != (JCF_u4)ZIPMAGIC)
+           return -1;
+         lseek (fd, 0L, SEEK_SET);
+         zipf = ALLOC (sizeof (struct ZipFileCache) + strlen (zipfile) + 1);
+         zipf->next = SeenZipFiles;
+         zipf->name = (char*)(zipf+1);
+         strcpy (zipf->name, zipfile);
+         SeenZipFiles = zipf;
+         zipf->z.fd = fd;
+         if (fd == -1)
+           {
+             /* A missing zip file is not considered an error. */
+             zipf->z.count = 0;
+             zipf->z.dir_size = 0;
+             zipf->z.central_directory = NULL;
+             return -1;
+           }
+         else
+           {
+             if (read_zip_archive (&zipf->z) != 0)
+               return -2; /* This however should be an error - FIXME */
+           }
+         break;
+       }
+      if (strcmp (zipf->name, zipfile) == 0)
+       break;
+    }
+
+  if (!zipmember)
+    return 0;
+
+  len = strlen (zipmember);
+  
+  zipd = (struct ZipDirectory*) zipf->z.central_directory;
+  for (i = 0; i < zipf->z.count; i++, zipd = ZIPDIR_NEXT (zipd))
+    {
+      if (len == zipd->filename_length &&
+         strncmp (ZIPDIR_FILENAME (zipd), zipmember, len) == 0)
+       {
+         JCF_ZERO (jcf);
+         jcf->buffer = ALLOC (zipd->size);
+         jcf->buffer_end = jcf->buffer + zipd->size;
+         jcf->read_ptr = jcf->buffer;
+         jcf->read_end = jcf->buffer_end;
+         jcf->filbuf = jcf_unexpected_eof;
+         jcf->filename = (char *) strdup (zipfile);
+         jcf->classname = (char *) strdup (zipmember);
+         jcf->zipd = (void *)zipd;
+         if (lseek (zipf->z.fd, zipd->filestart, 0) < 0
+             || read (zipf->z.fd, jcf->buffer, zipd->size) != zipd->size)
+           return -2;
+         return 0;
+       }
+    }
+  return -1;
+}
+#endif /* ENABLE_UNZIP */
+
+#if JCF_USE_STDIO
+char*
+DEFUN(open_class, (filename, jcf, stream),
+      char *filename AND JCF *jcf AND FILE* stream)
+{
+  if (jcf)
+    {
+      JCF_ZERO (jcf);
+      jcf->buffer = NULL;
+      jcf->buffer_end = NULL;
+      jcf->read_ptr = NULL;
+      jcf->read_end = NULL;
+      jcf->read_state = stream;
+      jcf->filbuf = jcf_filbuf_from_stdio;
+    }
+  else
+    fclose (stream);
+  return filename;
+}
+#else
+char*
+DEFUN(open_class, (filename, jcf, fd),
+      char *filename AND JCF *jcf AND int fd)
+{
+  if (jcf)
+    {
+      struct stat stat_buf;
+      if (fstat (fd, &stat_buf) != 0
+         || ! S_ISREG (stat_buf.st_mode))
+       {
+         perror ("Could not figure length of .class file");
+         return NULL;
+       }
+      JCF_ZERO (jcf);
+      jcf->buffer = ALLOC (stat_buf.st_size);
+      jcf->buffer_end = jcf->buffer + stat_buf.st_size;
+      jcf->read_ptr = jcf->buffer;
+      jcf->read_end = jcf->buffer_end;
+      jcf->read_state = NULL;
+      jcf->filename = filename;
+      if (read (fd, jcf->buffer, stat_buf.st_size) != stat_buf.st_size)
+       {
+         perror ("Failed to read .class file");
+         return NULL;
+       }
+      close (fd);
+      jcf->filbuf = jcf_unexpected_eof;
+    }
+  else
+    close (fd);
+  return filename;
+}
+#endif
+
+
+char *
+DEFUN(find_classfile, (filename_length, jcf),
+      char *filename AND JCF *jcf)
+{
+#if JCF_USE_STDIO
+  FILE *stream = fopen (filename, "rb");
+  if (stream == NULL)
+    return NULL;
+  return open_class (arg, jcf, stream);
+#else
+  int fd = open (filename, O_RDONLY | O_BINARY);
+  if (fd < 0)
+    return NULL;
+  return open_class (filename, jcf, fd);
+#endif
+}
+
+/* Returns a freshly malloc'd string with the fully qualified pathname
+   of the .class file for the class CLASSNAME.  Returns NULL on
+   failure.  If JCF != NULL, it is suitably initialized.  With
+   DO_CLASS_FILE set to 1, search a .class/.java file named after
+   CLASSNAME, otherwise, search a ZIP directory entry named after
+   CLASSNAME.  */
+
+char *
+DEFUN(find_class, (classname, classname_length, jcf, do_class_file),
+      const char *classname AND int classname_length AND JCF *jcf AND int do_class_file)
+
+{
+#if JCF_USE_STDIO
+  FILE *stream;
+#else
+  int fd;
+#endif
+  int i, j, k, java, class;
+  struct stat java_buf, class_buf;
+
+  /* Allocate and zero out the buffer, since we don't explicitly put a
+     null pointer when we're copying it below.  */
+  int buflen = strlen (classpath) + classname_length + 10;
+  char *buffer = (char *) ALLOC (buflen);
+  bzero (buffer, buflen);
+
+  jcf->java_source = jcf->outofsynch = 0;
+  for (j = 0; classpath[j] != '\0'; )
+    {
+      for (i = 0; classpath[j] != ':' && classpath[j] != '\0'; i++, j++)
+       buffer[i] = classpath[j];
+      if (classpath[j] == ':')
+       j++;
+      if (i > 0)  /* Empty directory is redundant */
+       {
+         int dir_len;
+         if (buffer[i-1] != '/')
+           buffer[i++] = '/';
+         dir_len = i-1;
+         for (k = 0; k < classname_length; k++, i++)
+           {
+             char ch = classname[k];
+             buffer[i] = ch == '.' ? '/' : ch;
+           }
+         if (do_class_file)
+           strcpy (buffer+i, ".class");
+#if ENABLE_UNZIP
+         if (dir_len > 4
+             && buffer[dir_len-4] == '.' && buffer[dir_len-3] == 'z'
+             && buffer[dir_len-2] == 'i' && buffer[dir_len-1] == 'p')
+           {
+             int err_code;
+             JCF _jcf;
+             if (!do_class_file)
+               strcpy (buffer+i, "/");
+             buffer[dir_len] = '\0';
+             if (do_class_file)
+               SOURCE_FRONTEND_DEBUG 
+                 (("Trying [...%s]:%s", 
+                   &buffer[dir_len-(dir_len > 15 ? 15 : dir_len)], 
+                   buffer+dir_len+1));
+             if (jcf == NULL)
+               jcf = &_jcf;
+             err_code = open_in_zip (jcf, buffer, buffer+dir_len+1);
+             if (err_code == 0)
+               {
+                 if (!do_class_file)
+                   jcf->seen_in_zip = 1;
+                 else
+                   {
+                     buffer[dir_len] = '(';
+                     strcpy (buffer+i, ".class)");
+                   }
+                 if (jcf == &_jcf)
+                   JCF_FINISH (jcf);
+                 return buffer;
+               }
+             else
+               continue;
+           }
+#endif
+         /* If we do directories, do them here */
+         if (!do_class_file)
+           {
+             struct stat dir_buff;
+             int dir;
+             buffer[i] = '\0'; /* Was previously unterminated here. */
+             if (!(dir = stat (buffer, &dir_buff)))
+               {
+                 jcf->seen_in_zip = 0;
+                 goto found;
+               }
+           }
+         
+         /* Check for out of synch .class/.java files */
+         class = stat (buffer, &class_buf);
+         strcpy (buffer+i, ".java");
+         java = stat (buffer, &java_buf);
+         if ((!java && !class) && java_buf.st_mtime >= class_buf.st_mtime)
+           jcf->outofsynch = 1;
+#if JCF_USE_STDIO
+         if (!class)
+           {
+             strcpy (buffer+i, ".class");
+             SOURCE_FRONTEND_DEBUG (("Trying %s", buffer));
+             stream = fopen (buffer, "rb");
+             if (stream)
+               goto found;
+           }
+         /* Give .java a try, if necessary */
+         if (!java)
+           {
+             strcpy (buffer+i, ".java");
+             SOURCE_FRONTEND_DEBUG (("Trying %s", buffer));
+             stream = fopen (buffer, "r");
+             if (stream)
+               {
+                 jcf->java_source = 1;
+                 goto found;
+               }
+           }
+#else
+         if (!class)
+           {
+             strcpy (buffer+i, ".class");
+             SOURCE_FRONTEND_DEBUG (("Trying %s", buffer));
+             fd = open (buffer, O_RDONLY | O_BINARY);
+             if (fd >= 0)
+               goto found;
+           }
+         /* Give .java a try, if necessary */
+         if (!java)
+           {
+             if (do_class_file)
+               strcpy (buffer+i, ".java");
+             SOURCE_FRONTEND_DEBUG (("Trying %s", buffer));
+             fd = open (buffer, O_RDONLY | O_BINARY);
+             if (fd >= 0)
+               {
+                 jcf->java_source = 1;
+                 goto found;
+               }
+           }
+#endif
+       }
+    }
+  free (buffer);
+  return NULL;
+ found:
+#if JCF_USE_STDIO
+  if (jcf->java_source)
+    return NULL;               /* FIXME */
+  else
+    return open_class (buffer, jcf, stream);
+#else
+  if (jcf->java_source)
+    {
+      JCF_ZERO (jcf);          /* JCF_FINISH relies on this */
+      jcf->java_source = 1;
+      jcf->filename = (char *) strdup (buffer);
+      close (fd);              /* We use STDIO for source file */
+    }
+  else if (do_class_file)
+    buffer = open_class (buffer, jcf, fd);
+  jcf->classname = (char *) ALLOC (classname_length + 1);
+  strncpy (jcf->classname, classname, classname_length + 1);
+  jcf->classname = (char *) strdup (classname);
+  return buffer;
+#endif
+}
+
+void
+DEFUN(jcf_print_char, (stream, ch),
+      FILE *stream AND int ch)
+{
+  switch (ch)
+    {
+    case '\'':
+    case '\\':
+    case '\"':
+      fprintf (stream, "\\%c", ch);
+      break;
+    case '\n':
+      fprintf (stream, "\\n");
+      break;
+    case '\t':
+      fprintf (stream, "\\t");
+      break;
+    case '\r':
+      fprintf (stream, "\\r");
+      break;
+    default:
+      if (ch >= ' ' && ch < 127)
+       putc (ch, stream);
+      else if (ch < 256)
+       fprintf (stream, "\\%03x", ch);
+      else
+       fprintf (stream, "\\u%04x", ch);
+    }
+}
+
+/* Print UTF8 string at STR of length LENGTH bytes to STREAM. */
+
+void
+DEFUN(jcf_print_utf8, (stream, str, length),
+      FILE *stream AND register unsigned char *str AND int length)
+{
+  unsigned char* limit = str + length;
+  while (str < limit)
+    {
+      int ch = UTF8_GET (str, limit);
+      if (ch < 0)
+       {
+         fprintf (stream, "\\<invalid>");
+         return;
+       }
+      jcf_print_char (stream, ch);
+    }
+}
+
+/* Same as jcf_print_utf8, but print IN_CHAR as OUT_CHAR. */
+
+void
+DEFUN(jcf_print_utf8_replace, (stream, str, length, in_char, out_char),
+      FILE *stream AND unsigned char *str AND int length
+      AND int in_char AND int out_char)
+{
+
+  int i;/* FIXME - actually handle Unicode! */
+  for (i = 0; i < length; i++)
+    {
+      int ch = str[i];
+      jcf_print_char (stream, ch == in_char ? out_char : ch);
+    }
+}
+
+/* Check that all the cross-references in the constant pool are
+   valid.  Returns 0 on success.
+   Otherwise, returns the index of the (first) invalid entry. */
+
+int
+DEFUN(verify_constant_pool, (jcf),
+      JCF *jcf)
+{
+  int i, n;
+  for (i = 1; i < JPOOL_SIZE (jcf); i++)
+    {
+      switch (JPOOL_TAG (jcf, i))
+       {
+       case CONSTANT_NameAndType:
+         n = JPOOL_USHORT2 (jcf, i);
+         if (n <= 0 || n >= JPOOL_SIZE(jcf)
+             || JPOOL_TAG (jcf, n) != CONSTANT_Utf8)
+           return i;
+         /* ... fall through ... */
+       case CONSTANT_Class:
+       case CONSTANT_String:
+         n = JPOOL_USHORT1 (jcf, i);
+         if (n <= 0 || n >= JPOOL_SIZE(jcf)
+             || JPOOL_TAG (jcf, n) != CONSTANT_Utf8)
+           return i;
+         break;
+       case CONSTANT_Fieldref:
+       case CONSTANT_Methodref:
+       case CONSTANT_InterfaceMethodref:
+         n = JPOOL_USHORT1 (jcf, i);
+         if (n <= 0 || n >= JPOOL_SIZE(jcf)
+             || JPOOL_TAG (jcf, n) != CONSTANT_Class)
+           return i;
+         n = JPOOL_USHORT2 (jcf, i);
+         if (n <= 0 || n >= JPOOL_SIZE(jcf)
+             || JPOOL_TAG (jcf, n) != CONSTANT_NameAndType)
+           return i;
+         break;
+       case CONSTANT_Long:
+       case CONSTANT_Double:
+         i++;
+         break;
+       case CONSTANT_Float:
+       case CONSTANT_Integer:
+       case CONSTANT_Utf8:
+       case CONSTANT_Unicode:
+         break;
+       default:
+         return i;
+       }
+    }
+  return 0;
+}
+
+void
+DEFUN(format_uint, (buffer, value, base),
+      char *buffer AND uint64 value AND int base)
+{
+#define WRITE_BUF_SIZE (4 + sizeof(uint64) * 8)
+  char buf[WRITE_BUF_SIZE];
+  register char *buf_ptr = buf+WRITE_BUF_SIZE; /* End of buf. */
+  int chars_written;
+  int i;
+
+  /* Now do the actual conversion, placing the result at the *end* of buf. */
+  /* Note this code does not pretend to be optimized. */
+  do {
+    int digit = value % base;
+    static char digit_chars[] = "0123456789abcdefghijklmnopqrstuvwxyz";
+    *--buf_ptr = digit_chars[digit];
+    value /= base;
+  } while (value != 0);
+
+  chars_written = buf+WRITE_BUF_SIZE - buf_ptr;
+  for (i = 0; i < chars_written; i++)
+    buffer[i] = *buf_ptr++;
+  buffer[i] = 0;
+}
+
+void
+DEFUN(format_int, (buffer, value, base),
+      char *buffer AND jlong value AND int base)
+{
+  uint64 abs_value;
+  if (value < 0)
+    {
+      abs_value = -(uint64)value;
+      *buffer++ = '-';
+    }
+  else
+    abs_value = (uint64) value;
+  format_uint (buffer, abs_value, base);
+}
diff --git a/gcc/java/jcf-parse.c b/gcc/java/jcf-parse.c
new file mode 100644 (file)
index 0000000..0e8c51c
--- /dev/null
@@ -0,0 +1,917 @@
+/* Parser for Java(TM) .class files.
+   Copyright (C) 1996 Free Software Foundation, Inc.
+
+This file is part of GNU CC.
+
+GNU CC is free software; you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation; either version 2, or (at your option)
+any later version.
+
+GNU CC is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with GNU CC; see the file COPYING.  If not, write to
+the Free Software Foundation, 59 Temple Place - Suite 330,
+Boston, MA 02111-1307, USA.
+
+Java and all Java-based marks are trademarks or registered trademarks
+of Sun Microsystems, Inc. in the United States and other countries.
+The Free Software Foundation is independent of Sun Microsystems, Inc.  */
+
+/* Written by Per Bothner <bothner@cygnus.com> */
+
+#include <stdio.h>
+#include <ctype.h>
+#include "config.h"
+#include "tree.h"
+#include "obstack.h"
+#include "flags.h"
+#include "java-except.h"
+#include "input.h"
+#include "java-tree.h"
+
+/* A CONSTANT_Utf8 element is converted to an IDENTIFIER_NODE at parse time. */
+#define JPOOL_UTF(JCF, INDEX) CPOOL_UTF(&(JCF)->cpool, INDEX)
+#define JPOOL_UTF_LENGTH(JCF, INDEX) IDENTIFIER_LENGTH (JPOOL_UTF (JCF, INDEX))
+#define JPOOL_UTF_DATA(JCF, INDEX) \
+  ((unsigned char*) IDENTIFIER_POINTER (JPOOL_UTF (JCF, INDEX)))
+#define HANDLE_CONSTANT_Utf8(JCF, INDEX, LENGTH) \
+  do { \
+    unsigned char save;  unsigned char *text; \
+    JCF_FILL (JCF, (LENGTH)+1); /* Make sure we read 1 byte beyond string. */ \
+    text = (JCF)->read_ptr; \
+    save = text[LENGTH]; \
+    text[LENGTH] = 0; \
+    (JCF)->cpool.data[INDEX] = (jword) get_identifier (text); \
+    text[LENGTH] = save; \
+    JCF_SKIP (JCF, LENGTH); } while (0)
+
+#include "jcf.h"
+#ifdef __STDC__
+/* For getenv */
+#include <stdlib.h>
+#endif
+
+#ifndef SEEK_SET
+#include <unistd.h>
+#endif
+
+extern struct obstack *saveable_obstack;
+extern struct obstack temporary_obstack;
+extern struct obstack permanent_obstack;
+
+/* The class we are currently processing. */
+tree current_class = NULL_TREE;
+
+/* The class we started with. */
+tree main_class = NULL_TREE;
+
+/* The FIELD_DECL for the current field. */
+static tree current_field = NULL_TREE;
+
+static tree current_method = NULL_TREE;
+
+static tree give_name_to_class PROTO ((JCF *jcf, int index));
+
+void parse_zip_file_entries (void);
+void process_zip_dir();
+
+/* Source file compilation declarations */
+static void parse_source_file ();
+extern int java_error_count;
+#define java_parse_abort_on_error()            \
+  {                                            \
+     if (java_error_count)                     \
+       {                                       \
+         java_report_errors ();                        \
+        java_pop_parser_context ();            \
+        return;                                \
+       }                                       \
+   }
+
+/* Handle "SourceFile" attribute. */
+
+void
+set_source_filename (jcf, index)
+     JCF *jcf;
+     int index;
+{
+  tree sfname_id = get_name_constant (jcf, index);
+  char *sfname = IDENTIFIER_POINTER (sfname_id);
+  if (input_filename != NULL)
+    {
+      int old_len = strlen (input_filename);
+      int new_len = IDENTIFIER_LENGTH (sfname_id);
+      /* Use the current input_filename (derived from the class name)
+        if it has a directory prefix, but otherwise matches sfname. */
+      if (old_len > new_len
+         && strcmp (sfname, input_filename + old_len - new_len) == 0
+         && (input_filename[old_len - new_len - 1] == '/'
+             || input_filename[old_len - new_len - 1] == '\\'))
+       return;
+    }
+  input_filename = sfname;
+  DECL_SOURCE_FILE (TYPE_NAME (current_class)) = sfname;
+  if (current_class == main_class) main_input_filename = input_filename;
+}
+
+#define HANDLE_SOURCEFILE(INDEX) set_source_filename (jcf, INDEX)
+
+#define HANDLE_CLASS_INFO(ACCESS_FLAGS, THIS, SUPER, INTERFACES_COUNT) \
+{ tree super_class = SUPER==0 ? NULL_TREE : get_class_constant (jcf, SUPER); \
+  current_class = give_name_to_class (jcf, THIS); \
+  set_super_info (ACCESS_FLAGS, current_class, super_class, INTERFACES_COUNT);}
+
+#define HANDLE_CLASS_INTERFACE(INDEX) \
+  add_interface (current_class, get_class_constant (jcf, INDEX))
+
+#define HANDLE_START_FIELD(ACCESS_FLAGS, NAME, SIGNATURE, ATTRIBUTE_COUNT) \
+{ int sig_index = SIGNATURE; \
+  current_field = add_field (current_class, get_name_constant (jcf, NAME), \
+                            parse_signature (jcf, sig_index), ACCESS_FLAGS); \
+ set_java_signature (TREE_TYPE (current_field), JPOOL_UTF (jcf, sig_index)); }
+
+#define HANDLE_END_FIELDS() \
+  (current_field = NULL_TREE)
+
+#define HANDLE_CONSTANTVALUE(INDEX) \
+{ tree constant;  int index = INDEX; \
+  if (JPOOL_TAG (jcf, index) == CONSTANT_String) { \
+    tree name = get_name_constant (jcf, JPOOL_USHORT1 (jcf, index)); \
+    constant = build_utf8_ref (name); \
+  } \
+  else \
+    constant = get_constant (jcf, index); \
+  set_constant_value (current_field, constant); }
+
+#define HANDLE_METHOD(ACCESS_FLAGS, NAME, SIGNATURE, ATTRIBUTE_COUNT) \
+ (current_method = add_method (current_class, ACCESS_FLAGS, \
+                              get_name_constant (jcf, NAME), \
+                              get_name_constant (jcf, SIGNATURE)), \
+  DECL_LOCALVARIABLES_OFFSET (current_method) = 0, \
+  DECL_LINENUMBERS_OFFSET (current_method) = 0)
+
+#define HANDLE_END_METHODS() \
+{ tree handle_type = CLASS_TO_HANDLE_TYPE (current_class); \
+  if (handle_type != current_class) layout_type (handle_type); }
+
+#define HANDLE_CODE_ATTRIBUTE(MAX_STACK, MAX_LOCALS, CODE_LENGTH) \
+{ DECL_MAX_STACK (current_method) = (MAX_STACK); \
+  DECL_MAX_LOCALS (current_method) = (MAX_LOCALS); \
+  DECL_CODE_LENGTH (current_method) = (CODE_LENGTH); \
+  DECL_CODE_OFFSET (current_method) = JCF_TELL (jcf); }
+
+#define HANDLE_LOCALVARIABLETABLE_ATTRIBUTE(COUNT) \
+{ int n = (COUNT); \
+  DECL_LOCALVARIABLES_OFFSET (current_method) = JCF_TELL (jcf) - 2; \
+  JCF_SKIP (jcf, n * 10); }
+
+#define HANDLE_LINENUMBERTABLE_ATTRIBUTE(COUNT) \
+{ int n = (COUNT); \
+  DECL_LINENUMBERS_OFFSET (current_method) = JCF_TELL (jcf) - 2; \
+  JCF_SKIP (jcf, n * 4); }
+
+#include "jcf-reader.c"
+
+static int yydebug;
+
+tree
+parse_signature (jcf, sig_index)
+     JCF *jcf;
+     int sig_index;
+{
+  if (sig_index <= 0 || sig_index >= JPOOL_SIZE(jcf)
+      || JPOOL_TAG (jcf, sig_index) != CONSTANT_Utf8)
+    fatal ("invalid field/method signature");
+  else
+    {
+      return parse_signature_string (JPOOL_UTF_DATA (jcf, sig_index),
+                                    JPOOL_UTF_LENGTH (jcf, sig_index));
+    }
+}
+
+void
+init_lex ()
+{
+  /* Make identifier nodes long enough for the language-specific slots.  */
+  set_identifier_size (sizeof (struct lang_identifier));
+}
+
+void
+set_yydebug (value)
+     int value;
+{
+  yydebug = value;
+}
+
+tree
+get_constant (jcf, index)
+  JCF *jcf;
+  int index;
+{
+  tree value;
+  int tag;
+  if (index <= 0 || index >= JPOOL_SIZE(jcf))
+    goto bad;
+  tag = JPOOL_TAG (jcf, index);
+  if ((tag & CONSTANT_ResolvedFlag) || tag == CONSTANT_Utf8)
+    return (tree) jcf->cpool.data[index];
+  push_obstacks (&permanent_obstack, &permanent_obstack);
+  switch (tag)
+    {
+    case CONSTANT_Integer:
+      {
+       jint num = JPOOL_INT(jcf, index);
+       value = build_int_2 (num, num < 0 ? -1 : 0);
+       TREE_TYPE (value) = int_type_node;
+       break;
+      }
+    case CONSTANT_Long:
+      {
+       jint num = JPOOL_INT (jcf, index);
+       HOST_WIDE_INT lo, hi;
+       lshift_double (num, 0, 32, 64, &lo, &hi, 0);
+       num = JPOOL_INT (jcf, index+1);
+       add_double (lo, hi, num, 0, &lo, &hi);
+       value = build_int_2 (lo, hi);
+       TREE_TYPE (value) = long_type_node;
+       force_fit_type (value, 0);
+       break;
+      }
+#if TARGET_FLOAT_FORMAT == IEEE_FLOAT_FORMAT
+    case CONSTANT_Float:
+      {
+       jint num = JPOOL_INT(jcf, index);
+       REAL_VALUE_TYPE d;
+#ifdef REAL_ARITHMETIC
+       d = REAL_VALUE_FROM_TARGET_SINGLE (num);
+#else
+       union { float f;  jint i; } u;
+       u.i = num;
+       d = u.f;
+#endif
+       value = build_real (float_type_node, d);
+       break;
+      }
+    case CONSTANT_Double:
+      {
+       HOST_WIDE_INT num[2];
+       REAL_VALUE_TYPE d;
+       HOST_WIDE_INT lo, hi;
+       num[0] = JPOOL_INT (jcf, index);
+       lshift_double (num[0], 0, 32, 64, &lo, &hi, 0);
+       num[0] = JPOOL_INT (jcf, index+1);
+       add_double (lo, hi, num[0], 0, &lo, &hi);
+       if (FLOAT_WORDS_BIG_ENDIAN)
+         {
+           num[0] = hi;
+           num[1] = lo;
+         }
+       else
+         {
+           num[0] = lo;
+           num[1] = hi;
+         }
+#ifdef REAL_ARITHMETIC
+       d = REAL_VALUE_FROM_TARGET_DOUBLE (num);
+#else
+       union { double d;  jint i[2]; } u;
+       u.i[0] = (jint) num[0];
+       u.i[1] = (jint) num[1];
+       d = u.d;
+#endif
+       value = build_real (double_type_node, d);
+       break;
+      }
+#endif /* TARGET_FLOAT_FORMAT == IEEE_FLOAT_FORMAT */
+    case CONSTANT_String:
+      {
+       extern struct obstack *expression_obstack;
+       tree name = get_name_constant (jcf, JPOOL_USHORT1 (jcf, index));
+       char *utf8_ptr = IDENTIFIER_POINTER (name);
+       unsigned char *str_ptr;
+       int utf8_len = IDENTIFIER_LENGTH (name);
+       unsigned char *str = (unsigned char*)utf8_ptr;
+       int i = utf8_len;
+       int str_len;
+
+       /* Count the number of Unicode characters in the string,
+          while checking for a malformed Utf8 string. */
+       for (str_len = 0; i > 0; str_len++)
+         {
+           int char_len = UT8_CHAR_LENGTH (*str);
+           if (char_len < 0 || char_len > 2 || char_len > i)
+             fatal ("bad string constant");
+           str += char_len;
+           i -= char_len;
+         }
+
+       value = make_node (STRING_CST);
+       TREE_STRING_LENGTH (value) = 2 * str_len;
+       TREE_STRING_POINTER (value)
+         = obstack_alloc (expression_obstack, 2 * str_len);
+       str_ptr = (unsigned char *) TREE_STRING_POINTER (value);
+       str = (unsigned char*)utf8_ptr;
+       for (i = 0; i < str_len; i++)
+         {
+           int char_value;
+           int char_len = UT8_CHAR_LENGTH (*str);
+           switch (char_len)
+             {
+             case 1:
+               char_value = *str++;
+               break;
+             case 2:
+               char_value = *str++ & 0x1F;
+               char_value = (char_value << 6) | (*str++ & 0x3F);
+               break;
+             case 3:
+               char_value = *str_ptr++ & 0x0F;
+               char_value = (char_value << 6) | (*str++ & 0x3F);
+               char_value = (char_value << 6) | (*str++ & 0x3F);
+               break;
+             default:
+               goto bad;
+             }
+           if (BYTES_BIG_ENDIAN)
+             {
+               *str_ptr++ = char_value >> 8;
+               *str_ptr++ = char_value & 0xFF;
+             }
+           else
+             {
+               *str_ptr++ = char_value & 0xFF;
+               *str_ptr++ = char_value >> 8;
+             }
+         }
+      }
+      break;
+    default:
+      goto bad;
+    }
+  pop_obstacks ();
+  JPOOL_TAG(jcf, index) = tag | CONSTANT_ResolvedFlag;
+  jcf->cpool.data [index] = (jword) value;
+  return value;
+ bad:
+  fatal ("bad value constant type %d, index %d", 
+        JPOOL_TAG( jcf, index ), index);
+}
+
+tree
+get_name_constant (jcf, index)
+  JCF *jcf;
+  int index;
+{
+  tree name = get_constant (jcf, index);
+  if (TREE_CODE (name) != IDENTIFIER_NODE)
+    fatal ("bad nameandtype index %d", index);
+  return name;
+}
+
+static tree
+give_name_to_class (jcf, i)
+     JCF *jcf;
+     int i;
+{
+  if (i <= 0 || i >= JPOOL_SIZE(jcf)
+      || JPOOL_TAG (jcf, i) != CONSTANT_Class)
+    fatal ("bad class index %d", i);
+  else
+    {
+      tree this_class;
+      int j = JPOOL_USHORT1 (jcf, i);
+      /* verify_constant_pool confirmed that j is a CONSTANT_Utf8. */
+      tree class_name = unmangle_classname (JPOOL_UTF_DATA (jcf, j),
+                                           JPOOL_UTF_LENGTH (jcf, j));
+      this_class = lookup_class (class_name);
+      input_filename = DECL_SOURCE_FILE (TYPE_NAME (this_class));
+      lineno = 0;
+      if (main_input_filename == NULL && jcf == main_jcf)
+       main_input_filename = input_filename;
+
+      jcf->cpool.data[i] = (jword) this_class;
+      JPOOL_TAG (jcf, i) = CONSTANT_ResolvedClass;
+      return this_class;
+    }
+}
+
+/* Get the class of the CONSTANT_Class whose constant pool index is I. */
+
+tree
+get_class_constant (JCF *jcf , int i)
+{
+  tree type;
+  if (i <= 0 || i >= JPOOL_SIZE(jcf)
+      || (JPOOL_TAG (jcf, i) & ~CONSTANT_ResolvedFlag) != CONSTANT_Class)
+    fatal ("bad class index %d", i);
+
+  if (JPOOL_TAG (jcf, i) != CONSTANT_ResolvedClass)
+    {
+      int name_index = JPOOL_USHORT1 (jcf, i);
+      /* verify_constant_pool confirmed that name_index is a CONSTANT_Utf8. */
+      char *name = JPOOL_UTF_DATA (jcf, name_index);
+      int nlength = JPOOL_UTF_LENGTH (jcf, name_index);
+      if (name[0] == '[')  /* Handle array "classes". */
+         type = parse_signature_string (name, nlength);
+      else
+        { 
+          tree cname = unmangle_classname (name, nlength);
+          type = lookup_class (cname);
+       }
+      jcf->cpool.data[i] = (jword) type;
+      JPOOL_TAG (jcf, i) = CONSTANT_ResolvedClass;
+    }
+  else
+    type = (tree) jcf->cpool.data[i];
+  return type;
+}
+
+void
+fix_classpath ()
+{
+  static char default_path[] = DEFAULT_CLASS_PATH;
+
+  if (classpath == NULL)
+    {
+      classpath = (char *) getenv ("CLASSPATH");
+      if (classpath == NULL)
+       {
+         warning ("CLASSPATH not set");
+         classpath = default_path;
+       }
+    }
+}
+
+void
+DEFUN(jcf_out_of_synch, (jcf),
+      JCF *jcf)
+{
+  char *source = strdup (jcf->filename);
+  int i = strlen (source);
+
+  while (source[i] != '.')
+    i--;
+
+  source [i] = '\0';
+  warning ("Class file `%s' out of synch with `%s.java'", 
+          jcf->filename, source);
+  free (source);
+}
+
+/* Load CLASS_OR_NAME. CLASS_OR_NAME can be a mere identifier if
+   called from the parser, otherwise it's a RECORD_TYPE node. If
+   VERBOSE is 1, print error message on failure to load a class. */
+
+void
+load_class (class_or_name, verbose)
+     tree class_or_name;
+     int verbose;
+{
+  JCF this_jcf, *jcf;
+  tree name = (TREE_CODE (class_or_name) == IDENTIFIER_NODE ?
+              class_or_name : DECL_NAME (TYPE_NAME (class_or_name)));
+  tree save_current_class = current_class;
+  char *save_input_filename = input_filename;
+  JCF *save_current_jcf = current_jcf;
+  long saved_pos;
+  if (current_jcf->read_state)
+    saved_pos = ftell (current_jcf->read_state);
+
+  push_obstacks (&permanent_obstack, &permanent_obstack);
+
+  if (!classpath)
+    fix_classpath ();
+  /* Search in current zip first.  */
+  if (find_in_current_zip (IDENTIFIER_POINTER (name),
+                          IDENTIFIER_LENGTH (name), &jcf) == 0)
+    if (find_class (IDENTIFIER_POINTER (name), IDENTIFIER_LENGTH (name),
+                    &this_jcf, 1) == 0)
+      {
+       if (verbose)
+         {
+           error ("Cannot find class file class %s.", 
+                  IDENTIFIER_POINTER (name));
+           TYPE_SIZE (class_or_name) = error_mark_node;
+           if (!strcmp (classpath, DEFAULT_CLASS_PATH))
+             fatal ("giving up");
+           pop_obstacks ();    /* FIXME: one pop_obstack() per function */
+         }
+       return;
+      }
+    else
+      {
+        this_jcf.seen_in_zip = 0;
+        current_jcf = &this_jcf;
+       if (this_jcf.outofsynch)
+         jcf_out_of_synch (current_jcf);
+      }
+  else
+    current_jcf = jcf;
+
+  if (current_jcf->java_source)
+    jcf_parse_source (current_jcf);
+  else {
+    int saved_lineno = lineno;
+    input_filename = current_jcf->filename;
+    jcf_parse (current_jcf);
+    lineno = saved_lineno;
+  }
+
+  if (!current_jcf->seen_in_zip)
+    JCF_FINISH (current_jcf);
+/*  DECL_IGNORED_P (TYPE_NAME (class_or_name)) = 1;*/
+  pop_obstacks ();
+
+  current_class = save_current_class;
+  input_filename = save_input_filename;
+  current_jcf = save_current_jcf;
+  if (current_jcf->read_state)
+    fseek (current_jcf->read_state, saved_pos, SEEK_SET);
+}
+
+/* Parse a source file when JCF refers to a source file. This piece
+   needs further work as far as error handling and report. */
+
+int
+jcf_parse_source (jcf)
+     JCF *jcf;
+{
+  java_parser_context_save_global ();
+
+  input_filename = current_jcf->filename;
+  if (!(finput = fopen (input_filename, "r")))
+    fatal ("input file `%s' just disappeared - jcf_parse_source",
+          input_filename);
+
+  parse_source_file (1);       /* Parse only */
+  if (current_class && TREE_TYPE (current_class))
+    CLASS_FROM_SOURCE_P (TREE_TYPE (current_class)) = 1;
+
+  fclose (finput);
+  java_parser_context_restore_global ();
+}
+
+/* Parse the .class file JCF. */
+
+int
+jcf_parse (jcf)
+     JCF* jcf;
+{
+  int i, code;
+
+  if (jcf_parse_preamble (jcf) != 0)
+    fatal ("Not a valid Java .class file.\n");
+  code = jcf_parse_constant_pool (jcf);
+  if (code != 0)
+    fatal ("error while parsing constant pool");
+  code = verify_constant_pool (jcf);
+  if (code > 0)
+    fatal ("error in constant pool entry #%d\n", code);
+
+  jcf_parse_class (jcf);
+  if (main_class == NULL_TREE)
+    main_class = current_class;
+  if (! quiet_flag && TYPE_NAME (current_class))
+    fprintf (stderr, " class %s",
+            IDENTIFIER_POINTER (DECL_NAME (TYPE_NAME (current_class))));
+  CLASS_LOADED_P (current_class) = 1;
+
+  for (i = 1; i < JPOOL_SIZE(jcf); i++)
+    {
+      switch (JPOOL_TAG (jcf, i))
+       {
+       case CONSTANT_Class:
+         get_class_constant (jcf, i);
+         break;
+       }
+    }
+  
+  code = jcf_parse_fields (jcf);
+  if (code != 0)
+    fatal ("error while parsing fields");
+  code = jcf_parse_methods (jcf);
+  if (code != 0)
+    fatal ("error while parsing methods");
+  code = jcf_parse_final_attributes (jcf);
+  if (code != 0)
+    fatal ("error while parsing final attributes");
+
+  /* The fields of class_type_node are already in correct order. */
+  if (current_class != class_type_node && current_class != object_type_node)
+    TYPE_FIELDS (current_class) = nreverse (TYPE_FIELDS (current_class));
+
+  push_obstacks (&permanent_obstack, &permanent_obstack);
+  layout_class (current_class);
+  pop_obstacks ();
+}
+
+void
+init_outgoing_cpool ()
+{
+  current_constant_pool_data_ref = NULL_TREE; 
+  if (outgoing_cpool == NULL)
+    {
+      static CPool outgoing_cpool_buffer;
+      outgoing_cpool = &outgoing_cpool_buffer;
+      CPOOL_INIT(outgoing_cpool);
+    }
+  else
+    {
+      CPOOL_REINIT(outgoing_cpool);
+    }
+}
+
+void
+parse_class_file ()
+{
+  tree method;
+  char *save_input_filename = input_filename;
+  int save_lineno = lineno;
+
+  input_filename = DECL_SOURCE_FILE (TYPE_NAME (current_class));
+  lineno = 0;
+  debug_start_source_file (input_filename);
+  init_outgoing_cpool ();
+
+  for ( method = TYPE_METHODS (CLASS_TO_HANDLE_TYPE (current_class));
+       method != NULL_TREE; method = TREE_CHAIN (method))
+    {
+      JCF *jcf = current_jcf;
+
+      if (METHOD_NATIVE (method) || METHOD_ABSTRACT (method))
+       continue;
+
+      if (DECL_CODE_OFFSET (method) == 0)
+       {
+         error ("missing Code attribute");
+         continue;
+       }
+
+      lineno = 0;
+      if (DECL_LINENUMBERS_OFFSET (method))
+       {
+         register int i;
+         register unsigned char *ptr;
+         JCF_SEEK (jcf, DECL_LINENUMBERS_OFFSET (method));
+         linenumber_count = i = JCF_readu2 (jcf);
+         linenumber_table = ptr = jcf->read_ptr;
+
+         for (ptr += 2; --i >= 0; ptr += 4)
+           {
+             int line = GET_u2 (ptr);
+             /* Set initial lineno lineno to smallest linenumber.
+              * Needs to be set before init_function_start. */
+             if (lineno == 0 || line < lineno)
+               lineno = line;
+           }  
+       }
+      else
+       {
+         linenumber_table = NULL;
+         linenumber_count = 0;
+       }
+
+      start_java_method (method);
+
+      give_name_to_locals (jcf);
+
+      /* Actually generate code. */
+      expand_byte_code (jcf, method);
+
+      end_java_method ();
+    }
+
+  if (flag_emit_class_files)
+    write_classfile (current_class);
+  make_class_data (current_class);
+  register_class ();
+  rest_of_decl_compilation (TYPE_NAME (current_class), (char*) 0, 1, 0);
+
+  debug_end_source_file (save_lineno);
+  input_filename = save_input_filename;
+  lineno = save_lineno;
+}
+
+/* Parse a source file, as pointed by the current JCF. If PARSE_ONLY
+   is non zero, we're not parsing a file found on the command line and
+   we skip things related to code generation. */
+
+static void
+parse_source_file (parse_only)
+     int parse_only;
+{
+  lang_init_source (1);                    /* Error msgs have no method prototypes */
+  java_push_parser_context ();
+  java_init_lex ();                /* Initialize the parser */
+  java_parse_abort_on_error ();
+  java_parse ();                   /* Parse and build partial tree nodes. */
+  java_parse_abort_on_error ();
+  java_complete_class ();          /* Parse unsatisfied class decl. */
+  java_parse_abort_on_error ();
+  java_check_circular_reference (); /* Check on circular references */
+  java_parse_abort_on_error ();
+  java_check_methods ();            /* Check the methods */
+  java_parse_abort_on_error ();
+  java_layout_classes ();
+  java_parse_abort_on_error ();
+  if (!parse_only)
+    {
+      lang_init_source (2);            /* Error msgs have method prototypes */
+      java_complete_expand_methods ();  /* Complete and expand method bodies */
+      java_parse_abort_on_error ();
+      java_expand_finals ();           /* Expand and check the finals */
+      java_parse_abort_on_error ();
+      java_check_final ();              /* Check unitialized final  */
+      java_parse_abort_on_error ();
+      if (! flag_emit_class_files)
+       emit_register_class ();
+      java_report_errors ();           /* Final step for this file */
+    }
+  if (flag_emit_class_files)
+    write_classfile (current_class);
+  java_pop_parser_context ();
+}
+
+int
+yyparse ()
+{
+  /* Everything migh be enclosed within a loop processing each file after
+     the other one.  */
+
+  switch (jcf_figure_file_type (current_jcf))
+    {
+    case JCF_ZIP:
+      parse_zip_file_entries ();
+      emit_register_class ();
+      break;
+    case JCF_CLASS:
+      jcf_parse (current_jcf);
+      parse_class_file ();
+      emit_register_class ();
+      break;
+    case JCF_SOURCE:
+      parse_source_file (0);   /* Parse and generate */
+      break;
+    }
+  return 0;
+}
+
+static struct ZipFileCache *localToFile;
+
+/* Process all class entries found in the zip file.  */
+void
+parse_zip_file_entries (void)
+{
+  struct ZipDirectory *zdir;
+  int i;
+
+  for (i = 0, zdir = (ZipDirectory *)localToFile->z.central_directory;
+       i < localToFile->z.count; i++, zdir = ZIPDIR_NEXT (zdir))
+    {
+      tree class;
+      
+      /* We don't need to consider those files.  */
+      if (!zdir->size || !zdir->filename_offset)
+       continue;
+
+      class = lookup_class (get_identifier (ZIPDIR_FILENAME (zdir)));
+      current_jcf = TYPE_LANG_SPECIFIC (class)->jcf;
+      current_class = class;
+
+      if ( !CLASS_LOADED_P (class))
+       {
+          fseek (current_jcf->read_state, current_jcf->zip_offset, SEEK_SET);
+         jcf_parse (current_jcf);
+       }
+
+      input_filename = current_jcf->filename;
+
+      parse_class_file ();
+      FREE (current_jcf->buffer); /* No longer necessary */
+      /* Note: there is a way to free this buffer right after a class seen
+        in a zip file has been parsed. The idea is the set its jcf in such
+        a way that buffer will be reallocated the time the code for the class
+        will be generated. FIXME.  */
+    }
+}
+
+/* Read all the entries of the zip file, creates a class and a JCF. Sets the
+   jcf up for further processing and link it to the created class.  */
+
+void process_zip_dir()
+{
+  int i;
+  ZipDirectory *zdir;
+
+  for (i = 0, zdir = (ZipDirectory *)localToFile->z.central_directory;
+       i < localToFile->z.count; i++, zdir = ZIPDIR_NEXT (zdir))
+    {
+      char *class_name, *file_name, *class_name_in_zip_dir;
+      tree class;
+      JCF  *jcf;
+      int   j;
+
+      class_name_in_zip_dir = ZIPDIR_FILENAME (zdir);
+
+      /* We choose to not to process entries with a zero size or entries
+        not bearing the .class extention.  */
+      if (!zdir->size || !zdir->filename_offset ||
+         strncmp (&class_name_in_zip_dir[zdir->filename_length-6], 
+                  ".class", 6))
+       {
+         /* So it will be skipped in parse_zip_file_entries  */
+         zdir->size = 0;  
+         continue;
+       }
+
+      class_name = ALLOC (zdir->filename_length+1-6);
+      file_name  = ALLOC (zdir->filename_length+1);
+      jcf = ALLOC (sizeof (JCF));
+      JCF_ZERO (jcf);
+
+      strncpy (class_name, class_name_in_zip_dir, zdir->filename_length-6);
+      class_name [zdir->filename_length-6] = '\0';
+      strncpy (file_name, class_name_in_zip_dir, zdir->filename_length);
+      file_name [zdir->filename_length] = '\0';
+
+      for (j=0; class_name[j]; j++)
+        class_name [j] = (class_name [j] == '/' ? '.' : class_name [j]);
+
+      /* Yes, we write back the true class name into the zip directory.  */
+      strcpy (class_name_in_zip_dir, class_name);
+      zdir->filename_length = j;
+      class = lookup_class (get_identifier (class_name));
+
+      jcf->read_state  = finput;
+      jcf->filbuf      = jcf_filbuf_from_stdio;
+      jcf->seen_in_zip = 1;
+      jcf->java_source = 0;
+      jcf->zip_offset  = zdir->filestart;
+      jcf->classname   = class_name;
+      jcf->filename    = file_name;
+
+      TYPE_LANG_SPECIFIC (class) = 
+        (struct lang_type *) perm_calloc (1, sizeof (struct lang_type));
+      TYPE_LANG_SPECIFIC (class)->jcf = jcf;
+    }
+}
+
+/* Lookup class NAME and figure whether is a class already found in the current
+   zip file.  */
+int
+DEFUN(find_in_current_zip, (name, length, jcf),
+      char *name AND int length AND JCF **jcf)
+{
+  JCF *local_jcf;
+  tree class_name = maybe_get_identifier (name), class, icv;
+
+  if (!class_name)
+    return 0;
+
+  if (!(icv = IDENTIFIER_CLASS_VALUE (class_name)))
+    return 0;
+
+  class = TREE_TYPE (icv);
+
+  /* Doesn't have jcf specific info ? It's not ours */
+  if (!TYPE_LANG_SPECIFIC (class) || !TYPE_LANG_SPECIFIC (class)->jcf)
+    return 0;
+
+  *jcf = local_jcf = TYPE_LANG_SPECIFIC (class)->jcf;
+  fseek (local_jcf->read_state, local_jcf->zip_offset, SEEK_SET);
+  return 1;
+}
+
+/* Figure what kind of file we're dealing with */
+int
+DEFUN(jcf_figure_file_type, (jcf),
+      JCF *jcf)
+{
+  unsigned char magic_string[4];
+  uint32 magic;
+
+  if (fread (magic_string, 1, 4, jcf->read_state) != 4)
+    jcf_unexpected_eof (jcf, 4);
+
+  fseek (jcf->read_state, 0L, SEEK_SET);
+  magic = GET_u4 (magic_string);
+
+  if (magic == 0xcafebabe)
+    return JCF_CLASS;
+
+  if (!open_in_zip (jcf, input_filename, NULL))
+    {
+      localToFile = ALLOC (sizeof (struct ZipFileCache));
+      bcopy (SeenZipFiles, localToFile, sizeof (struct ZipFileCache));
+      process_zip_dir ();      /* Register all the class defined there */
+      return JCF_ZIP;
+    }
+
+  return JCF_SOURCE;
+}
+
diff --git a/gcc/java/jcf-reader.c b/gcc/java/jcf-reader.c
new file mode 100644 (file)
index 0000000..cf5c042
--- /dev/null
@@ -0,0 +1,352 @@
+/* This file read a Java(TM) .class file.
+   It is not stand-alone:  It depends on tons of macros, and the
+   intent is you #include this file after you've defined the macros.
+
+   Copyright (C) 1996  Free Software Foundation, Inc.
+
+This program is free software; you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation; either version 2, or (at your option)
+any later version.
+
+This program is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with GNU CC; see the file COPYING.  If not, write to
+the Free Software Foundation, 59 Temple Place - Suite 330,
+Boston, MA 02111-1307, USA.  
+
+Java and all Java-based marks are trademarks or registered trademarks
+of Sun Microsystems, Inc. in the United States and other countries.
+The Free Software Foundation is independent of Sun Microsystems, Inc.  */
+
+#include "jcf.h"
+#include "zipfile.h"
+
+int
+DEFUN(get_attribute, (jcf),
+      JCF *jcf)
+{
+  uint16 attribute_name = (JCF_FILL (jcf, 6), JCF_readu2 (jcf));
+  uint32 attribute_length = JCF_readu4 (jcf);
+  uint32 start_pos = JCF_TELL(jcf);
+  int name_length;
+  unsigned char *name_data;
+  JCF_FILL (jcf, attribute_length);
+  if (attribute_name <= 0 || attribute_name >= JPOOL_SIZE(jcf))
+    return -2;
+  if (JPOOL_TAG (jcf, attribute_name) != CONSTANT_Utf8)
+    return -2;
+  name_length = JPOOL_UTF_LENGTH (jcf, attribute_name);
+  name_data = JPOOL_UTF_DATA (jcf, attribute_name);
+
+#ifdef IGNORE_ATTRIBUTE
+   if (IGNORE_ATTRIBUTE (jcf, attribute_name, attribute_length))
+     {
+       JCF_SKIP (jcf, attribute_length);
+     }
+   else
+#endif
+#ifdef HANDLE_SOURCEFILE
+  if (name_length == 10 && memcmp (name_data, "SourceFile", 10) == 0)
+    {
+      uint16 sourcefile_index = JCF_readu2 (jcf);
+      HANDLE_SOURCEFILE(sourcefile_index);
+    }
+  else
+#endif
+#ifdef HANDLE_CONSTANTVALUE
+  if (name_length == 13 && memcmp (name_data, "ConstantValue", 13) == 0)
+    {
+      uint16 constantvalue_index = JCF_readu2 (jcf);
+      if (constantvalue_index <= 0 || constantvalue_index >= JPOOL_SIZE(jcf))
+       return -2;
+      HANDLE_CONSTANTVALUE(constantvalue_index);
+    }
+  else
+#endif
+#ifdef HANDLE_CODE_ATTRIBUTE
+  if (name_length == 4 && memcmp (name_data, "Code", 4) == 0)
+    {
+      uint16 j;
+      uint16 max_stack = JCF_readu2 (jcf);
+      uint16 max_locals = JCF_readu2 (jcf);
+      uint32 code_length = JCF_readu4 (jcf);
+      uint16 exception_table_length, attributes_count;
+      if (code_length + 12 > attribute_length)
+       return -1;
+      HANDLE_CODE_ATTRIBUTE(max_stack, max_locals, code_length);
+      JCF_SKIP (jcf, code_length);
+      exception_table_length = JCF_readu2 (jcf);
+      if (code_length + 8 * exception_table_length + 12 > attribute_length)
+       return -1;
+      JCF_SKIP (jcf, 2 * 4 * exception_table_length);
+      attributes_count = JCF_readu2 (jcf);
+      for (j = 0; j < attributes_count; j++)
+       {
+         int code = get_attribute (jcf);
+         if (code != 0)
+           return code;
+       }
+    }
+  else
+#endif /* HANDLE_CODE_ATTRIBUTE */
+#ifdef HANDLE_EXCEPTIONS_ATTRIBUTE
+  if (name_length == 10 && memcmp (name_data, "Exceptions", 10) == 0)
+    {
+      uint16 count = JCF_readu2 (jcf);
+      HANDLE_EXCEPTIONS_ATTRIBUTE (count);
+    }
+  else
+#endif
+#ifdef HANDLE_LINENUMBERTABLE_ATTRIBUTE
+  if (name_length == 15 && memcmp (name_data, "LineNumberTable", 15) == 0)
+    {
+      uint16 count = JCF_readu2 (jcf);
+      HANDLE_LINENUMBERTABLE_ATTRIBUTE (count);
+    }
+  else
+#endif
+#ifdef HANDLE_LOCALVARIABLETABLE_ATTRIBUTE
+  if (name_length == 18 && memcmp (name_data, "LocalVariableTable", 18) == 0)
+    {
+      uint16 count = JCF_readu2 (jcf);
+      HANDLE_LOCALVARIABLETABLE_ATTRIBUTE (count);
+    }
+  else
+#endif
+    {
+#ifdef PROCESS_OTHER_ATTRIBUTE
+      PROCESS_OTHER_ATTRIBUTE(jcf, attribute_name, attribute_length);
+#else
+      JCF_SKIP (jcf, attribute_length);
+#endif
+    }
+  if (start_pos + attribute_length != JCF_TELL(jcf))
+    return -1;
+  return 0;
+}
+
+/* Read and handle the pre-amble. */
+int
+DEFUN(jcf_parse_preamble, (jcf),
+      JCF* jcf)
+{
+  uint32 magic = (JCF_FILL (jcf, 8), JCF_readu4 (jcf));
+  uint16 minor_version = JCF_readu2 (jcf);
+  uint16 major_version = JCF_readu2 (jcf);
+#ifdef HANDLE_MAGIC
+  HANDLE_MAGIC (magic, minor_version, major_version);
+#endif
+  if (magic != 0xcafebabe)
+    return -1;
+  else
+    return 0;
+}
+
+/* Read and handle the constant pool.
+
+   Return 0 if OK.
+   Return -2 if a bad cross-reference (index of other constant) was seen.
+*/
+int
+DEFUN(jcf_parse_constant_pool, (jcf),
+      JCF* jcf)
+{
+  int i, n;
+  JPOOL_SIZE (jcf) = (JCF_FILL (jcf, 2), JCF_readu2 (jcf));
+  jcf->cpool.tags = ALLOC (JPOOL_SIZE (jcf));
+  jcf->cpool.data = ALLOC (sizeof (jword) * JPOOL_SIZE (jcf));
+  jcf->cpool.tags[0] = 0;
+#ifdef HANDLE_START_CONSTANT_POOL
+  HANDLE_START_CONSTANT_POOL (JPOOL_SIZE (jcf));
+#endif
+  for (i = 1; i < (int) JPOOL_SIZE (jcf); i++)
+    {
+      int constant_kind;
+       
+      /* Make sure at least 9 bytes are available.  This is enough
+        for all fixed-sized constant pool entries (so we don't need many
+        more JCF_FILL calls below), but is is small enough that
+        we are guaranteed to not hit EOF (in a valid .class file). */
+      JCF_FILL (jcf, 9);
+      constant_kind = JCF_readu (jcf);
+      jcf->cpool.tags[i] = constant_kind;
+      switch (constant_kind)
+       {
+       case CONSTANT_String:
+       case CONSTANT_Class:
+         jcf->cpool.data[i] = JCF_readu2 (jcf);
+         break;
+       case CONSTANT_Fieldref:
+       case CONSTANT_Methodref:
+       case CONSTANT_InterfaceMethodref:
+       case CONSTANT_NameAndType:
+         jcf->cpool.data[i] = JCF_readu2 (jcf);
+         jcf->cpool.data[i] |= JCF_readu2 (jcf) << 16;
+         break;
+       case CONSTANT_Integer:
+       case CONSTANT_Float:
+         jcf->cpool.data[i] = JCF_readu4 (jcf);
+         break;
+       case CONSTANT_Long:
+       case CONSTANT_Double:
+         jcf->cpool.data[i] = JCF_readu4 (jcf);
+         i++; /* These take up two spots in the constant pool */
+         jcf->cpool.tags[i] = 0;
+         jcf->cpool.data[i] = JCF_readu4 (jcf);
+         break;
+       case CONSTANT_Utf8:
+         n = JCF_readu2 (jcf);
+         JCF_FILL (jcf, n);
+#ifdef HANDLE_CONSTANT_Utf8
+         HANDLE_CONSTANT_Utf8(jcf, i, n);
+#else
+         jcf->cpool.data[i] = JCF_TELL(jcf) - 2;
+         JCF_SKIP (jcf, n);
+#endif
+         break;
+       default:
+         return i;
+       }
+    }
+  return 0;
+}
+
+/* Read various class flags and numbers. */
+
+void
+DEFUN(jcf_parse_class, (jcf),
+      JCF* jcf)
+{
+  int i;
+  uint16 interfaces_count;
+  JCF_FILL (jcf, 8);
+  jcf->access_flags = JCF_readu2 (jcf);
+  jcf->this_class = JCF_readu2 (jcf);
+  jcf->super_class = JCF_readu2 (jcf);
+  interfaces_count = JCF_readu2 (jcf);
+
+#ifdef HANDLE_CLASS_INFO
+  HANDLE_CLASS_INFO(jcf->access_flags, jcf->this_class, jcf->super_class, interfaces_count);
+#endif
+
+  JCF_FILL (jcf, 2 * interfaces_count);
+
+  /* Read interfaces. */
+  for (i = 0; i < interfaces_count; i++)
+    {
+      uint16 index = JCF_readu2 (jcf);
+#ifdef HANDLE_CLASS_INTERFACE
+      HANDLE_CLASS_INTERFACE (index);
+#endif
+    }
+}
+
+/* Read fields. */
+int
+DEFUN(jcf_parse_fields, (jcf),
+      JCF* jcf)
+{
+  int i, j;
+  uint16 fields_count;
+  JCF_FILL (jcf, 2);
+  fields_count = JCF_readu2 (jcf);
+
+#ifdef HANDLE_START_FIELDS
+  HANDLE_START_FIELDS (fields_count);
+#endif
+  for (i = 0; i < fields_count; i++)
+    {
+      uint16 access_flags = (JCF_FILL (jcf, 8), JCF_readu2 (jcf));
+      uint16 name_index = JCF_readu2 (jcf);
+      uint16 signature_index = JCF_readu2 (jcf);
+      uint16 attribute_count = JCF_readu2 (jcf);
+#ifdef HANDLE_START_FIELD
+      HANDLE_START_FIELD (access_flags, name_index, signature_index,
+                   attribute_count);
+#endif
+      for (j = 0; j < attribute_count; j++)
+       {
+         int code = get_attribute (jcf);
+         if (code != 0)
+           return code;
+       }
+#ifdef HANDLE_END_FIELD
+      HANDLE_END_FIELD ();
+#endif
+    }
+#ifdef HANDLE_END_FIELDS
+  HANDLE_END_FIELDS ();
+#endif
+  return 0;
+}
+
+/* Read methods. */
+
+int
+DEFUN(jcf_parse_one_method, (jcf),
+      JCF* jcf)
+{
+  int i;
+  uint16 access_flags = (JCF_FILL (jcf, 8), JCF_readu2 (jcf));
+  uint16 name_index = JCF_readu2 (jcf);
+  uint16 signature_index = JCF_readu2 (jcf);
+  uint16 attribute_count = JCF_readu2 (jcf);
+#ifdef HANDLE_METHOD
+  HANDLE_METHOD(access_flags, name_index, signature_index, attribute_count);
+#endif
+  for (i = 0; i < attribute_count; i++)
+    {
+      int code = get_attribute (jcf);
+      if (code != 0)
+       return code;
+    }
+  return 0;
+}
+
+int
+DEFUN(jcf_parse_methods, (jcf),
+      JCF* jcf)
+{
+  int i;
+  uint16 methods_count;
+  JCF_FILL (jcf, 2);
+  methods_count = JCF_readu2 (jcf);
+#ifdef HANDLE_START_METHODS
+  HANDLE_START_METHODS (methods_count);
+#endif
+  for (i = 0; i < methods_count; i++)
+    {
+      int code = jcf_parse_one_method (jcf);
+      if (code != 0)
+       return code;
+    }
+#ifdef HANDLE_END_METHODS
+  HANDLE_END_METHODS ();
+#endif
+  return 0;
+}
+
+/* Read attributes. */
+int
+DEFUN(jcf_parse_final_attributes, (jcf),
+      JCF *jcf)
+{
+  int i;
+  uint16 attributes_count = (JCF_FILL (jcf, 2), JCF_readu2 (jcf));
+#ifdef START_FINAL_ATTRIBUTES
+  START_FINAL_ATTRIBUTES (attributes_count)
+#endif
+  for (i = 0; i < attributes_count; i++)
+    {
+      int code = get_attribute (jcf);
+      if (code != 0)
+       return code;
+    }
+  return 0;
+}
+
diff --git a/gcc/java/jcf-write.c b/gcc/java/jcf-write.c
new file mode 100644 (file)
index 0000000..931c9d5
--- /dev/null
@@ -0,0 +1,964 @@
+/* Write out a Java(TM) class file.
+   Copyright (C) 1998 Free Software Foundation, Inc.
+
+This file is part of GNU CC.
+
+GNU CC is free software; you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation; either version 2, or (at your option)
+any later version.
+
+GNU CC is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+You should have received a copy of the GNU General Public License
+along with GNU CC; see the file COPYING.  If not, write to
+the Free Software Foundation, 59 Temple Place - Suite 330,
+Boston, MA 02111-1307, USA. 
+
+Java and all Java-based marks are trademarks or registered trademarks
+of Sun Microsystems, Inc. in the United States and other countries.
+The Free Software Foundation is independent of Sun Microsystems, Inc.  */
+
+#include "config.h"
+#include "tree.h"
+#include "java-tree.h"
+#include "jcf.h"
+#include <stdio.h>
+#include "obstack.h"
+#undef AND
+#include "rtl.h"
+#include "java-opcodes.h"
+#include "parse.h" /* for BLOCK_EXPR_BODY */
+#include "buffer.h"
+
+extern struct obstack temporary_obstack;
+
+/* The buffer allocated for bytecode for the current method. */
+
+struct buffer bytecode = NULL_BUFFER;
+
+/* Make sure bytecode.data is big enough for at least N more bytes. */
+
+#define RESERVE(N) \
+  do { if (bytecode.ptr + (N) > bytecode.limit) buffer_grow (&bytecode, N); } while (0)
+
+/* Add a 1-byte instruction/operand I to bytecode.data,
+   assuming space has already been RESERVE'd. */
+
+#define OP1(I) (*bytecode.ptr++ = (I))
+
+/* Like OP1, but I is a 2-byte big endian integer. */
+
+#define OP2(I) \
+  do { int _I = (I);  OP1 (_I >> 8);  OP1 (_I); } while (0)
+
+/* Like OP1, but I is a 4-byte big endian integer. */
+
+#define OP4(I) \
+  do { int _I = (I);  OP1 (_I >> 24);  OP1 (_I >> 16); \
+       OP1 (_I >> 8); OP1 (_I); } while (0)
+
+/* The current stack size (stack pointer) in the current method. */
+
+int code_SP = 0;
+
+/* The largest extent of stack size (stack pointer) in the current method. */
+
+int code_SP_max = 0;
+
+CPool *code_cpool;
+
+/* Macro to call each time we push I words on the JVM stack. */
+
+#define NOTE_PUSH(I) \
+  do { code_SP += (I); if (code_SP > code_SP_max) code_SP_max = code_SP; } while (0)
+
+/* Macro to call each time we pop I words from the JVM stack. */
+
+#define NOTE_POP(I) \
+  do { code_SP -= (I); if (code_SP < 0) abort(); } while (0)
+
+/* A chunk or segment of a .class file. */
+
+struct chunk
+{
+  /* The next segment of this .class file. */
+  struct chunk *next;
+
+  /* The actual data in this segment to be written to the .class file. */
+  unsigned char *data;
+
+  /* The size of the segment to be written to the .class file. */
+  int size;
+};
+
+/* Utility macros for appending (big-endian) data to a buffer.
+   We assume a local variable 'ptr' points into where we want to
+   write next, and we assume enoygh space has been allocated. */
+
+#define PUT1(X)  (*ptr++ = (X))
+#define PUT2(X)  (PUT1((X) >> 8), PUT1((X) & 0xFF))
+#define PUT4(X)  (PUT2((X) >> 16), PUT2((X) & 0xFFFF))
+#define PUTN(P, N)  (bcopy(P, ptr, N), ptr += (N))
+
+\f
+/* A buffer for storing line number entries for the current method. */
+struct buffer linenumbers = NULL_BUFFER;
+
+/* Append a line number entry for the given PC and LINE into
+   linenumbers.data.  This will later before a LineNumberTable attribute. */
+
+void
+put_linenumber (pc, line)
+     int pc, line;
+{
+  register unsigned char *ptr;
+  if (linenumbers.ptr == linenumbers.limit)
+    buffer_grow (&linenumbers, 4);
+  ptr = linenumbers.ptr;
+  PUT2 (pc);
+  PUT2 (line);
+  linenumbers.ptr = ptr;
+}
+\f
+/* The index of jvm local variable allocated for this DECL.
+   This is assign when generating .class files;
+   contrast DECL_LOCAL_SLOT_NUMBER whcih is set when *reading* a .class file.
+   (We don't allocate DECL_LANG_SPECIFIC for locals from Java sourc code.) */
+
+#define DECL_LOCAL_INDEX(DECL) DECL_ALIGN(DECL)
+
+struct localvar_info
+{
+  tree decl;
+
+  int start_pc;
+
+  /* Offset in LocalVariableTable. */
+  int debug_offset;
+};
+
+struct buffer localvars = NULL_BUFFER;
+
+#define localvar_buffer ((struct localvar_info*) localvars.data)
+#define localvar_max ((struct localvar_info*) localvars.ptr - localvar_buffer)
+
+/* A buffer for storing LocalVariableTable entries entries. */
+
+struct buffer localvartable = NULL_BUFFER;
+
+int
+localvar_alloc (decl, start_pc)
+     tree decl;
+     int start_pc;
+{
+  int wide = TYPE_IS_WIDE (TREE_TYPE (decl));
+  int index;
+  register struct localvar_info *info = (struct localvar_info*)localvars.data;
+  register struct localvar_info *limit = (struct localvar_info*)localvars.ptr;
+  for (index = 0;  info < limit;  index++, info++)
+    {
+      if (info->decl == NULL_TREE
+         && (! wide || (info+1)->decl == NULL_TREE))
+       break;
+    }
+  if (info == limit)
+    {
+      buffer_grow (&localvars, sizeof (struct localvar_info));
+      info = (struct localvar_info*)localvars.data + index;
+      localvars.ptr = (unsigned char *) (info + 1 + wide);
+    }
+  info->decl = decl;
+  if (wide)
+    (info+1)->decl = TYPE_SECOND;
+  DECL_LOCAL_INDEX (decl) = index;
+  info->start_pc = start_pc;
+
+  if (DECL_NAME (decl) != NULL_TREE)
+    {
+      /* Generate debugging info. */
+      int i;
+      register unsigned char *ptr;
+      buffer_grow (&localvartable, 10);
+      ptr = localvartable.ptr;
+      info->debug_offset = ptr - localvartable.data;
+      PUT2 (start_pc);
+      PUT2 (0);  /* length - fill in later */
+      i = find_utf8_constant (code_cpool, DECL_NAME (decl));
+      PUT2 (i); /* name_index*/
+      i = find_utf8_constant (code_cpool,
+                             build_java_signature (TREE_TYPE (decl)));
+      PUT2 (i);  /* descriptor_index */
+      PUT2 (index);
+      localvartable.ptr = ptr;
+    }
+  else
+    info->debug_offset = -1;
+}
+
+int
+localvar_free (decl, end_pc)
+     tree decl;
+     int end_pc;
+{
+  register unsigned char *ptr;
+  int index = DECL_LOCAL_INDEX (decl);
+  register struct localvar_info *info = &localvar_buffer [index];
+  int wide = TYPE_IS_WIDE (TREE_TYPE (decl));
+  int i;
+
+  i = info->debug_offset;
+  if (i >= 0)
+    {
+      register unsigned char *ptr;
+      /* Point to length field of local_variable_table. */
+      ptr = localvartable.data + i + 2;
+      i = end_pc - info->start_pc;
+      PUT2 (i);
+    }
+
+  if (info->decl != decl)
+    abort ();
+  info->decl = NULL_TREE;
+  if (wide)
+    {
+      info++;
+      if (info->decl != TYPE_SECOND)
+       abort ();
+      info->decl = NULL_TREE;
+    }
+
+}
+
+\f
+#define STACK_TARGET 1
+#define IGNORE_TARGET 2
+
+/* Allocate a new chunk on obstack WORK, and link it in after LAST.
+   Set the data and size fields to DATA and SIZE, respectively.
+   However, if DATA is NULL and SIZE>0, allocate a buffer as well. */
+
+struct chunk *
+alloc_chunk (last, data, size, work)
+     struct chunk *last;
+     unsigned char *data;
+     int size;
+     struct obstack *work;
+{
+  struct chunk *chunk = (struct chunk *)
+    obstack_alloc (work, sizeof(struct chunk));
+
+  if (data == NULL && size > 0)
+    data = obstack_alloc (work, size);
+
+  chunk->next = NULL;
+  chunk->data = data;
+  chunk->size = size;
+  last->next = chunk;
+  return chunk;
+}
+
+/* Get the access flags of a class (TYPE_DECL), a method (FUNCTION_DECL), or
+   a field (FIELD_DECL or VAR_DECL, if static), as encoded in a .class file. */
+
+int
+get_access_flags (decl)
+    tree decl;
+{
+  int flags = 0;
+  int isfield = TREE_CODE (decl) == FIELD_DECL || TREE_CODE (decl) == VAR_DECL;
+  if (CLASS_PUBLIC (decl))  /* same as FIELD_PUBLIC and METHOD_PUBLIC */
+    flags |= ACC_PUBLIC;
+  if (CLASS_FINAL (decl))  /* same as FIELD_FINAL and METHOD_FINAL */
+    flags |= ACC_PUBLIC;
+  if (isfield || TREE_CODE (decl) == FUNCTION_DECL)
+    {
+      if (TREE_PROTECTED (decl))
+       flags |= ACC_PROTECTED;
+      if (TREE_PRIVATE (decl))
+       flags |= ACC_PRIVATE;
+    }
+  else if (TREE_CODE (decl) == TYPE_DECL)
+    {
+      if (CLASS_SUPER (decl))
+       flags |= ACC_SUPER;
+      if (CLASS_ABSTRACT (decl))
+       flags |= ACC_ABSTRACT;
+      if (CLASS_INTERFACE (decl))
+       flags |= ACC_INTERFACE;
+    }
+  else
+    fatal ("internal error - bad argument to get_access_flags");
+  if (TREE_CODE (decl) == FUNCTION_DECL)
+    {
+      if (METHOD_NATIVE (decl))
+       flags |= ACC_NATIVE;
+      if (METHOD_STATIC (decl))
+       flags |= ACC_STATIC;
+      if (METHOD_FINAL (decl))
+       flags |= ACC_FINAL;
+      if (METHOD_SYNCHRONIZED (decl))
+       flags |= ACC_SYNCHRONIZED;
+      if (METHOD_ABSTRACT (decl))
+       flags |= ACC_ABSTRACT;
+    }
+  if (isfield)
+    {
+      if (FIELD_STATIC (decl))
+       flags |= ACC_STATIC;
+      if (FIELD_VOLATILE (decl))
+       flags |= ACC_VOLATILE;
+      if (FIELD_TRANSIENT (decl))
+       flags |= ACC_TRANSIENT;
+    }
+  return flags;
+}
+
+/* Write the list of segments starting at CHUNKS to STREAM. */
+
+void
+write_chunks (stream, chunks)
+     FILE* stream;
+     struct chunk *chunks;
+{
+  for (;  chunks != NULL;  chunks = chunks->next)
+    fwrite (chunks->data, chunks->size, 1, stream);
+}
+
+void
+push_constant1 (index)
+     int index;
+{
+  if (index < 256)
+    {
+      OP1 (OPCODE_ldc);
+      OP1 (index);
+    }
+  else
+    {
+      OP1 (OPCODE_ldc_w);
+      OP2 (index);
+    }
+}
+
+void
+push_constant2 (index)
+     int index;
+{
+  RESERVE (3);
+  OP1 (OPCODE_ldc2_w);
+  OP2 (index);
+}
+
+void
+push_int_const (i)
+     HOST_WIDE_INT i;
+{
+  RESERVE(3);
+  if (i >= -1 && i <= 5)
+    OP1(OPCODE_iconst_0 + i);
+  else if (i >= -128 && i < 128)
+    {
+      OP1(OPCODE_bipush);
+      OP1(i);
+    }
+  else if (i >= -32768 && i < 32768)
+    {
+      OP1(OPCODE_sipush);
+      OP2(i);
+    }
+  else
+    {
+      i = find_constant1 (code_cpool, CONSTANT_Integer, i & 0xFFFFFFFF);
+      push_constant1 (i);
+    }
+}
+
+void
+push_long_const (lo, hi)
+     HOST_WIDE_INT lo, hi;
+{
+  if (hi == 0 && lo >= 0 && lo <= 1)
+    {
+      RESERVE(1);
+      OP1(OPCODE_lconst_0 + lo);
+    }
+#if 0
+    else if ((jlong) (jint) i == i)
+      {
+        push_int_const ((jint) i);
+        RESERVE (1);
+        OP1 (OPCODE_i2l);
+      }
+#endif
+  else
+    {
+      HOST_WIDE_INT w1, w2;
+      lshift_double (lo, hi, -32, 64, &w1, &w2, 1);
+      hi = find_constant1 (code_cpool, CONSTANT_Long,
+                          w1 & 0xFFFFFFFF, lo & 0xFFFFFFFF);
+      push_constant2 (hi);
+    }
+}
+
+void
+field_op (field, opcode)
+     tree field;
+     int opcode;
+{
+  int index = find_fieldref_index (code_cpool, field);
+  RESERVE (3);
+  OP1 (opcode);
+  OP2 (index);
+}
+
+/* Returns an integer in the range 0 (for 'int') through 4 (for object
+   reference) to 7 (for 'short') which matches the pattern of how JVM
+   opcodes typically depend on the operand type. */
+
+int
+adjust_typed_op (type)
+     tree type;
+{
+  switch (TREE_CODE (type))
+    {
+    case BOOLEAN_TYPE:  return 5;
+    case CHAR_TYPE:     return 6;
+    case POINTER_TYPE:
+    case RECORD_TYPE:   return 4;
+    case INTEGER_TYPE:
+      switch (TYPE_PRECISION (type))
+       {
+       case  8:       return 5;
+       case 16:       return 7;
+       case 32:       return 0;
+       case 64:       return 1;
+       }
+      break;
+    case REAL_TYPE:
+      switch (TYPE_PRECISION (type))
+       {
+       case 32:       return 2;
+       case 64:       return 3;
+       }
+      break;
+    }
+  abort ();
+}
+
+void
+maybe_wide (opcode, index)
+     int opcode, index;
+{
+  if (index >= 256)
+    {
+      RESERVE (4);
+      OP1 (196); /* wide */
+      OP1 (opcode);
+      OP2 (index);
+    }
+  else
+    {
+      RESERVE (2);
+      OP1 (opcode);
+      OP1 (index);
+    }
+}
+
+#define PC BUFFER_LENGTH(&bytecode)
+
+/* Generate byetcode for sub-expression EXP of METHOD.
+   TARGET is one of STACK_TARGET or IGNORE_TARGET. */
+
+void
+generate_bytecode_insns (method, exp, target)
+     tree method;
+     tree exp;
+     int target;
+{
+  rtx value;
+  tree type = TREE_TYPE (exp);
+  enum java_opcode jopcode;
+  int op;
+  switch (TREE_CODE (exp))
+    {
+    case BLOCK:
+      if (BLOCK_EXPR_BODY (exp))
+       {
+         tree local;
+         for (local = BLOCK_EXPR_DECLS (exp); local; )
+           {
+             tree next = TREE_CHAIN (local);
+             localvar_alloc (local, PC);
+             local = next;
+           }
+         generate_bytecode_insns (method, BLOCK_EXPR_BODY (exp), target);
+         for (local = BLOCK_EXPR_DECLS (exp); local; )
+           {
+             tree next = TREE_CHAIN (local);
+             localvar_free (local, PC);
+             local = next;
+           }
+       }
+      break;
+      case COMPOUND_EXPR:      
+       generate_bytecode_insns (method, TREE_OPERAND (exp, 0), IGNORE_TARGET);
+       generate_bytecode_insns (method, TREE_OPERAND (exp, 1), target);
+      break;
+    case EXPR_WITH_FILE_LOCATION:
+      {
+       char *saved_input_filename = input_filename;
+       int saved_lineno = lineno;
+       input_filename = EXPR_WFL_FILENAME (exp);
+       lineno = EXPR_WFL_LINENO (exp);
+       if (EXPR_WFL_EMIT_LINE_NOTE (exp))
+         put_linenumber (PC, EXPR_WFL_LINENO (exp));
+       generate_bytecode_insns (method, EXPR_WFL_NODE (exp), target);
+       input_filename = saved_input_filename;
+       lineno = saved_lineno;
+      }
+      break;
+    case INTEGER_CST:
+      if (target == IGNORE_TARGET) ; /* do nothing */
+      else if (TREE_CODE (type) == POINTER_TYPE)
+       {
+         if (! integer_zerop (exp))
+           abort();
+         RESERVE(1);
+         OP1 (OPCODE_aconst_null);
+         NOTE_PUSH (1);
+       }
+      else if (TYPE_PRECISION (type) <= 32)
+       {
+         push_int_const (TREE_INT_CST_LOW (exp));
+         NOTE_PUSH (1);
+       }
+      else
+       {
+         push_long_const (TREE_INT_CST_LOW (exp), TREE_INT_CST_HIGH (exp));
+         NOTE_PUSH (2);
+       }
+      break;
+    case VAR_DECL:
+      if (TREE_STATIC (exp))
+       {
+         field_op (exp, OPCODE_getstatic);
+         break;
+       }
+      /* ... fall through ... */
+    case PARM_DECL:
+      {
+       int kind = adjust_typed_op (type);
+       int index = DECL_LOCAL_INDEX (exp);
+       if (index <= 3)
+         {
+           RESERVE (1);
+           OP1 (26 + 4 * kind + index);    /* [ilfda]load_[0123] */
+         }
+       else
+         maybe_wide (21 + kind, index);  /* [ilfda]load */
+      }
+      break;
+    case INDIRECT_REF:
+      generate_bytecode_insns (method, TREE_OPERAND (exp, 0), target);
+      break;
+    case ARRAY_REF:
+      generate_bytecode_insns (method, TREE_OPERAND (exp, 0), target);
+      generate_bytecode_insns (method, TREE_OPERAND (exp, 1), target);
+      if (target != IGNORE_TARGET)
+       {
+         jopcode = OPCODE_iaload + adjust_typed_op (type);
+         RESERVE(1);
+         OP1 (jopcode);
+       }
+      break;
+    case COMPONENT_REF:
+      {
+       tree obj = TREE_OPERAND (exp, 0);
+       tree field = TREE_OPERAND (exp, 1);
+       int is_static = FIELD_STATIC (field);
+       generate_bytecode_insns (method, obj,
+                                is_static ? IGNORE_TARGET : target);
+       if (target != IGNORE_TARGET)
+         {
+           if (DECL_NAME (field) == length_identifier_node && !is_static
+               && TYPE_ARRAY_P (TREE_TYPE (obj)))
+             {
+               RESERVE (1);
+               OP1 (OPCODE_arraylength);
+             }
+           else
+             field_op (field, is_static ? OPCODE_getstatic : OPCODE_getfield);
+         }
+      }
+      break;
+    case RETURN_EXPR:
+      if (!TREE_OPERAND (exp, 0))
+       op = OPCODE_return;
+      else
+       {
+         exp = TREE_OPERAND (exp, 0);
+         if (TREE_CODE (exp) != MODIFY_EXPR)
+           abort ();
+         exp = TREE_OPERAND (exp, 1);
+         op = OPCODE_ireturn + adjust_typed_op (TREE_TYPE (exp));
+         generate_bytecode_insns (method, exp, STACK_TARGET);
+       }
+      RESERVE (1);
+      OP1 (op);
+      break;
+    case MODIFY_EXPR:
+      {
+       tree lhs = TREE_OPERAND (exp, 0);
+       tree rhs = TREE_OPERAND (exp, 1);
+       HOST_WIDE_INT value;
+#if 0
+       if (TREE_CODE (rhs) == PLUS_EXPR
+           && TREE_CODE (lhs) == VAR_DECL
+           /* && FIXME lhs is a local variable */
+           && TYPE_MODE (TREE)TYPE (lhs) == SImode /* ??? */
+           && TREE_OPERAND (rhs, 0) == lhs
+           && TREE_CODE (TREE_OPERAND (rhs, 1)) == INTEGER_CST
+           /* or vice versa FIXME */
+           && (value = TREE_INT_CST_LOW (TREE_OPERAND (rhs, 1)),
+               (value >= -32768 && value <= 32767)))
+         {
+           emit_insn (gen_rtx (SET, SImode,
+                               DECL_RTL (lhs),
+                               gen_rtx (PLUS, SImode,
+                                        DECL_RTL (lhs),
+                                        gen_rtx_CONST_INT (SImode, value))));
+           return DECL_RTL (lhs);
+         }
+#endif
+       if (TREE_CODE (lhs) == COMPONENT_REF)
+         generate_bytecode_insns (method, TREE_OPERAND (lhs, 0), STACK_TARGET);
+       else if (TREE_CODE (lhs) == ARRAY_REF)
+         {
+           generate_bytecode_insns (method,
+                                    TREE_OPERAND (lhs, 0), STACK_TARGET);
+           generate_bytecode_insns (method,
+                                    TREE_OPERAND (lhs, 1), STACK_TARGET);
+         }
+       generate_bytecode_insns (method, rhs, STACK_TARGET);
+       if (target != IGNORE_TARGET)
+         {
+           RESERVE (1);
+           OP1 (TYPE_IS_WIDE (type) ? OPCODE_dup2_x1 : OPCODE_dup_x1);
+         }
+       if (TREE_CODE (lhs) == COMPONENT_REF)
+         {
+           tree field = TREE_OPERAND (lhs, 1);
+           field_op (field,
+                     FIELD_STATIC (field) ? OPCODE_putstatic
+                     : OPCODE_putfield);
+         }
+       else if (TREE_CODE (lhs) == VAR_DECL
+                || TREE_CODE (lhs) == PARM_DECL)
+         {
+           if (FIELD_STATIC (lhs))
+             {
+               field_op (lhs, OPCODE_putstatic);
+             }
+           else
+             {
+               int index = DECL_LOCAL_INDEX (lhs);
+               int opcode = adjust_typed_op (TREE_TYPE (lhs));
+               if (index <= 3)
+                 {
+                   RESERVE (1);
+                   opcode = 59 + 4 * opcode + index;
+                   OP1 (opcode);  /* [ilfda]store_[0123] */
+                 }
+               else
+                 {
+                   maybe_wide (54 + opcode, index);  /* [ilfda]store */
+                 }
+             }
+         }
+       else if (TREE_CODE (lhs) == ARRAY_REF)
+         {
+           jopcode = OPCODE_iastore + adjust_typed_op (TREE_TYPE (lhs));
+           RESERVE(1);
+           OP1 (jopcode);
+         }
+       else
+         fatal ("internal error (bad lhs to MODIFY_EXPR)");
+      }
+      break;
+    case PLUS_EXPR:
+      jopcode = OPCODE_iadd + adjust_typed_op (type);
+      goto binop;
+    case MINUS_EXPR:
+      jopcode = OPCODE_isub + adjust_typed_op (type);
+      goto binop;
+    case MULT_EXPR:
+      jopcode = OPCODE_imul + adjust_typed_op (type);
+      goto binop;
+    case TRUNC_DIV_EXPR:
+    case RDIV_EXPR:
+      jopcode = OPCODE_idiv + adjust_typed_op (type);
+      goto binop;
+    binop:
+      generate_bytecode_insns (method, TREE_OPERAND (exp, 0), target);
+      generate_bytecode_insns (method, TREE_OPERAND (exp, 1), target);
+      if (target == STACK_TARGET)
+       {
+         RESERVE(1);
+         OP1 (jopcode);
+       }
+      break;
+    case CALL_EXPR:
+      {
+       tree t;
+       for (t = TREE_OPERAND (exp, 1);  t != NULL_TREE;  t = TREE_CHAIN (t))
+         {
+           generate_bytecode_insns (method, TREE_VALUE (t), STACK_TARGET);
+         }
+       t = TREE_OPERAND (exp, 0);
+       if (TREE_CODE (t) == FUNCTION_DECL)
+         {
+           int index = find_methodref_index (code_cpool, t);
+           RESERVE (3);
+           if (DECL_CONSTRUCTOR_P (t))
+             OP1 (OPCODE_invokespecial);
+           else if (METHOD_STATIC (t))
+             OP1 (OPCODE_invokestatic);
+           else
+             OP1 (OPCODE_invokevirtual);
+           OP2 (index);
+           break;
+         }
+      }
+      /* fall through */
+    default:
+      error("internal error - tree code not implemented: ", TREE_CODE (exp));
+    }
+}
+
+/* Generate and return a list of chunks containing the class CLAS
+   in the .class file representation.  The list can be written to a
+   .class file using write_chunks.  Allocate chunks from obstack WORK. */
+
+/* Currently does not write any attributes i.e. no code. */
+
+struct chunk *
+generate_classfile (clas, work)
+     tree clas;
+     struct obstack *work;
+{
+  CPool cpool;
+  struct chunk head;
+  struct chunk *chunk;
+  struct chunk *cpool_chunk;
+  char *ptr;
+  int i;
+  char *fields_count_ptr;
+  int fields_count = 0;
+  char *methods_count_ptr;
+  int methods_count = 0;
+  tree part;
+  int total_supers
+    = clas == object_type_node ? 0
+    : TREE_VEC_LENGTH (TYPE_BINFO_BASETYPES (clas));
+
+  chunk = alloc_chunk (&head, NULL, 8, work);
+  ptr = chunk->data;
+  PUT4 (0xCafeBabe);  /* Magic number */
+  PUT2 (3);  /* Minor version */
+  PUT2 (45);  /* Major version */
+  
+  CPOOL_INIT(&cpool);
+  cpool_chunk = chunk = alloc_chunk (chunk, NULL, 0, work);
+
+  /* Next allocate the chunk containing acces_flags through fields_counr. */
+  if (clas == object_type_node)
+    i = 10;
+  else
+    i = 8 + 2 * total_supers;
+  chunk = alloc_chunk (chunk, NULL, i, work);
+  ptr = chunk->data;
+  i = get_access_flags (TYPE_NAME (clas));  PUT2 (i); /* acces_flags */
+  i = find_class_constant (&cpool, clas);  PUT2 (i);  /* this_class */
+  if (clas == object_type_node)
+    {
+      PUT2(0);  /* super_class */
+      PUT2(0);  /* interfaces_count */
+    }
+  else
+    {
+      tree basetypes = TYPE_BINFO_BASETYPES (clas);
+      tree base = BINFO_TYPE (TREE_VEC_ELT (basetypes, 0));
+      int j = find_class_constant (&cpool, base);  PUT2 (j);  /* super_class */
+      PUT2 (total_supers - 1);  /* interfaces_count */
+      for (i = 1;  i < total_supers;  i++)
+       {
+         base = BINFO_TYPE (TREE_VEC_ELT (basetypes, i));
+         j = find_class_constant (&cpool, base);
+         PUT2 (j);
+       }
+    }
+  fields_count_ptr = ptr;
+
+  for (part = TYPE_FIELDS (clas);  part;  part = TREE_CHAIN (part))
+    {
+      if (DECL_NAME (part) == NULL_TREE)
+       continue;
+      chunk = alloc_chunk (chunk, NULL, 8, work);
+      ptr = chunk->data;
+      i = get_access_flags (part);  PUT2 (i);
+      i = find_utf8_constant (&cpool, DECL_NAME (part));  PUT2 (i);
+      i = find_utf8_constant (&cpool, build_java_signature (TREE_TYPE (part)));
+      PUT2(i);
+      PUT2 (0);  /* attributes_count */
+      /* FIXME - emit ConstantValue attribute when appropriate */
+      fields_count++;
+    }
+  ptr = fields_count_ptr;  PUT2 (fields_count);
+
+  chunk = alloc_chunk (chunk, NULL, 2, work);
+  ptr = methods_count_ptr = chunk->data;
+  PUT2 (0);
+
+  for (part = TYPE_METHODS (clas);  part;  part = TREE_CHAIN (part))
+    {
+      tree body = BLOCK_EXPR_BODY (DECL_FUNCTION_BODY (part));
+      int linenumber_size;  /* 4 * number of line number entries */
+      chunk = alloc_chunk (chunk, NULL, 8, work);
+      ptr = chunk->data;
+      i = get_access_flags (part);  PUT2 (i);
+      i = find_utf8_constant (&cpool, DECL_NAME (part));  PUT2 (i);
+      i = find_utf8_constant (&cpool, build_java_signature (TREE_TYPE (part)));
+      PUT2 (i);
+      PUT2 (body != NULL_TREE ? 1 : 0);   /* attributes_count */
+      if (body != NULL_TREE)
+       {
+         int code_attributes_count = 0;
+         int linenumber_size;  /* 4 * number of line number entries */
+         int localvartable_size;  /* 10 * number of local variable entries */
+         static tree Code_node = NULL_TREE;
+         tree t;
+         char *attr_len_ptr;
+         int code_length;
+         if (Code_node == NULL_TREE)
+           Code_node = get_identifier ("Code");
+         chunk = alloc_chunk (chunk, NULL, 14, work);
+         ptr = chunk->data;
+         i = find_utf8_constant (&cpool, Code_node);  PUT2 (i);
+         attr_len_ptr = ptr;
+         BUFFER_RESET (&bytecode);
+         BUFFER_RESET (&localvartable);
+         BUFFER_RESET (&linenumbers);
+         BUFFER_RESET (&localvars);
+         code_SP = 0;
+         code_SP_max = 0;
+         code_cpool = &cpool;
+         for (t = DECL_ARGUMENTS (part);  t != NULL_TREE;  t = TREE_CHAIN (t))
+           localvar_alloc (t, 0);
+         generate_bytecode_insns (part, body, IGNORE_TARGET);
+         code_length = PC;     
+         for (t = DECL_ARGUMENTS (part);  t != NULL_TREE;  t = TREE_CHAIN (t))
+           localvar_free (t, code_length);
+         linenumber_size = BUFFER_LENGTH (&linenumbers);
+         localvartable_size = BUFFER_LENGTH (&localvartable);
+         chunk = alloc_chunk (chunk, NULL, code_length, work);
+         bcopy (bytecode.data, chunk->data, code_length);
+         ptr = attr_len_ptr;
+         i = 8 + code_length + 4;
+         if (linenumber_size > 0)
+           {
+             code_attributes_count++;
+             i += 8 + linenumber_size;
+           }
+         if (localvartable_size > 0)
+           {
+             code_attributes_count++;
+             i += 8 + localvartable_size;
+           }
+         PUT4 (i); /* attribute_length */
+         PUT2 (code_SP_max);  /* max_stack */
+         PUT2 (localvar_max);  /* max_locals */
+         PUT4 (code_length);
+         chunk = alloc_chunk (chunk, NULL, 4, work);
+         ptr = chunk->data;
+         PUT2 (0);  /* exception_table_length */
+         PUT2 (code_attributes_count);
+
+         /* Write the LineNumberTable attribute. */
+         if (linenumber_size > 0)
+           {
+             static tree LineNumberTable_node = NULL_TREE;
+             chunk = alloc_chunk (chunk, NULL, 8 + linenumber_size, work);
+             ptr = chunk->data;
+             if (LineNumberTable_node == NULL_TREE)
+               LineNumberTable_node = get_identifier ("LineNumberTable");
+             i = find_utf8_constant (&cpool, LineNumberTable_node);
+             PUT2 (i);  /* attribute_name_index */
+             i = 2 + linenumber_size;  PUT4 (i);  /* attribute_length */
+             i = linenumber_size >> 2;  PUT2 (i);
+             PUTN (linenumbers.data, linenumber_size);
+           }
+
+         /* Write the LocalVariableTable attribute. */
+         if (localvartable_size > 0)
+           {
+             static tree LocalVariableTable_node = NULL_TREE;
+             chunk = alloc_chunk (chunk, NULL, 8 + localvartable_size, work);
+             ptr = chunk->data;
+             if (LocalVariableTable_node == NULL_TREE)
+               LocalVariableTable_node = get_identifier("LocalVariableTable");
+             i = find_utf8_constant (&cpool, LocalVariableTable_node);
+             PUT2 (i);  /* attribute_name_index */
+             i = 2 + localvartable_size;  PUT4 (i);  /* attribute_length */
+             i = localvartable_size / 10;  PUT2 (i);
+             PUTN (localvartable.data, localvartable_size);
+           }
+       }
+      methods_count++;
+    }
+  ptr = methods_count_ptr;  PUT2 (methods_count);
+
+  chunk = alloc_chunk (chunk, NULL, 2, work);
+  ptr = chunk->data;
+  PUT2 (0);  /* attributes_count */
+
+  /* New finally generate the contents of the constant pool chunk. */
+  i = count_constant_pool_bytes (&cpool);
+  ptr = obstack_alloc (work, i);
+  cpool_chunk->data = ptr;
+  cpool_chunk->size = i;
+  write_constant_pool (&cpool, ptr, i);
+  CPOOL_FINISH (&cpool);
+  return head.next;
+}
+
+char*
+make_class_file_name (clas)
+     tree clas;
+{
+  /* Should prepend an output directly, but need an option to specify it. */
+  return IDENTIFIER_POINTER (identifier_subst (DECL_NAME (TYPE_NAME (clas)),
+                                              "", '.', '/', ".class"));
+}
+
+/* Write out the contens of a class (RECORD_TYPE) CLAS, as a .class file.
+   The output .class file name is make_class_file_name(CLAS). */
+
+void
+write_classfile (clas)
+     tree clas;
+{
+  struct obstack *work = &temporary_obstack;
+  char *class_file_name = make_class_file_name (clas);
+  struct chunk *chunks;
+  FILE* stream = fopen (class_file_name, "wb");
+  if (stream == NULL)
+    fatal ("failed to open `%s' for writing", class_file_name);
+  chunks = generate_classfile (clas, work);
+  write_chunks (stream, chunks);
+  if (fclose (stream))
+    fatal ("failed to close after writing `%s'", class_file_name);
+  obstack_free (work, chunks);
+}
diff --git a/gcc/java/jcf.h b/gcc/java/jcf.h
new file mode 100644 (file)
index 0000000..5e82387
--- /dev/null
@@ -0,0 +1,260 @@
+/* Utility macros to read Java(TM) .class files and byte codes.
+
+   Copyright (C) 1996 Free Software Foundation, Inc.
+
+This program is free software; you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation; either version 2, or (at your option)
+any later version.
+
+This program is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with GNU CC; see the file COPYING.  If not, write to
+the Free Software Foundation, 59 Temple Place - Suite 330,
+Boston, MA 02111-1307, USA.  
+
+Java and all Java-based marks are trademarks or registered trademarks
+of Sun Microsystems, Inc. in the United States and other countries.
+The Free Software Foundation is independent of Sun Microsystems, Inc.  */
+
+/* Written by Per Bothner <bothner@cygnus.com>, February 1996. */
+
+#ifndef JCF_H
+#define JCF_H
+#include "javaop.h"
+#ifndef DEFUN
+#if defined (__STDC__)
+#define AND             ,
+#define PTR             void *
+#define DEFUN(name, arglist, args)      name(args)
+#else
+#define PTR             char *
+#define AND             ;
+#define DEFUN(name, arglist, args)      name arglist args;
+#define inline static
+#endif
+#endif /* !DEFUN */
+
+#ifndef PROTO
+#if defined (__STDC__)
+#define PROTO(paramlist)    paramlist
+#else
+#define PROTO(paramlist)    ()
+#endif
+#endif
+
+#ifndef JCF_u4
+#define JCF_u4 unsigned long
+#endif
+#ifndef JCF_u2
+#define JCF_u2 unsigned short
+#endif
+
+#define ALLOC (void*)malloc
+#define REALLOC (void*)realloc
+#ifndef FREE
+#define FREE(PTR) free(PTR)
+#endif
+
+#ifdef JCF_word
+#define JCF_word JCF_u4
+#endif
+
+#define JCF_ZIP    1
+#define JCF_CLASS  2
+#define JCF_SOURCE 3
+
+struct JCF;
+typedef int (*jcf_filbuf_t) PROTO ((struct JCF*, int needed));
+
+typedef struct CPool {
+  /* Available number of elements in the constants array, before it
+     must be re-allocated. */
+  int capacity;
+
+  /* The constant_pool_count. */
+  int          count;
+
+  uint8*       tags;
+
+  jword*       data;
+} CPool;
+
+/* JCF encapsulates the state of reading a Java Class File. */
+
+typedef struct JCF {
+  unsigned char *buffer;
+  unsigned char *buffer_end;
+  unsigned char *read_ptr;
+  unsigned char *read_end;
+  int seen_in_zip;
+  int java_source;
+  int  outofsynch;             /* Found a class file out of synch
+                                  with the matching source file. */
+  long zip_offset;    
+  jcf_filbuf_t filbuf;
+  void *read_state;
+  char *filename;
+  char *classname;
+  void *zipd;                  /* Directory entry where it was found */
+  JCF_u2 access_flags, this_class, super_class;
+  CPool cpool;
+} JCF;
+/*typedef JCF*  JCF_FILE;*/
+
+/* The CPOOL macros take a (pointer to a) CPool.
+   The JPOOL macros take a (pointer to a) JCF.
+   Some of the latter should perhaps be deprecated or removed. */
+
+#define CPOOL_COUNT(CPOOL) ((CPOOL)->count)
+#define JPOOL_SIZE(JCF) CPOOL_COUNT(&(JCF)->cpool)
+#define JPOOL_TAG(JCF, INDEX) ((JCF)->cpool.tags[INDEX])
+/* The INDEX'th constant pool entry as a JCF_u4. */
+#define CPOOL_UINT(CPOOL, INDEX) ((CPOOL)->data[INDEX])
+#define JPOOL_UINT(JCF, INDEX) CPOOL_UINT(&(JCF)->cpool, INDEX) /*deprecated*/
+/* The first uint16 of the INDEX'th constant pool entry. */
+#define CPOOL_USHORT1(CPOOL, INDEX) ((CPOOL)->data[INDEX] & 0xFFFF)
+#define JPOOL_USHORT1(JCF, INDEX) CPOOL_USHORT1(&(JCF)->cpool, INDEX)
+/* The second uint16 of the INDEX'th constant pool entry. */
+#define CPOOL_USHORT2(CPOOL, INDEX) ((CPOOL)->data[INDEX] >> 16)
+#define JPOOL_USHORT2(JCF, INDEX) CPOOL_USHORT2(&(JCF)->cpool, INDEX)
+#define JPOOL_LONG(JCF, INDEX) \
+  WORDS_TO_LONG (JPOOL_UINT(JCF, INDEX), JPOOL_UINT(JCF, (INDEX)+1))
+#define JPOOL_DOUBLE(JCF, INDEX) \
+  WORDS_TO_DOUBLE  (JPOOL_UINT(JCF, INDEX), JPOOL_UINT(JCF, (INDEX)+1))
+#ifndef JPOOL_UTF_LENGTH
+#define JPOOL_UTF_LENGTH(JCF, INDEX) \
+  GET_u2 ((JCF)->buffer+JPOOL_UINT(JCF, INDEX))
+#endif
+#ifndef JPOOL_UTF_DATA
+#define JPOOL_UTF_DATA(JCF, INDEX) \
+  ((JCF)->buffer+JPOOL_UINT(JCF, INDEX)+2)
+#endif
+#define JPOOL_INT(JCF, INDEX) ((jint) JPOOL_UINT (JCF, INDEX))
+#define JPOOL_FLOAT(JCF, INDEX) WORD_TO_FLOAT (JPOOL_UINT (JCF, INDEX))
+
+#define CPOOL_INDEX_IN_RANGE(CPOOL, INDEX) \
+ ((INDEX) > 0 && (INDEX) < CPOOL_COUNT(CPOOL))
+
+#define CPOOL_FINISH(CPOOL) { \
+  if ((CPOOL)->tags) FREE ((CPOOL)->tags); \
+  if ((CPOOL)->data) FREE ((CPOOL)->data); }
+
+#define JCF_FINISH(JCF) { \
+  CPOOL_FINISH(&(JCF)->cpool); \
+  if ((JCF)->buffer) FREE ((JCF)->buffer); \
+  if ((JCF)->filename) FREE ((JCF)->filename); \
+  if ((JCF)->classname) FREE ((JCF)->classname); }
+  
+#define CPOOL_INIT(CPOOL) \
+  ((CPOOL)->capacity = 0, (CPOOL)->count = 0, (CPOOL)->tags = 0, (CPOOL)->data = 0)
+
+#define CPOOL_REINIT(CPOOL) ((CPOOL)->count = 0)
+
+#define JCF_ZERO(JCF)  \
+  ((JCF)->buffer = (JCF)->buffer_end = (JCF)->read_ptr = (JCF)->read_end = 0,\
+   (JCF)->read_state = 0, (JCF)->filename = (JCF)->classname = 0, \
+   CPOOL_INIT(&(JCF)->cpool), (JCF)->java_source = 0)
+
+/* Given that PTR points to a 2-byte unsigned integer in network
+   (big-endian) byte-order, return that integer. */
+#define GET_u2(PTR) (((PTR)[0] << 8) | ((PTR)[1]))
+/* Like GET_u2, but for little-endian format. */
+#define GET_u2_le(PTR) (((PTR)[1] << 8) | ((PTR)[0]))
+
+/* Given that PTR points to a 4-byte unsigned integer in network
+   (big-endian) byte-order, return that integer. */
+#define GET_u4(PTR) (((JCF_u4)(PTR)[0] << 24) | ((JCF_u4)(PTR)[1] << 16) \
+  | ((JCF_u4)(PTR)[2] << 8) | ((JCF_u4)(PTR)[3]))
+/* Like GET_u4, but for little-endian order. */
+#define GET_u4_le(PTR) (((JCF_u4)(PTR)[3] << 24) | ((JCF_u4)(PTR)[2] << 16) \
+  | ((JCF_u4)(PTR)[1] << 8) | ((JCF_u4)(PTR)[0]))
+
+/* Make sure there are COUNT bytes readable. */
+#define JCF_FILL(JCF, COUNT) \
+  ((JCF)->read_end-(JCF)->read_ptr >= (COUNT) ? 0 : (*(JCF)->filbuf)(JCF, COUNT))
+#define JCF_GETC(JCF) (JCF_FILL(JCF, 1) ? -1 : *(JCF)->read_ptr++)
+#define JCF_READ(JCF, BUFFER, N) \
+    (memcpy (BUFFER, (JCF)->read_ptr, N), (JCF)->read_ptr += (N))
+#define JCF_SKIP(JCF,N) ((JCF)->read_ptr += (N))
+#define JCF_readu(JCF) (*(JCF)->read_ptr++)
+
+/* Reads an unsigned 2-byte integer in network (big-endian) byte-order
+   from JCF.  Returns that integer.
+   Does not check for EOF (make sure to call JCF_FILL before-hand). */
+#define JCF_readu2(JCF) ((JCF)->read_ptr += 2, GET_u2 ((JCF)->read_ptr-2))
+#define JCF_readu2_le(JCF) ((JCF)->read_ptr += 2, GET_u2_le((JCF)->read_ptr-2))
+
+/* Like JCF_readu2, but read a 4-byte unsigned integer. */
+#define JCF_readu4(JCF) ((JCF)->read_ptr += 4, GET_u4 ((JCF)->read_ptr-4))
+#define JCF_readu4_le(JCF) ((JCF)->read_ptr += 4, GET_u4_le((JCF)->read_ptr-4))
+
+#define JCF_TELL(JCF) ((JCF)->read_ptr - (JCF)->buffer)
+#define JCF_SEEK(JCF, POS) ((JCF)->read_ptr = (JCF)->buffer + (POS))
+
+#define ACC_PUBLIC 0x0001
+#define ACC_PRIVATE 0x0002
+#define ACC_PROTECTED 0x0004
+#define ACC_STATIC 0x0008
+#define ACC_FINAL 0x0010
+#define ACC_SYNCHRONIZED 0x0020
+#define ACC_SUPER 0x0020
+#define ACC_VOLATILE 0x0040
+#define ACC_TRANSIENT 0x0080
+#define ACC_NATIVE 0x0100
+#define ACC_INTERFACE 0x0200
+#define ACC_ABSTRACT 0x0400
+
+#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
+
+extern char *classpath;
+#define DEFAULT_CLASS_PATH "."
+
+extern char *find_class PROTO ((const char *, int, JCF*, int));
+extern char *find_classfile PROTO ((char *, JCF*));
+extern int jcf_filbuf_from_stdio PROTO ((JCF *jcf, int count));
+extern void jcf_out_of_synch PROTO((JCF *));
+extern int jcf_unexpected_eof PROTO ((JCF*, int));
+
+/* Extract a character from a Java-style Utf8 string.
+ * PTR points to the current character.
+ * LIMIT points to the end of the Utf8 string.
+ * PTR is incremented to point after the character thta gets returns.
+ * On an error, -1 is returned. */
+#define UTF8_GET(PTR, LIMIT) \
+  ((PTR) >= (LIMIT) ? -1 \
+   : *(PTR) < 128 ? *(PTR)++ \
+   : (*(PTR)&0xE0) == 0xC0 && ((PTR)+=2)<=(LIMIT) && ((PTR)[-1]&0xC0) == 0x80 \
+   ? (((PTR)[-2] & 0x1F) << 6) + ((PTR)[-1] & 0x3F) \
+   : (*(PTR) & 0xF0) == 0xE0 && ((PTR) += 3) <= (LIMIT) \
+   && ((PTR)[-2] & 0xC0) == 0x80 && ((PTR)[-1] & 0xC0) == 0x80 \
+   ? (((PTR)[-3]&0x1F) << 12) + (((PTR)[-2]&0x3F) << 6) + ((PTR)[-1]&0x3F) \
+   : ((PTR)++, -1))
+
+/* Debug macros, for the front end */
+
+extern int quiet_flag;
+#ifdef SOURCE_FRONTEND_DEBUG
+#undef SOURCE_FRONTEND_DEBUG
+#define SOURCE_FRONTEND_DEBUG(X)                               \
+  {if (!quiet_flag) {printf ("* "); printf X; putchar ('\n');} }
+#else
+#define SOURCE_FRONTEND_DEBUG(X)
+#endif
+
+#endif
diff --git a/gcc/java/jv-scan.c b/gcc/java/jv-scan.c
new file mode 100644 (file)
index 0000000..9906592
--- /dev/null
@@ -0,0 +1,235 @@
+/* Main for jv-scan
+   Copyright (C) 1998 Free Software Foundation, Inc.
+   Contributed by Alexandre Petit-Bianco (apbianco@cygnus.com)
+
+This file is part of GNU CC.
+
+GNU CC is free software; you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation; either version 2, or (at your option)
+any later version.
+
+GNU CC is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with GNU CC; see the file COPYING.  If not, write to
+the Free Software Foundation, 59 Temple Place - Suite 330,
+Boston, MA 02111-1307, USA.  */
+
+#include <stdio.h>
+#include <stdlib.h>
+#ifdef __STDC__
+#include <stdarg.h>
+#else
+#include <varargs.h>
+#endif
+
+#include "gansidecl.h"         /* Definitions of PROTO and VPROTO */
+#include "obstack.h"           /* We use obstacks in lex.c */
+
+void fatal VPROTO((char *s, ...));
+void warning VPROTO((char *s, ...));
+void gcc_obstack_init PROTO ((struct obstack *obstack));
+extern void reset_report PROTO ((void));
+
+#define JC1_LITE
+#include "parse.h"
+
+/* Current input file and output file IO streams.  */
+FILE *finput, *out;
+
+/* Current input filename.  */
+char *input_filename;
+
+/* Executable name.  */
+char *exec_name;
+
+/* Flags matching command line options.  */
+int flag_find_main = 0;
+int flag_dump_class = 0;
+int flag_list_filename = 0;
+
+/* jc1-lite main entry point */
+int
+main (argc, argv, envp)
+     int argc;
+     char **argv;
+     char **envp;
+{
+  int i = 1;
+  char *output_file = NULL;
+  long ft;
+
+  exec_name = argv[0];
+
+  /* Default for output */
+  out = stdout;
+
+  /* Process options first */
+  while (argv [i])
+    {
+      if (argv [i][0] == '-')
+       {
+         /* Dump result into a file */
+         if (!strcmp (argv [i], "-o") && i+1 < argc)
+           {
+             argv [i] = NULL;
+             output_file = argv [++i];
+             argv [i] = NULL;
+           }
+
+         /* Print the name of the class that contains main */
+         else if (!strcmp (argv [i], "--print-main"))
+           flag_find_main = 1;
+
+         else if (!strcmp (argv [i], "--list-filename"))
+           flag_list_filename = 1;
+
+         /* List all the classes found in a source file */
+         else if (!strcmp (argv [i], "--list-class"))
+           flag_dump_class = 1;
+
+         else
+           warning ("Unrecognized argument `%s'", argv[i]);
+
+         /* non recognized argument ignored silently */ 
+         argv [i] = NULL;      /* Nullify so it's not considered a file */
+       }
+      i++;
+    }
+
+  /* No flags? Do nothing */
+  if (!flag_find_main && !flag_dump_class)
+    exit (0);
+
+  /* Check on bad usage */
+  if (flag_find_main && flag_dump_class)
+    fatal ("Options `--print-main' and `--list-class' can't be turned on "
+          "at the same time", argv [0]);
+
+  if (output_file && !(out = fopen (output_file, "w")))
+    fatal ("Can't open output file `%s'", argv [0], output_file);
+
+  ft = ftell (out);
+
+  gcc_obstack_init (&temporary_obstack);
+  java_push_parser_context ();
+
+  for ( i = 1; i < argc; i++ )
+    if (argv [i])
+      {
+       input_filename = argv [i];
+       if ( (finput = fopen (argv [i], "r")) )
+         {
+           java_init_lex ();
+           yyparse ();
+           if (ftell (out) != ft)
+             fputc ('\n', out);
+           ft = ftell (out);
+           fclose (finput);
+           reset_report ();
+         }
+       else
+         fatal ("File not found `%s'", argv [0], argv [i]);
+      }
+
+  /* Flush and close */
+  if (ftell (out) != ft)
+    fputc ('\n', out);
+  if (!output_file)
+    fclose (out);
+
+  exit (0);
+}
+
+/* Error report, memory, obstack initialization and other utility
+   functions */
+
+void
+fatal VPROTO((char *s, ...))
+{
+#ifndef __STDC__
+  char *s;
+#endif
+  va_list ap;
+
+  VA_START (ap, s);
+
+#ifndef __STDC__
+  s = va_arg (ap, char *);
+#endif
+
+  fprintf (stderr, "%s: error: ", exec_name);
+  vfprintf (stderr, s, ap);
+  fputc ('\n', stderr);
+  va_end (ap);
+  exit (1);
+}
+
+char *
+xmalloc (size)
+     unsigned size;
+{
+  register char *value;
+
+  if (size == 0)
+    size = 1;
+
+  value = (char *) malloc (size);
+  if (value == 0)
+    fatal ("virtual memory exhausted");
+  return value;
+}
+
+char *
+xstrdup (string)
+     char *string;
+{
+  int length = strlen (string)+1;
+  char *to_return = xmalloc (length);
+  strcpy (to_return, string);
+  return to_return;
+}
+
+void
+warning VPROTO((char *s, ...))
+{
+#ifndef __STDC__
+  char *s;
+#endif
+  va_list ap;
+
+  VA_START (ap, s);
+
+#ifndef __STDC__
+  s = va_arg (ap, char *);
+#endif
+
+  fprintf (stderr, "%s: warning: ", exec_name);
+  vfprintf (stderr, s, ap);
+  fputc ('\n', stderr);
+  va_end (ap);
+}
+
+void
+gcc_obstack_init (obstack)
+     struct obstack *obstack;
+{
+  /* Let particular systems override the size of a chunk.  */
+#ifndef OBSTACK_CHUNK_SIZE
+#define OBSTACK_CHUNK_SIZE 0
+#endif
+  /* Let them override the alloc and free routines too.  */
+#ifndef OBSTACK_CHUNK_ALLOC
+#define OBSTACK_CHUNK_ALLOC xmalloc
+#endif
+#ifndef OBSTACK_CHUNK_FREE
+#define OBSTACK_CHUNK_FREE free
+#endif
+  _obstack_begin (obstack, OBSTACK_CHUNK_SIZE, 0,
+                 (void *(*) ()) OBSTACK_CHUNK_ALLOC,
+                 (void (*) ()) OBSTACK_CHUNK_FREE);
+}
diff --git a/gcc/java/jvgenmain.c b/gcc/java/jvgenmain.c
new file mode 100644 (file)
index 0000000..ce7c60a
--- /dev/null
@@ -0,0 +1,124 @@
+/* Program to generate "main" a Java(TM) class containing a main method.
+   Copyright (C) 1998 Free Software Foundation, Inc.
+
+This file is part of GNU CC.
+
+GNU CC is free software; you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation; either version 2, or (at your option)
+any later version.
+
+GNU CC is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with GNU CC; see the file COPYING.  If not, write to
+the Free Software Foundation, 59 Temple Place - Suite 330,
+Boston, MA 02111-1307, USA. 
+
+Java and all Java-based marks are trademarks or registered trademarks
+of Sun Microsystems, Inc. in the United States and other countries.
+The Free Software Foundation is independent of Sun Microsystems, Inc.  */
+
+/* Written by Per Bothner <bothner@cygnus.com> */
+
+#include <stdio.h>
+#include <stdlib.h>
+#include "config.h"
+#include "obstack.h"
+
+const char main_method_prefix[] = "main__";
+const char main_method_suffix[] = "Pt6JArray1ZPQ34java4lang6String";
+const char class_mangling_prefix[] = "_CL_";
+
+struct obstack name_obstack;
+
+void
+error (const char *str)
+{
+  fprintf (stderr, "%s\n", str);
+  exit (-1);
+}
+
+void *
+xmalloc (size)
+     size_t size;
+{
+  void *ptr = malloc (size);
+  if (ptr == NULL)
+    {
+      fprintf (stderr, "Not enough memory!\n");
+      exit (-1);
+    }
+  return ptr;
+}
+
+void
+gcc_obstack_init (obstack)
+     struct obstack *obstack;
+{
+  /* Let particular systems override the size of a chunk.  */
+#ifndef OBSTACK_CHUNK_SIZE
+#define OBSTACK_CHUNK_SIZE 0
+#endif
+  /* Let them override the alloc and free routines too.  */
+#ifndef OBSTACK_CHUNK_ALLOC
+#define OBSTACK_CHUNK_ALLOC xmalloc
+#endif
+#ifndef OBSTACK_CHUNK_FREE
+#define OBSTACK_CHUNK_FREE free
+#endif
+  _obstack_begin (obstack, OBSTACK_CHUNK_SIZE, 0,
+                 (void *(*) ()) OBSTACK_CHUNK_ALLOC,
+                 (void (*) ()) OBSTACK_CHUNK_FREE);
+}
+
+int
+main (int argc, const char **argv)
+{
+  const char *classname;
+  FILE *stream;
+  char *mangled_classname;
+
+  if (argc < 2 || argc > 3)
+    {
+      fprintf (stderr, "Usage: %s CLASSNAME [OUTFILE]\n", argv[0]);
+      exit(-1);
+    }
+
+  classname = argv[1];
+
+  gcc_obstack_init (&name_obstack);
+  append_gpp_mangled_classtype (&name_obstack, classname);
+  mangled_classname = obstack_finish (&name_obstack);
+
+  if (argc > 2 && strcmp (argv[2], "-") != 0)
+    {
+      const char *outfile = argv[2];
+      stream = fopen (outfile, "w");
+      if (stream == NULL)
+       {
+         fprintf (stderr, "%s: Cannot open output file: %s\n",
+                  argv[0], outfile);
+         exit (-1);
+       }
+    }
+  else
+    stream = stdout;
+  fprintf (stream, "extern struct Class %s%s;\n",
+          class_mangling_prefix, mangled_classname);
+  fprintf (stream, "int main (int argc, const char **argv)\n");
+  fprintf (stream, "{\n");
+  fprintf (stream, "   JvRunMain (&%s%s, argc, argv);\n",
+          class_mangling_prefix, mangled_classname);
+  fprintf (stream, "}\n");
+  if (stream != stdout && fclose (stream) != 0)
+    {
+      fprintf (stderr, "%s: Failed to close output file %s\n",
+              argv[0], argv[2]);
+      exit (-1);
+    }
+  return 0;
+}
diff --git a/gcc/java/jvspec.c b/gcc/java/jvspec.c
new file mode 100644 (file)
index 0000000..5f93811
--- /dev/null
@@ -0,0 +1,357 @@
+/* Specific flags and argument handling of the front-end of the 
+   GNU compiler for the Java(TM) language.
+   Copyright (C) 1996, 1997, 1998 Free Software Foundation, Inc.
+
+This file is part of GNU CC.
+
+GNU CC is free software; you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation; either version 2, or (at your option)
+any later version.
+
+GNU CC is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with GNU CC; see the file COPYING.  If not, write to
+the Free Software Foundation, 59 Temple Place - Suite 330,
+Boston, MA 02111-1307, USA. 
+
+Java and all Java-based marks are trademarks or registered trademarks
+of Sun Microsystems, Inc. in the United States and other countries.
+The Free Software Foundation is independent of Sun Microsystems, Inc.  */
+
+#include "config.h"
+
+#include "system.h"
+
+#include "gansidecl.h"
+
+#if defined (WITH_THREAD_posix) || defined (WITH_THREAD_pthreads)
+#define THREAD_NAME "-lpthread"
+#elif defined (WITH_THREAD_qt)
+#define THREAD_NAME "-lqthreads"
+#endif
+
+/* This bit is set if we saw a `-xfoo' language specification.  */
+#define LANGSPEC       (1<<1)
+/* This bit is set if they did `-lm' or `-lmath'.  */
+#define MATHLIB                (1<<2)
+/* This bit is set if they did `-lc'.  */
+#define WITHLIBC       (1<<3)
+/* This bit is set if they did `-lgc'.  */
+#define GCLIB          (1<<4)
+/* This bit is set if they did `-lpthread' (or added some other thread
+   library).  */
+#define THREADLIB      (1<<5)
+
+#ifndef MATH_LIBRARY
+#define MATH_LIBRARY "-lm"
+#endif
+
+extern char *xmalloc PROTO((size_t));
+extern int do_spec             PROTO((char *));
+extern char *input_filename;
+extern size_t input_filename_length;
+
+char *main_class_name = NULL;
+int lang_specific_extra_outfiles = 0;
+
+char jvgenmain_spec[] =
+  "jvgenmain %i %{!pipe:%g.i} |\n\
+   cc1 %{!pipe:%g.i} %1 \
+                  %{!Q:-quiet} -dumpbase %b.c %{d*} %{m*} %{a*}\
+                  %{g*} %{O*} \
+                  %{v:-version} %{pg:-p} %{p} %{f*}\
+                  %{aux-info*}\
+                  %{pg:%{fomit-frame-pointer:%e-pg and -fomit-frame-pointer are incompatible}}\
+                  %{S:%W{o*}%{!o*:-o %b.s}}%{!S:-o %{|!pipe:%g.s}} |\n\
+              %{!S:as %a %Y -o %w%b%O %{!pipe:%g.s} %A\n }";
+
+void
+lang_specific_driver (fn, in_argc, in_argv, in_added_libraries)
+     void (*fn)();
+     int *in_argc;
+     char ***in_argv;
+     int *in_added_libraries;
+{
+  int i, j;
+
+  /* If non-zero, the user gave us the `-v' flag.  */ 
+  int saw_verbose_flag = 0;
+
+  /* This will be 0 if we encounter a situation where we should not
+     link in libjava.  */
+  int library = 1;
+
+  /* The number of arguments being added to what's in argv, other than
+     libraries.  We use this to track the number of times we've inserted
+     -xc++/-xnone.  */
+  int added = 2;
+
+  /* Used to track options that take arguments, so we don't go wrapping
+     those with -xc++/-xnone.  */
+  char *quote = NULL;
+
+  /* The new argument list will be contained in this.  */
+  char **arglist;
+
+  /* Non-zero if we saw a `-xfoo' language specification on the
+     command line.  Used to avoid adding our own -xc++ if the user
+     already gave a language for the file.  */
+  int saw_speclang = 0;
+
+  /* "-lm" or "-lmath" if it appears on the command line.  */
+  char *saw_math = 0;
+
+  /* "-lc" if it appears on the command line.  */
+  char *saw_libc = 0;
+
+  /* "-lgc" if it appears on the command line.  */
+  char *saw_gc = 0;
+
+  /* Saw `-l' option for the thread library.  */
+  char *saw_threadlib = 0;
+
+  /* Saw `-ljava' on command line.  */
+  int saw_libjava = 0;
+
+  /* An array used to flag each argument that needs a bit set for
+     LANGSPEC, MATHLIB, WITHLIBC, or GCLIB.  */
+  int *args;
+
+  /* By default, we throw on the math library.  */
+  int need_math = 1;
+
+  /* By default, we throw in the thread library (if one is required).
+   */
+  int need_thread = 1;
+
+  /* The total number of arguments with the new stuff.  */
+  int argc;
+
+  /* The argument list.  */
+  char **argv;
+
+  /* The number of libraries added in.  */
+  int added_libraries;
+
+  /* The total number of arguments with the new stuff.  */
+  int num_args = 1;
+
+  argc = *in_argc;
+  argv = *in_argv;
+  added_libraries = *in_added_libraries;
+
+  args = (int *) xmalloc (argc * sizeof (int));
+  bzero ((char *) args, argc * sizeof (int));
+
+  for (i = 1; i < argc; i++)
+    {
+      /* If the previous option took an argument, we swallow it here.  */
+      if (quote)
+       {
+         quote = NULL;
+         continue;
+       }
+
+      /* We don't do this anymore, since we don't get them with minus
+        signs on them.  */
+      if (argv[i][0] == '\0' || argv[i][1] == '\0')
+       continue;
+
+      if (argv[i][0] == '-')
+       {
+         if (library != 0 && (strcmp (argv[i], "-nostdlib") == 0
+                              || strcmp (argv[i], "-nodefaultlibs") == 0))
+           {
+             library = 0;
+           }
+         else if (strcmp (argv[i], "-lm") == 0
+                  || strcmp (argv[i], "-lmath") == 0
+#ifdef ALT_LIBM
+                  || strcmp (argv[i], ALT_LIBM) == 0
+#endif
+                 )
+           {
+             args[i] |= MATHLIB;
+             need_math = 0;
+           }
+         else if (strncmp (argv[i], "-fmain=", 7) == 0)
+           main_class_name = argv[i] + 7;
+         else if (strcmp (argv[i], "-ljava") == 0)
+           saw_libjava = 1;
+         else if (strcmp (argv[i], "-lc") == 0)
+           args[i] |= WITHLIBC;
+         else if (strcmp (argv[i], "-lgc") == 0)
+           args[i] |= GCLIB;
+#ifdef THREAD_NAME
+         else if (strcmp (argv[i], THREAD_NAME) == 0)
+           {
+             args[i] |= THREADLIB;
+             need_thread = 0;
+           }
+#endif
+         else if (strcmp (argv[i], "-v") == 0)
+           {
+             saw_verbose_flag = 1;
+             if (argc == 2)
+               {
+                 /* If they only gave us `-v', don't try to link
+                    in libjava.  */ 
+                 library = 0;
+               }
+           }
+         else if (strncmp (argv[i], "-x", 2) == 0)
+           saw_speclang = 1;
+         else if (((argv[i][2] == '\0'
+                    && (char *)strchr ("bBVDUoeTuIYmLiA", argv[i][1]) != NULL)
+                   || strcmp (argv[i], "-Tdata") == 0))
+           quote = argv[i];
+         else if (library != 0 && ((argv[i][2] == '\0'
+                    && (char *) strchr ("cSEM", argv[i][1]) != NULL)
+                   || strcmp (argv[i], "-MM") == 0))
+           {
+             /* Don't specify libraries if we won't link, since that would
+                cause a warning.  */
+             library = 0;
+             added -= 2;
+           }
+         else
+           /* Pass other options through.  */
+           continue;
+       }
+      else
+       {
+         int len; 
+
+         if (saw_speclang)
+           {
+             saw_speclang = 0;
+             continue;
+           }
+
+         /* If the filename ends in .c or .i, put options around it.
+            But not if a specified -x option is currently active.  */
+         len = strlen (argv[i]);
+         if (len > 2
+             && (argv[i][len - 1] == 'c' || argv[i][len - 1] == 'i')
+             && argv[i][len - 2] == '.')
+           {
+             args[i] |= LANGSPEC;
+             added += 2;
+           }
+       }
+    }
+
+  if (quote)
+    (*fn) ("argument to `%s' missing\n", quote);
+
+  /* If we know we don't have to do anything, bail now.  */
+  if (! added && ! library && main_class_name == NULL)
+    {
+      free (args);
+      return;
+    }
+
+  num_args = argc + added + need_math + need_thread;
+  if (main_class_name)
+    {
+      lang_specific_extra_outfiles++;
+    }
+  arglist = (char **) xmalloc (num_args * sizeof (char *));
+
+  /* NOTE: We start at 1 now, not 0.  */
+  for (i = 0, j = 0; i < argc; i++, j++)
+    {
+      arglist[j] = argv[i];
+
+      if (strncmp (argv[i], "-fmain=", 7) == 0)
+       {
+         --j;
+         continue;
+       }
+
+      /* Make sure -ljava is before the math library, since libjava
+        itself uses those math routines.  */
+      if (!saw_math && (args[i] & MATHLIB) && library)
+       {
+         --j;
+         saw_math = argv[i];
+       }
+
+      /* Likewise -ljava must come before -lc.  */
+      if (!saw_libc && (args[i] & WITHLIBC) && library)
+       {
+         --j;
+         saw_libc = argv[i];
+       }
+
+      /* And -ljava must come before -lgc.  */
+      if (!saw_gc && (args[i] & GCLIB) && library)
+       {
+         --j;
+         saw_gc = argv[i];
+       }
+
+      /* And -ljava must come before thread library.  */
+      if (!saw_threadlib && (args[i] & THREADLIB) && library)
+       {
+         --j;
+         saw_threadlib = argv[i];
+       }
+  }
+
+  /* Add `-ljava' if we haven't already done so.  */
+  if (library && ! saw_libjava)
+    {
+      arglist[j++] = "-ljava";
+      added_libraries++;
+    }
+
+  if (saw_math)
+    arglist[j++] = saw_math;
+  else if (library)
+    {
+      arglist[j++] = MATH_LIBRARY;
+      added_libraries++;
+    }
+
+  /* FIXME: we need a way to know when the GC library should be
+     added.  Then we can add it if the user hasn't already.  */
+  if (saw_gc)
+    arglist[j++] = saw_gc;
+
+  /* Thread library must come after GC library as well as after
+     -ljava.  */
+  if (saw_threadlib)
+    arglist[j++] = saw_threadlib;
+#ifdef THREAD_NAME
+  else if (library)
+    {
+      arglist[j++] = THREAD_NAME;
+      added_libraries++;
+    }
+#endif
+
+  if (saw_libc)
+    arglist[j++] = saw_libc;
+
+  arglist[j] = NULL;
+
+  *in_argc = j;
+  *in_argv = arglist;
+  *in_added_libraries = added_libraries;
+}
+
+int
+lang_specific_pre_link ()
+{
+  if (main_class_name == NULL)
+    return 0;
+  input_filename = main_class_name;
+  input_filename_length = strlen (main_class_name);
+  return do_spec (jvgenmain_spec);
+}
diff --git a/gcc/java/keyword.gperf b/gcc/java/keyword.gperf
new file mode 100644 (file)
index 0000000..38f0257
--- /dev/null
@@ -0,0 +1,80 @@
+%{
+/* Keyword definition for the GNU compiler for the Java(TM) language.
+   Copyright (C) 1997, 1998 Free Software Foundation, Inc.
+   Contributed by Alexandre Petit-Bianco (apbianco@cygnus.com)
+
+This file is part of GNU CC.
+
+GNU CC is free software; you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation; either version 2, or (at your option)
+any later version.
+
+GNU CC is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with GNU CC; see the file COPYING.  If not, write to
+the Free Software Foundation, 59 Temple Place - Suite 330,
+Boston, MA 02111-1307, USA.
+
+Java and all Java-based marks are trademarks or registered trademarks
+of Sun Microsystems, Inc. in the United States and other countries.
+The Free Software Foundation is independent of Sun Microsystems, Inc.  */
+
+%}
+struct java_keyword { char *name; int token; };
+%%
+abstract, ABSTRACT_TK
+default, DEFAULT_TK
+if, IF_TK
+private, PRIVATE_TK
+throw, THROW_TK
+boolean, BOOLEAN_TK
+do, DO_TK
+implements, IMPLEMENTS_TK
+protected, PROTECTED_TK
+throws, THROWS_TK
+break, BREAK_TK
+double, DOUBLE_TK
+import, IMPORT_TK
+public, PUBLIC_TK
+transient, TRANSIENT_TK
+byte, BYTE_TK
+else, ELSE_TK
+instanceof, INSTANCEOF_TK
+return, RETURN_TK
+try, TRY_TK
+case, CASE_TK
+extends, EXTENDS_TK
+int, INT_TK
+short, SHORT_TK
+void, VOID_TK
+catch, CATCH_TK
+final, FINAL_TK
+interface, INTERFACE_TK
+static, STATIC_TK
+volatile, VOLATILE_TK
+char, CHAR_TK
+finally, FINALLY_TK
+long, LONG_TK
+super, SUPER_TK
+while, WHILE_TK
+class, CLASS_TK
+float, FLOAT_TK
+native, NATIVE_TK
+switch, SWITCH_TK
+const, CONST_TK
+for, FOR_TK
+new, NEW_TK
+synchronized, SYNCHRONIZED_TK
+continue, CONTINUE_TK
+goto, GOTO_TK
+package, PACKAGE_TK
+this, THIS_TK
+# true, false and null aren't keyword. But we match them easily this way
+true, TRUE_TK
+false, FALSE_TK
+null, NULL_TK
diff --git a/gcc/java/keyword.h b/gcc/java/keyword.h
new file mode 100644 (file)
index 0000000..c9bd171
--- /dev/null
@@ -0,0 +1,165 @@
+/* C code produced by gperf version 2.5 (GNU C++ version) */
+/* Command-line: gperf -p -t -j1 -i 1 -g -o -N java_keyword -k1,3,$ keyword.gperf  */
+/* Keyword definitions for the GNU compiler for the Java(TM) language.
+   Copyright (C) 1997, 1998 Free Software Foundation, Inc.
+   Contributed by Alexandre Petit-Bianco (apbianco@cygnus.com)
+
+This file is part of GNU CC.
+
+GNU CC is free software; you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation; either version 2, or (at your option)
+any later version.
+
+GNU CC is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with GNU CC; see the file COPYING.  If not, write to
+the Free Software Foundation, 59 Temple Place - Suite 330,
+Boston, MA 02111-1307, USA.
+
+Java and all Java-based marks are trademarks or registered trademarks
+of Sun Microsystems, Inc. in the United States and other countries.
+The Free Software Foundation is independent of Sun Microsystems, Inc.  */
+
+struct java_keyword { char *name; int token; };
+
+#define TOTAL_KEYWORDS 50
+#define MIN_WORD_LENGTH 2
+#define MAX_WORD_LENGTH 12
+#define MIN_HASH_VALUE 6
+#define MAX_HASH_VALUE 86
+/* maximum key range = 81, duplicates = 0 */
+
+#ifdef __GNUC__
+inline
+#endif
+static unsigned int
+hash (str, len)
+     register char *str;
+     register int unsigned len;
+{
+  static unsigned char asso_values[] =
+    {
+     87, 87, 87, 87, 87, 87, 87, 87, 87, 87,
+     87, 87, 87, 87, 87, 87, 87, 87, 87, 87,
+     87, 87, 87, 87, 87, 87, 87, 87, 87, 87,
+     87, 87, 87, 87, 87, 87, 87, 87, 87, 87,
+     87, 87, 87, 87, 87, 87, 87, 87, 87, 87,
+     87, 87, 87, 87, 87, 87, 87, 87, 87, 87,
+     87, 87, 87, 87, 87, 87, 87, 87, 87, 87,
+     87, 87, 87, 87, 87, 87, 87, 87, 87, 87,
+     87, 87, 87, 87, 87, 87, 87, 87, 87, 87,
+     87, 87, 87, 87, 87, 87, 87, 18, 37, 38,
+     27,  1, 30,  3, 12,  8, 87,  2, 11, 87,
+      8,  1,  5, 87, 24,  1,  1, 30,  2, 36,
+     87,  1, 87, 87, 87, 87, 87, 87,
+    };
+  register int hval = len;
+
+  switch (hval)
+    {
+      default:
+      case 3:
+        hval += asso_values[str[2]];
+      case 2:
+      case 1:
+        hval += asso_values[str[0]];
+        break;
+    }
+  return hval + asso_values[str[len - 1]];
+}
+
+#ifdef __GNUC__
+inline
+#endif
+struct java_keyword *
+java_keyword (str, len)
+     register char *str;
+     register unsigned int len;
+{
+  static struct java_keyword wordlist[] =
+    {
+      {"",}, {"",}, {"",}, {"",}, {"",}, {"",}, 
+      {"try",  TRY_TK},
+      {"else",  ELSE_TK},
+      {"short",  SHORT_TK},
+      {"goto",  GOTO_TK},
+      {"extends",  EXTENDS_TK},
+      {"",}, {"",}, 
+      {"int",  INT_TK},
+      {"this",  THIS_TK},
+      {"",}, 
+      {"native",  NATIVE_TK},
+      {"",}, {"",}, 
+      {"interface",  INTERFACE_TK},
+      {"import",  IMPORT_TK},
+      {"private",  PRIVATE_TK},
+      {"volatile",  VOLATILE_TK},
+      {"",}, 
+      {"implements",  IMPLEMENTS_TK},
+      {"",}, 
+      {"long",  LONG_TK},
+      {"switch",  SWITCH_TK},
+      {"abstract",  ABSTRACT_TK},
+      {"transient",  TRANSIENT_TK},
+      {"do",  DO_TK},
+      {"",}, 
+      {"throws",  THROWS_TK},
+      {"",}, 
+      {"null",  NULL_TK},
+      {"super",  SUPER_TK},
+      {"true",  TRUE_TK},
+      {"float",  FLOAT_TK},
+      {"",}, 
+      {"return",  RETURN_TK},
+      {"if",  IF_TK},
+      {"void",  VOID_TK},
+      {"protected",  PROTECTED_TK},
+      {"byte",  BYTE_TK},
+      {"case",  CASE_TK},
+      {"break",  BREAK_TK},
+      {"finally",  FINALLY_TK},
+      {"false",  FALSE_TK},
+      {"synchronized",  SYNCHRONIZED_TK},
+      {"instanceof",  INSTANCEOF_TK},
+      {"while",  WHILE_TK},
+      {"package",  PACKAGE_TK},
+      {"const",  CONST_TK},
+      {"boolean",  BOOLEAN_TK},
+      {"final",  FINAL_TK},
+      {"continue",  CONTINUE_TK},
+      {"catch",  CATCH_TK},
+      {"",}, {"",}, {"",}, {"",}, {"",}, 
+      {"class",  CLASS_TK},
+      {"static",  STATIC_TK},
+      {"double",  DOUBLE_TK},
+      {"default",  DEFAULT_TK},
+      {"throw",  THROW_TK},
+      {"",}, {"",}, {"",}, {"",}, {"",}, {"",}, {"",}, {"",}, {"",}, 
+      {"",}, {"",}, {"",}, {"",}, {"",}, 
+      {"for",  FOR_TK},
+      {"",}, 
+      {"new",  NEW_TK},
+      {"char",  CHAR_TK},
+      {"",}, 
+      {"public",  PUBLIC_TK},
+    };
+
+  if (len <= MAX_WORD_LENGTH && len >= MIN_WORD_LENGTH)
+    {
+      register int key = hash (str, len);
+
+      if (key <= MAX_HASH_VALUE && key >= 0)
+        {
+          register char *s = wordlist[key].name;
+
+          if (*s == *str && !strcmp (str + 1, s + 1))
+            return &wordlist[key];
+        }
+    }
+  return 0;
+}
diff --git a/gcc/java/lang-options.h b/gcc/java/lang-options.h
new file mode 100644 (file)
index 0000000..f421e6a
--- /dev/null
@@ -0,0 +1,35 @@
+/* Switch definitions for the GNU compiler for the Java(TM) language.
+   Copyright (C) 1998 Free Software Foundation, Inc.
+
+This file is part of GNU CC.
+
+GNU CC is free software; you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation; either version 2, or (at your option)
+any later version.
+
+GNU CC is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with GNU CC; see the file COPYING.  If not, write to
+the Free Software Foundation, 59 Temple Place - Suite 330,
+Boston, MA 02111-1307, USA.  
+
+Java and all Java-based marks are trademarks or registered trademarks
+of Sun Microsystems, Inc. in the United States and other countries.
+The Free Software Foundation is independent of Sun Microsystems, Inc.  */
+
+/* This is the contribution to the `lang_options' array in gcc.c for
+   java.  */
+
+DEFINE_LANG_NAME ("Java")
+     
+  { "-fbounds-check", "" },
+  { "-fno-bounds-check", "Disable automatic array bounds checking" },
+  { "-fassume-compiled", "Make is_compiled_class return 1"},
+  { "-fno-assume-compiled", "" },
+  { "-femit-class-file", "" },
+  { "-femit-class-files", "Dump class files to <name>.class" },
diff --git a/gcc/java/lang-specs.h b/gcc/java/lang-specs.h
new file mode 100644 (file)
index 0000000..f86c327
--- /dev/null
@@ -0,0 +1,38 @@
+/* Definitions for specs for the GNU compiler for the Java(TM) language.
+   Copyright (C) 1996 Free Software Foundation, Inc.
+
+This file is part of GNU CC.
+
+GNU CC is free software; you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation; either version 2, or (at your option)
+any later version.
+
+GNU CC is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with GNU CC; see the file COPYING.  If not, write to
+the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA.
+
+Java and all Java-based marks are trademarks or registered trademarks
+of Sun Microsystems, Inc. in the United States and other countries.
+The Free Software Foundation is independent of Sun Microsystems, Inc.  */
+
+/* This is the contribution to the `default_compilers' array in gcc.c for
+   Java.  */
+
+  {".java",   "@java" },
+  {".class",  "@java" },
+  {"@java",
+   "%{!M:%{!MM:%{!E:jc1 %i %1 %2 %{!Q:-quiet} %{d*} %{m*} %{a}\
+                           %{g*} %{O*} %{W*} %{w} %{pedantic*} %{ansi}\
+                           %{traditional} %{v:-version} %{pg:-p} %{p}\
+                           %{f*} %{+e*} %{aux-info*}\
+                           %{pg:%{fomit-frame-pointer:%e-pg and -fomit-frame-pointer are incompatible}}\
+                           %{S:%W{o*}%{!o*:-o %b.s}}%{!S:-o %{|!pipe:%g.s}} |\n\
+                   %{!S:as %a %Y\
+                           %{c:%W{o*}%{!o*:-o %w%b%O}}%{!c:-o %d%w%u%O}\
+                           %{!pipe:%g.s} %A\n }}}}"},
diff --git a/gcc/java/lang.c b/gcc/java/lang.c
new file mode 100644 (file)
index 0000000..ddaff0c
--- /dev/null
@@ -0,0 +1,406 @@
+/* Java(TM) language-specific utility routines.
+   Copyright (C) 1996, 1997, 1998 Free Software Foundation, Inc.
+
+This file is part of GNU CC.
+
+GNU CC is free software; you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation; either version 2, or (at your option)
+any later version.
+
+GNU CC is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with GNU CC; see the file COPYING.  If not, write to
+the Free Software Foundation, 59 Temple Place - Suite 330,
+Boston, MA 02111-1307, USA. 
+
+Java and all Java-based marks are trademarks or registered trademarks
+of Sun Microsystems, Inc. in the United States and other countries.
+The Free Software Foundation is independent of Sun Microsystems, Inc.  */
+
+/* Hacked by Per Bothner <bothner@cygnus.com> February 1996. */
+
+#include <stdio.h>
+#include "config.h"
+#include "tree.h"
+#include "input.h"
+#include "java-tree.h"
+#include "jcf.h"
+
+int compiling_from_source;
+
+char *language_string = "GNU Java";
+
+/* Nonzero if we want to automatically do array bounds checking;
+   on by default.  Use -fno-bounds-check to disable.  */
+
+int flag_bounds_check = 1;
+
+/* Nonzero if we should make is_compiled_class always return 1 for
+   appropriate classes that we're referencing.  */
+
+int flag_assume_compiled = 1;
+
+int flag_emit_class_files = 0;
+
+/* From gcc/flags.h, and idicates if exceptions are turned on or not. */
+
+extern int flag_new_exceptions;
+extern int flag_exceptions;
+
+/* Table of language-dependent -f options.
+   STRING is the option name.  VARIABLE is the address of the variable.
+   ON_VALUE is the value to store in VARIABLE
+    if `-fSTRING' is seen as an option.
+   (If `-fno-STRING' is seen as an option, the opposite value is stored.)  */
+
+static struct { char *string; int *variable; int on_value;} lang_f_options[] =
+{
+  {"bounds-check", &flag_bounds_check, 1},
+  {"assume-compiled", &flag_assume_compiled, 1},
+  {"emit-class-file", &flag_emit_class_files, 1},
+  {"emit-class-files", &flag_emit_class_files, 1},
+};
+
+JCF main_jcf[1];
+JCF *current_jcf;
+
+/*
+ * process java-specific compiler command-line options
+ */
+int
+lang_decode_option (argc, argv)
+     int argc;
+     char **argv;
+{
+  char *p = argv[0];
+  if (p[0] == '-' && p[1] == 'f')
+    {
+      /* Some kind of -f option.
+        P's value is the option sans `-f'.
+        Search for it in the table of options.  */
+      int found = 0, j;
+
+      p += 2;
+
+      for (j = 0;
+               !found && j < sizeof (lang_f_options) / sizeof (lang_f_options[0]);
+               j++)
+       {
+         if (!strcmp (p, lang_f_options[j].string))
+           {
+             *lang_f_options[j].variable = lang_f_options[j].on_value;
+             /* A goto here would be cleaner,
+                but breaks the vax pcc.  */
+             found = 1;
+           }
+         if (p[0] == 'n' && p[1] == 'o' && p[2] == '-'
+             && ! strcmp (p+3, lang_f_options[j].string))
+           {
+             *lang_f_options[j].variable = ! lang_f_options[j].on_value;
+             found = 1;
+           }
+       }
+      return found;
+    }
+
+  return 0;
+}
+
+FILE *finput;
+char *
+init_parse (filename)
+     char *filename;
+{
+  /* Open input file.  */
+
+  if (filename == 0 || !strcmp (filename, "-"))
+    {
+      finput = stdin;
+      filename = "stdin";
+    }
+  else
+    finput = fopen (filename, "r");
+  if (finput == 0)
+    pfatal_with_name (filename);
+
+#ifdef IO_BUFFER_SIZE
+  setvbuf (finput, (char *) xmalloc (IO_BUFFER_SIZE), _IOFBF, IO_BUFFER_SIZE);
+#endif
+  init_lex ();
+
+  return filename;
+}
+
+void
+finish_parse ()
+{
+  fclose (finput);
+}
+
+/* Buffer used by lang_printable_name. */
+static char *decl_buf = NULL;
+
+/* Allocated size of decl_buf. */
+static int decl_buflen = 0;
+
+/* Length of used part of decl_buf;  position for next character. */
+static int decl_bufpos = 0;
+
+/* Append the string STR to decl_buf.
+   It length is given by LEN;  -1 means the string is nul-terminated. */
+
+static void
+put_decl_string (str, len)
+     char *str;
+     int len;
+{
+  if (len < 0)
+    len = strlen (str);
+  if (decl_bufpos + len >= decl_buflen)
+    {
+      if (decl_buf == NULL)
+       {
+         decl_buflen = len + 100;
+         decl_buf = (char *) xmalloc (decl_buflen);
+       }
+      else
+       {
+         decl_buflen *= 2;
+         decl_buf = (char *) xrealloc (decl_buf, decl_buflen);
+       }
+    }
+  strcpy (decl_buf + decl_bufpos, str);
+  decl_bufpos += len;
+}
+
+/* Append to decl_buf a printable name for NODE. */
+
+static void
+put_decl_node (node)
+     tree node;
+{
+  int was_pointer = 0;
+  if (TREE_CODE (node) == POINTER_TYPE)
+    {
+      node = TREE_TYPE (node);
+      was_pointer = 1;
+    }
+  if (TREE_CODE_CLASS (TREE_CODE (node)) == 'd'
+      && DECL_NAME (node) != NULL_TREE)
+    {
+#if 0
+      if (DECL_CONTEXT (node) != NULL_TREE)
+       {
+         put_decl_node (DECL_CONTEXT (node));
+         put_decl_string (".", 1);
+       }
+#endif
+      put_decl_node (DECL_NAME (node));
+      if (TREE_CODE (node) == FUNCTION_DECL && TREE_TYPE (node) != NULL_TREE)
+       {
+         int i = 0;
+         tree args = TYPE_ARG_TYPES (TREE_TYPE (node));
+         if (TREE_CODE (TREE_TYPE (node)) == METHOD_TYPE)
+           args = TREE_CHAIN (args);
+         put_decl_string ("(", 1);
+         for ( ; args != NULL_TREE;  args = TREE_CHAIN (args), i++)
+           {
+             if (i > 0)
+               put_decl_string (",", 1);
+             put_decl_node (TREE_VALUE (args));
+           }
+         put_decl_string (")", 1);
+       }
+    }
+  else if (TREE_CODE_CLASS (TREE_CODE (node)) == 't'
+      && TYPE_NAME (node) != NULL_TREE)
+    {
+      if (TREE_CODE (node) == RECORD_TYPE && TYPE_ARRAY_P (node))
+       {
+         put_decl_node (TYPE_ARRAY_ELEMENT (node));
+         put_decl_string("[]", 2);
+       }
+      else if (node == promoted_byte_type_node)
+       put_decl_string ("byte", 4);
+      else if (node == promoted_short_type_node)
+       put_decl_string ("short", 5);
+      else if (node == promoted_char_type_node)
+       put_decl_string ("char", 4);
+      else if (node == promoted_boolean_type_node)
+       put_decl_string ("boolean", 7);
+      else if (node == void_type_node && was_pointer)
+       put_decl_string ("null", 4);
+      else
+       put_decl_node (TYPE_NAME (node));
+    }
+  else if (TREE_CODE (node) == IDENTIFIER_NODE)
+    put_decl_string (IDENTIFIER_POINTER (node), IDENTIFIER_LENGTH (node));
+  else
+    put_decl_string ("<unknown>", -1);
+}
+
+/* Return a user-friendly name for DECL.
+   The resulting string is only valid until the next call.
+   The value of the hook decl_printable_name is this function,
+   which is also called directly by lang_print_error. */
+
+char *
+lang_printable_name (decl, v)
+     tree decl;
+     int v;
+{
+  decl_bufpos = 0;
+  put_decl_node (decl);
+  put_decl_string ("", 1);
+  return decl_buf;
+}
+
+/* Print on stderr the current class and method context.  This function
+   is the value of the hook print_error_function, called from toplev.c. */
+
+void
+lang_print_error (file)
+     char *file;
+{
+  static tree last_error_function_context = NULL_TREE;
+  static tree last_error_function = NULL;
+
+  if (current_function_decl != NULL
+      && DECL_CONTEXT (current_function_decl) != last_error_function_context)
+    {
+      if (file)
+       fprintf (stderr, "%s: ", file);
+
+      last_error_function_context = DECL_CONTEXT (current_function_decl);
+      fprintf (stderr, "In class `%s':\n",
+              lang_printable_name (last_error_function_context));
+    }
+  if (last_error_function != current_function_decl)
+    {
+      if (file)
+       fprintf (stderr, "%s: ", file);
+
+      if (current_function_decl == NULL)
+       fprintf (stderr, "At top level:\n");
+      else
+       {
+         char *name = lang_printable_name (current_function_decl, 2);
+         fprintf (stderr, "In method `%s':\n", name);
+       }
+
+      last_error_function = current_function_decl;
+    }
+
+}
+
+void
+lang_init ()
+{
+  extern struct rtx_def * java_lang_expand_expr ();
+  extern struct rtx_def * (*lang_expand_expr) ();
+  extern void (*print_error_function) PROTO((char *));
+#if 0
+  extern int flag_minimal_debug;
+  flag_minimal_debug = 0;
+#endif
+
+  decl_printable_name = lang_printable_name;
+  print_error_function = lang_print_error;
+  lang_expand_expr = java_lang_expand_expr;
+
+  JCF_ZERO (main_jcf);
+  main_jcf->read_state = finput;
+  main_jcf->filbuf = jcf_filbuf_from_stdio;
+  current_jcf = main_jcf;
+
+  flag_exceptions = 1;
+}
+
+/* This doesn't do anything on purpose. It's used to satisfy the
+   print_error_function hook we don't print error messages with bogus
+   function prototypes.  */
+
+void java_dummy_print (s)
+     char *s;
+{
+}
+
+/* Called to install the PRINT_ERROR_FUNCTION hook differently
+   according to LEVEL. LEVEL is 1 during early parsing, when function
+   prototypes aren't fully resolved. print_error_function is set so it
+   doesn't print incomplete function prototypes. When LEVEL is 2,
+   function prototypes are fully resolved and can be printed when
+   reporting errors.  */
+
+void lang_init_source (level)
+     int level;
+{
+  extern void (*print_error_function) PROTO((char *));
+  if (level == 1)
+    print_error_function = java_dummy_print;
+  else 
+    print_error_function = lang_print_error;
+}
+
+void
+lang_init_options ()
+{
+  flag_new_exceptions = 1;
+}
+
+void
+lang_finish ()
+{
+}
+
+char*
+lang_identify ()
+{
+  return "Java";
+}
+
+/* Hooks for print_node.  */
+
+void
+print_lang_decl (file, node, indent)
+     FILE *file;
+     tree node;
+     int indent;
+{
+}
+
+void
+print_lang_type (file, node, indent)
+     FILE *file;
+     tree node;
+     int indent;
+{
+}
+
+void
+print_lang_identifier (file, node, indent)
+     FILE *file;
+     tree node;
+     int indent;
+{
+}
+
+void
+print_lang_statistics ()
+{
+}
+
+/* used by print-tree.c */
+
+void
+lang_print_xnode (file, node, indent)
+     FILE *file;
+     tree node;
+     int indent;
+{
+}
diff --git a/gcc/java/lex.c b/gcc/java/lex.c
new file mode 100644 (file)
index 0000000..5099965
--- /dev/null
@@ -0,0 +1,1355 @@
+/* Language lexer for the GNU compiler for the Java(TM) language.
+   Copyright (C) 1997, 1998 Free Software Foundation, Inc.
+   Contributed by Alexandre Petit-Bianco (apbianco@cygnus.com)
+
+This file is part of GNU CC.
+
+GNU CC is free software; you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation; either version 2, or (at your option)
+any later version.
+
+GNU CC is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with GNU CC; see the file COPYING.  If not, write to
+the Free Software Foundation, 59 Temple Place - Suite 330,
+Boston, MA 02111-1307, USA. 
+
+Java and all Java-based marks are trademarks or registered trademarks
+of Sun Microsystems, Inc. in the United States and other countries.
+The Free Software Foundation is independent of Sun Microsystems, Inc.  */
+
+/* It defines java_lex (yylex) that reads a Java ASCII source file
+possibly containing Unicode escape sequence or utf8 encoded characters
+and returns a token for everything found but comments, white spaces
+and line terminators. When necessary, it also fills the java_lval
+(yylval) union. It's implemented to be called by a re-entrant parser
+generated by Bison.
+
+The lexical analysis conforms to the Java grammar described in "The
+Java(TM) Language Specification. J. Gosling, B. Joy, G. Steele.
+Addison Wesley 1996" (http://java.sun.com/docs/books/jls/html/3.doc.html)  */
+
+#include <stdio.h>
+#include <string.h>
+#include <setjmp.h>
+
+#ifdef JAVA_LEX_DEBUG
+#include <ctype.h>
+#endif
+
+#ifdef inline                  /* javaop.h redefines inline as static */
+#undef inline
+#endif
+#include "keyword.h"
+
+#ifndef SEEK_SET
+#include <unistd.h>
+#endif
+
+void
+java_init_lex ()
+{
+  int java_lang_imported = 0;
+
+#ifndef JC1_LITE
+  if (!java_lang_imported)
+    {
+      tree node = build_tree_list 
+       (build_expr_wfl (get_identifier ("java.lang"), NULL, 0, 0), NULL_TREE);
+      read_import_dir (TREE_PURPOSE (node));
+      TREE_CHAIN (node) = ctxp->import_demand_list;
+      ctxp->import_demand_list = node;
+      java_lang_imported = 1;
+    }
+
+  if (!wfl_operator)
+    wfl_operator = build_expr_wfl (NULL_TREE, ctxp->filename, 0, 0);
+  if (!label_id)
+    label_id = get_identifier ("$L");
+
+  ctxp->static_initialized = ctxp->non_static_initialized = 
+    ctxp->incomplete_class = NULL_TREE;
+  
+  bzero (ctxp->modifier_ctx, 11*sizeof (ctxp->modifier_ctx[0]));
+  classpath = NULL;
+  bzero (current_jcf, sizeof (JCF));
+  ctxp->current_parsed_class = NULL;
+  ctxp->package = NULL_TREE;
+#endif
+
+  ctxp->filename = input_filename;
+  ctxp->lineno = lineno = 0;
+  ctxp->p_line = NULL;
+  ctxp->c_line = NULL;
+  ctxp->unget_utf8_value = 0;
+  ctxp->minus_seen = 0;
+  ctxp->java_error_flag = 0;
+}
+
+static char *
+java_sprint_unicode (line, i)
+    struct java_line *line;
+    int i;
+{
+  static char buffer [10];
+  if (line->unicode_escape_p [i] || line->line [i] > 128)
+    sprintf (buffer, "\\u%04x", line->line [i]);
+  else
+    {
+      buffer [0] = line->line [i];
+      buffer [1] = '\0';
+    }
+  return buffer;
+}
+
+static unicode_t
+java_sneak_unicode ()
+{
+  return (ctxp->c_line->line [ctxp->c_line->current]);
+}
+
+static void
+java_unget_unicode (c)
+     unicode_t c;
+{
+  if (!ctxp->c_line->current)
+    fatal ("can't unget unicode - java_unget_unicode");
+  ctxp->c_line->current--;
+  ctxp->c_line->char_col -= JAVA_COLUMN_DELTA (0);
+}
+
+void
+java_allocate_new_line ()
+{
+  int i;
+  unicode_t ahead = (ctxp->c_line ? ctxp->c_line->ahead[0] : '\0');
+  char ahead_escape_p = (ctxp->c_line ? 
+                        ctxp->c_line->unicode_escape_ahead_p : 0);
+
+  if (ctxp->c_line && !ctxp->c_line->white_space_only)
+    {
+      if (ctxp->p_line)
+       {
+         free (ctxp->p_line->unicode_escape_p);
+         free (ctxp->p_line->line);
+         free (ctxp->p_line);
+       }
+      ctxp->p_line = ctxp->c_line;
+      ctxp->c_line = NULL;             /* Reallocated */
+    }
+
+  if (!ctxp->c_line)
+    {
+      ctxp->c_line = (struct java_line *)malloc (sizeof (struct java_line));
+      ctxp->c_line->max = JAVA_LINE_MAX;
+      ctxp->c_line->line = (unicode_t *)malloc 
+         (sizeof (unicode_t)*ctxp->c_line->max);
+      ctxp->c_line->unicode_escape_p = 
+         (char *)malloc (sizeof (char)*ctxp->c_line->max);
+      ctxp->c_line->white_space_only = 0;
+    }
+
+  ctxp->c_line->line [0] = ctxp->c_line->size = 0;
+  ctxp->c_line->char_col = ctxp->c_line->current = 0;
+  if (ahead)
+    {
+      ctxp->c_line->line [ctxp->c_line->size] = ahead;
+      ctxp->c_line->unicode_escape_p [ctxp->c_line->size] = ahead_escape_p;
+      ctxp->c_line->size++;
+    }
+  ctxp->c_line->ahead [0] = 0;
+  ctxp->c_line->unicode_escape_ahead_p = 0;
+  ctxp->c_line->lineno = ++lineno;
+  ctxp->c_line->white_space_only = 1;
+}
+
+static unicode_t
+java_read_char ()
+{
+  int c;
+  int c1, c2;
+
+  if (ctxp->unget_utf8_value)
+    {
+      int to_return = ctxp->unget_utf8_value;
+      ctxp->unget_utf8_value = 0;
+      return (to_return);
+    }
+
+  c = GETC ();
+
+  if (c < 128)
+    return (unicode_t)c;
+  if (c == EOF)
+    return UEOF;
+  else
+    {
+      if (c & 0xe0 == 0xc0)
+        {
+          c1 = GETC ();
+         if (c1 & 0xc0 == 0x80)
+           return (unicode_t)(((c &0x1f) << 6) + (c1 & 0x3f));
+       }
+      else if (c & 0xf0 == 0xe0)
+        {
+          c1 = GETC ();
+         if (c1 & 0xc0 == 0x80)
+           {
+             c2 = GETC ();
+             if (c2 & 0xc0 == 0x80)
+               return (unicode_t)(((c & 0xf) << 12) + 
+                                  (( c1 & 0x3f) << 6) + (c2 & 0x3f));
+           }
+       }
+      java_lex_error ("Bad utf8 encoding", 0);
+    }
+}
+
+static void
+java_store_unicode (l, c, unicode_escape_p)
+    struct java_line *l;
+    unicode_t c;
+    int unicode_escape_p;
+{
+  if (l->size == l->max)
+    {
+      l->max += JAVA_LINE_MAX;
+      l->line = (unicode_t *)realloc (l->line, sizeof (unicode_t)*l->max);
+      l->unicode_escape_p = (char *)realloc (l->unicode_escape_p, 
+                                            sizeof (char)*l->max);
+    }
+  l->line [l->size] = c;
+  l->unicode_escape_p [l->size++] = unicode_escape_p;
+}
+
+static unicode_t
+java_read_unicode (term_context, unicode_escape_p)
+    int term_context;
+    int *unicode_escape_p;
+{
+  unicode_t c;
+  long i, base;
+
+  c = java_read_char ();
+  *unicode_escape_p = 0;
+
+  if (c != '\\')
+    return ((term_context ? c : 
+            java_lineterminator (c) ? '\n' : (unicode_t)c));
+
+  /* Count the number of preceeding '\' */
+  for (base = ftell (finput), i = base-2; c == '\\';)
+    { 
+      fseek (finput, i--, SEEK_SET);
+      c = java_read_char ();   /* Will fail if reading utf8 stream. FIXME */
+    }
+  fseek (finput, base, SEEK_SET);
+  if ((base-i-3)%2 == 0)       /* If odd number of \ seen */
+    {
+      c = java_read_char ();
+      if (c == 'u')
+        {
+         unsigned short unicode = 0;
+         int shift = 12;
+         /* Next should be 4 hex digits, otherwise it's an error.
+            The hex value is converted into the unicode, pushed into
+            the Unicode stream.  */
+         for (shift = 12; shift >= 0; shift -= 4)
+           {
+             if ((c = java_read_char ()) == UEOF)
+               return UEOF;
+             if (c >= '0' && c <= '9')
+               unicode |= (unicode_t)((c-'0') << shift);
+             else if ((c >= 'a' && c <= 'f') || (c >= 'A' && c <= 'F'))
+               unicode |= (unicode_t)(10+(c | 0x20)-'a' << shift);
+             else
+                 java_lex_error 
+                   ("Non hex digit in Unicode escape sequence", 0);
+           }
+         *unicode_escape_p = 1;
+         return (term_context ? unicode :
+                 (java_lineterminator (c) ? '\n' : unicode));
+       }
+      UNGETC (c);
+    }
+  return (unicode_t)'\\';
+}
+
+static unicode_t
+java_get_unicode ()
+{
+  /* It's time to read a line when... */
+  if (!ctxp->c_line || ctxp->c_line->current == ctxp->c_line->size)
+    {
+      unicode_t c;
+      java_allocate_new_line ();
+      if (ctxp->c_line->line[0] != '\n')
+       for (;;)
+         {
+           int unicode_escape_p;
+           c = java_read_unicode (0, &unicode_escape_p);
+           java_store_unicode (ctxp->c_line, c, unicode_escape_p);
+           if (ctxp->c_line->white_space_only 
+               && !JAVA_WHITE_SPACE_P (c) && c!='\n')
+             ctxp->c_line->white_space_only = 0;
+           if ((c == '\n') || (c == UEOF))
+             break;
+         }
+    }
+  ctxp->c_line->char_col += JAVA_COLUMN_DELTA (0);
+  JAVA_LEX_CHAR (ctxp->c_line->line [ctxp->c_line->current]);
+  return ctxp->c_line->line [ctxp->c_line->current++];
+}
+
+static int
+java_lineterminator (c)
+     unicode_t c;
+{
+  int unicode_escape_p;
+  if (c == '\n')               /* CR */
+    {
+      if ((c = java_read_unicode (1, &unicode_escape_p)) != '\r')
+       {
+         ctxp->c_line->ahead [0] = c;
+         ctxp->c_line->unicode_escape_ahead_p = unicode_escape_p;
+       }
+      return 1;
+    }
+  else if (c == '\r')          /* LF */
+    {
+      if ((c = java_read_unicode (1, &unicode_escape_p)) != '\n')
+       {
+         ctxp->c_line->ahead [0] = c;
+         ctxp->c_line->unicode_escape_ahead_p = unicode_escape_p;
+       }
+      return 1;
+    }
+  else 
+    return 0;
+}
+
+/* Parse the end of a C style comment */
+static void
+java_parse_end_comment ()
+{
+  unicode_t c;
+
+  for (c = java_get_unicode ();; c = java_get_unicode ())
+    {
+      switch (c)
+       {
+       case UEOF:
+         java_lex_error ("Comment not terminated at end of input", 0);
+       case '*':
+         switch (c = java_get_unicode ())
+           {
+           case UEOF:
+             java_lex_error ("Comment not terminated at end of input", 0);
+           case '/':
+             return;
+           case '*':   /* reparse only '*' */
+             java_unget_unicode (c);
+           }
+       }
+    }
+}
+
+/* This function to be used only by JAVA_ID_CHAR_P (), otherwise it
+   will return a wrong result.  */
+static int
+java_letter_or_digit_p (c)
+     unicode_t c;
+{
+  return _JAVA_LETTER_OR_DIGIT_P (c);
+}
+
+static unicode_t
+java_parse_escape_sequence ()
+{
+  unicode_t char_lit;
+  unicode_t c;
+
+  switch (c = java_get_unicode ())
+    {
+    case 'b':
+      return (unicode_t)0x8;
+    case 't':
+      return (unicode_t)0x9;
+    case 'n':
+      return (unicode_t)0xa;
+    case 'f':
+      return (unicode_t)0xc;
+    case 'r':
+      return (unicode_t)0xd;
+    case '"':
+      return (unicode_t)0x22;
+    case '\'':
+      return (unicode_t)0x27;
+    case '\\':
+      return (unicode_t)0x5c;
+    case '0': case '1': case '2': case '3': case '4':
+    case '5': case '6': case '7': case '8': case '9':
+      {
+       int octal_escape[3];
+       int octal_escape_index = 0;
+       
+       for (; octal_escape_index < 3 && RANGE (c, '0', '9');
+            c = java_get_unicode ())
+         octal_escape [octal_escape_index++] = c;
+
+       java_unget_unicode (c);
+
+       if ((octal_escape_index == 3) && (octal_escape [0] > '3'))
+         {
+           java_lex_error ("Literal octal escape out of range", 0);
+           return JAVA_CHAR_ERROR;
+         }
+       else
+         {
+           int i, shift;
+           for (char_lit=0, i = 0, shift = 3*(octal_escape_index-1);
+                i < octal_escape_index; i++, shift -= 3)
+             char_lit |= (octal_escape [i] - '0') << shift;
+
+           return (char_lit);
+         }
+       break;
+      }
+    case '\n':
+      return '\n';             /* ULT, caught latter as a specific error */
+    default:
+      java_lex_error ("Illegal character in escape sequence", 0);
+      return JAVA_CHAR_ERROR;
+    }
+}
+
+int
+#ifdef JC1_LITE
+yylex (java_lval)
+#else
+java_lex (java_lval)
+#endif
+     YYSTYPE *java_lval;
+{
+  unicode_t c, first_unicode;
+  int line_terminator;
+  int ascii_index, all_ascii;
+  char *string;
+
+  /* Translation of the Unicode escape in the raw stream of Unicode
+     characters. Takes care of line terminator.  */
+ step1:
+  /* Skip white spaces: SP, TAB and FF or ULT */ 
+  for (c = java_get_unicode ();
+       c == '\n' || JAVA_WHITE_SPACE_P (c); c = java_get_unicode ())
+    if (c == '\n')
+      {
+       ctxp->elc.line = ctxp->c_line->lineno;
+       ctxp->elc.col  = ctxp->c_line->char_col-2;
+      }
+
+  ctxp->elc.col = (ctxp->elc.col < 0 ? 0 : ctxp->elc.col);
+
+  if (c == 0x1a)               /* CTRL-Z */
+    {
+      if ((c = java_get_unicode ()) == UEOF)
+       return 0;               /* Ok here */
+      else
+       java_unget_unicode (c); /* Caught latter at the end the function */
+    }
+  /* Handle EOF here */
+  if (c == UEOF)       /* Should probably do something here... */
+    return 0;
+
+  /* Take care of eventual comments.  */
+  if (c == '/')
+    {
+      switch (c = java_get_unicode ())
+       {
+       case '/':
+         for (c = java_get_unicode ();;c = java_get_unicode ())
+           {
+             if (c == UEOF)
+               java_lex_error ("Comment not terminated at end of input", 0);
+             if (c == '\n')    /* ULT */
+               goto step1;
+           }
+         break;
+
+       case '*':
+         if ((c = java_get_unicode ()) == '*')
+           {
+             if ((c = java_get_unicode ()) == '/')
+               goto step1;     /* Empy documentation comment  */
+
+             else
+               /* Parsing the documentation section. We're looking
+                for the @depracated pseudo keyword.  the @deprecated
+                tag must be at the beginning of a doc comment line
+                (ignoring white space and any * character)  */
+
+               { 
+                 int valid_tag = 0, seen_star;
+
+                 while (JAVA_WHITE_SPACE_P (c) || (c == '*') || c == '\n')
+                   {
+                     switch (c)
+                       {
+                       case '*':
+                         seen_star = 1;
+                         break;
+                       case '\n': /* ULT */
+                         valid_tag = 1;
+                         break;
+                       default:
+                         seen_star = 0;
+                       }
+                     c = java_get_unicode();
+                   }
+                 
+                 if (c == UEOF)
+                   java_lex_error 
+                     ("Comment not terminated at end of input", 0);
+
+                 if (seen_star && (c == '/'))
+                   goto step1; /* End of documentation */
+
+                 if (valid_tag && (c == '@'))
+                   {
+                     char deprecated [10];
+                     int  deprecated_index = 0;
+
+                     for (deprecated_index = 0, c = java_get_unicode (); 
+                          deprecated_index < 10 && c != UEOF;
+                          c = java_get_unicode ())
+                       deprecated [deprecated_index++] = c;
+
+                     if (c == UEOF)
+                       java_lex_error 
+                         ("Comment not terminated at end of input", 0);
+                     
+                     java_unget_unicode (c);
+                     deprecated [deprecated_index] = '\0';
+                     if (!strcmp (deprecated, "deprecated"))
+                       {
+                         /* Set global flag to be checked by class. FIXME  */
+                         warning ("deprecated implementation found");
+                       }
+                   }
+               }
+           }
+         else
+           java_unget_unicode (c);
+
+         java_parse_end_comment ();
+         goto step1;
+         break;
+       default:
+         java_unget_unicode (c);
+         c = '/';
+         break;
+       }
+    }
+
+  ctxp->elc.line = ctxp->c_line->lineno;
+  ctxp->elc.col = ctxp->c_line->char_col - JAVA_COLUMN_DELTA (-1);
+  if (ctxp->elc.col < 0)
+    fatal ("ctxp->elc.col < 0 - java_lex");
+
+  /* Numeric literals */
+  if (JAVA_ASCII_DIGIT (c) || (c == '.'))
+    {
+      unicode_t peep;
+      /* This section of code is borrowed from gcc/c-lex.c  */
+#define TOTAL_PARTS ((HOST_BITS_PER_WIDE_INT / HOST_BITS_PER_CHAR) * 2 + 2)
+      int parts[TOTAL_PARTS];
+      HOST_WIDE_INT high, low;
+      /* End borrowed section  */
+      char literal_token [256];
+      int  literal_index = 0, radix = 10, long_suffix = 0, overflow = 0, bytes;
+      int  i;
+      int  number_beginning = ctxp->c_line->current;
+      
+      /* We might have a . separator instead of a FP like .[0-9]* */
+      if (c == '.')
+       {
+         unicode_t peep = java_sneak_unicode ();
+
+         if (!JAVA_ASCII_DIGIT (peep))
+           {
+             JAVA_LEX_SEP('.');
+             BUILD_OPERATOR (DOT_TK);
+           }
+       }
+
+      for (i = 0; i < TOTAL_PARTS; i++)
+       parts [i] = 0;
+
+      if (c == '0')
+       {
+         c = java_get_unicode ();
+         if (c == 'x' || c == 'X')
+           {
+             radix = 16;
+             c = java_get_unicode ();
+           }
+         else if (JAVA_ASCII_DIGIT (c))
+           radix = 8;
+         else if (c == '.')
+           {
+             /* Push the '.' back and prepare for a FP parsing... */
+             java_unget_unicode (c);
+             c = '0';
+           }
+         else
+           {
+             /* We have a zero literal: 0, 0{f,F}, 0{d,D} */
+             JAVA_LEX_LIT ("0", 10);
+              switch (c)
+               {               
+               case 'L': case 'l':
+                 SET_LVAL_NODE_TYPE (integer_zero_node, long_type_node);
+                 return (INT_LIT_TK);
+               case 'f': case 'F':
+                 SET_LVAL_NODE_TYPE (build_real (float_type_node, dconst0),
+                                       float_type_node);
+                 return (FP_LIT_TK);
+               case 'd': case 'D':
+                 SET_LVAL_NODE_TYPE (build_real (double_type_node, dconst0),
+                                       double_type_node);
+                 return (FP_LIT_TK);
+               default:
+                 java_unget_unicode (c);
+                 SET_LVAL_NODE_TYPE (integer_zero_node, int_type_node);
+                 return (INT_LIT_TK);
+               }
+           }
+       }
+      /* Parse the first part of the literal, until we find something
+        which is not a number.  */
+      while ((radix == 10 && JAVA_ASCII_DIGIT (c)) ||
+            (radix == 16 && JAVA_ASCII_HEXDIGIT (c)) ||
+            (radix == 8  && JAVA_ASCII_OCTDIGIT (c)))
+       {
+         /* We store in a string (in case it turns out to be a FP) and in
+            PARTS if we have to process a integer literal.  */
+         int numeric = (RANGE (c, '0', '9') ? c-'0' : 10 +(c|0x20)-'a');
+         int count;
+
+         literal_token [literal_index++] = c;
+         /* This section of code if borrowed from gcc/c-lex.c  */
+         for (count = 0; count < TOTAL_PARTS; count++)
+           {
+             parts[count] *= radix;
+             if (count)
+               {
+                 parts[count]   += (parts[count-1] >> HOST_BITS_PER_CHAR);
+                 parts[count-1] &= (1 << HOST_BITS_PER_CHAR) - 1;
+               }
+             else
+               parts[0] += numeric;
+           }
+         if (parts [TOTAL_PARTS-1] != 0)
+           overflow = 1;
+         /* End borrowed section.  */
+         c = java_get_unicode ();
+       }
+
+      /* If we have something from the FP char set but not a digit, parse
+        a FP literal.  */
+      if (JAVA_ASCII_FPCHAR (c) && !JAVA_ASCII_DIGIT (c))
+       {
+         int stage = 0;
+         int seen_digit = (literal_index ? 1 : 0);
+         int seen_exponent = 0;
+         int fflag = 0;        /* 1 for {f,F}, 0 for {d,D}. FP literal are
+                                  double unless specified. */
+         if (radix != 10)
+           java_lex_error ("Can't express non-decimal FP literal", 0);
+
+         for (;;)
+           {
+             if (c == '.')
+               {
+                 if (stage < 1)
+                   {
+                     stage = 1;
+                     literal_token [literal_index++ ] = c;
+                     c = java_get_unicode ();
+                   }
+                 else
+                   java_lex_error ("Invalid character in FP literal", 0);
+               }
+
+             if (c == 'e' || c == 'E')
+               {
+                 if (stage < 2)
+                   {
+                     /* {E,e} must have seen at list a digit */
+                     if (!seen_digit)
+                       java_lex_error ("Invalid FP literal", 0);
+                     seen_digit = 0;
+                     seen_exponent = 1;
+                     stage = 2;
+                     literal_token [literal_index++] = c;
+                     c = java_get_unicode ();
+                   }
+                 else
+                   java_lex_error ("Invalid character in FP literal", 0);
+               }
+             if ( c == 'f' || c == 'F' || c == 'd' || c == 'D')
+               {
+                 fflag = ((c == 'd') || (c == 'D')) ? 0 : 1;
+                 stage = 4;    /* So we fall through */
+               }
+
+             if ((c=='-' || c =='+') && stage < 3)
+               {
+                 stage = 3;
+                 literal_token [literal_index++] = c;
+                 c = java_get_unicode ();
+               }
+
+             if ((stage == 0 && JAVA_ASCII_FPCHAR (c)) ||
+                 (stage == 1 && JAVA_ASCII_FPCHAR (c) && !(c == '.')) ||
+                 (stage == 2 && (JAVA_ASCII_DIGIT (c) || JAVA_FP_PM (c))) ||
+                 (stage == 3 && JAVA_ASCII_DIGIT (c)))
+               {
+                 if (JAVA_ASCII_DIGIT (c))
+                   seen_digit = 1;
+                 literal_token [literal_index++ ] = c;
+                 c = java_get_unicode ();
+               }
+             else
+               {
+                 jmp_buf handler;
+                 REAL_VALUE_TYPE value;
+#ifndef JC1_LITE
+                 tree type = (fflag ? FLOAT_TYPE_NODE : DOUBLE_TYPE_NODE);
+#endif
+
+                 if (stage != 4) /* Don't push back fF/dD */
+                   java_unget_unicode (c);
+                 
+                 /* An exponent (if any) must have seen a digit.  */
+                 if (seen_exponent && !seen_digit)
+                   java_lex_error ("Invalid FP literal", 0);
+
+                 literal_token [literal_index] = '\0';
+                 JAVA_LEX_LIT (literal_token, radix);
+
+                 if (setjmp (handler))
+                   {
+                     JAVA_FLOAT_RANGE_ERROR ((fflag ? "float" : "double"));
+                     value = DCONST0;
+                   }
+                 else
+                   {
+                     SET_FLOAT_HANDLER (handler);
+                     SET_REAL_VALUE_ATOF 
+                       (value, REAL_VALUE_ATOF (literal_token, 
+                                                TYPE_MODE (type)));
+
+                     if (REAL_VALUE_ISINF (value))
+                       JAVA_FLOAT_RANGE_ERROR ((fflag ? "float" : "double"));
+
+                     if (REAL_VALUE_ISNAN (value))
+                       JAVA_FLOAT_RANGE_ERROR ((fflag ? "float" : "double"));
+
+                     SET_LVAL_NODE_TYPE (build_real (type, value), type);
+                     SET_FLOAT_HANDLER (NULL_PTR);
+                     return FP_LIT_TK;
+                   }
+               }
+           }
+       } /* JAVA_ASCCI_FPCHAR (c) */
+
+      /* Here we get back to converting the integral literal.  */
+      if (c == 'L' || c == 'l')
+       long_suffix = 1;
+      else if (radix == 16 && JAVA_ASCII_LETTER (c))
+       java_lex_error ("Digit out of range in hexadecimal literal", 0);
+      else if (radix == 8  && JAVA_ASCII_DIGIT (c))
+       java_lex_error ("Digit out of range in octal literal", 0);
+      else if (radix == 16 && !literal_index)
+       java_lex_error ("No digit specified for hexadecimal literal", 0);
+      else
+       java_unget_unicode (c);
+
+#ifdef JAVA_LEX_DEBUG
+      literal_token [literal_index] = '\0'; /* So JAVA_LEX_LIT is safe. */
+      JAVA_LEX_LIT (literal_token, radix);
+#endif
+      /* This section of code is borrowed from gcc/c-lex.c  */
+      if (!overflow)
+       {
+         bytes = GET_TYPE_PRECISION (long_type_node);
+         for (i = bytes; i < TOTAL_PARTS; i++)
+           if (parts [i])
+             {
+               overflow = 1;
+               break;
+             }
+       }
+      high = low = 0;
+      for (i = 0; i < HOST_BITS_PER_WIDE_INT / HOST_BITS_PER_CHAR; i++)
+       {
+         high |= ((HOST_WIDE_INT) parts[i + (HOST_BITS_PER_WIDE_INT
+                                             / HOST_BITS_PER_CHAR)]
+                  << (i * HOST_BITS_PER_CHAR));
+         low |= (HOST_WIDE_INT) parts[i] << (i * HOST_BITS_PER_CHAR);
+       }
+      /* End borrowed section.  */
+
+      /* Range checking */
+      if (long_suffix)
+       {
+         /* 9223372036854775808L is valid if operand of a '-'. Otherwise
+            9223372036854775807L is the biggest `long' literal that can be
+            expressed using a 10 radix. For other radixes, everything that
+            fits withing 64 bits is OK. */
+         int hb = (high >> 31);
+         if (overflow || (hb && low && radix == 10) ||  
+             (hb && high & 0x7fffffff && radix == 10) ||
+             (hb && !(high & 0x7fffffff) && !ctxp->minus_seen && radix == 10))
+           JAVA_INTEGRAL_RANGE_ERROR ("Numeric overflow for `long' literal");
+       }
+      else
+       {
+         /* 2147483648 is valid if operand of a '-'. Otherwise,
+            2147483647 is the biggest `int' literal that can be
+            expressed using a 10 radix. For other radixes, everything
+            that fits within 32 bits is OK. */
+         int hb = (low >> 31) & 0x1;
+         if (overflow || high || (hb && low & 0x7fffffff && radix == 10) ||
+             (hb && !(low & 0x7fffffff) && !ctxp->minus_seen && radix == 10))
+           JAVA_INTEGRAL_RANGE_ERROR ("Numeric overflow for `int' literal");
+       }
+      ctxp->minus_seen = 0;
+      SET_LVAL_NODE_TYPE (build_int_2 (low, high),
+                         (long_suffix ? long_type_node : int_type_node));
+      return INT_LIT_TK;
+    }
+
+  ctxp->minus_seen = 0;
+  /* Character literals */
+  if (c == '\'')
+    {
+      unicode_t char_lit;
+      if ((c = java_get_unicode ()) == '\\')
+       char_lit = java_parse_escape_sequence ();
+      else
+       char_lit = c;
+
+      c = java_get_unicode ();
+      
+      if ((c == '\n') || (c == UEOF))
+       java_lex_error ("Character literal not terminated at end of line", 0);
+      if (c != '\'')
+       java_lex_error ("Syntax error in character literal", 0);
+
+      if (c == JAVA_CHAR_ERROR)
+        char_lit = 0;          /* We silently convert it to zero */
+
+      JAVA_LEX_CHAR_LIT (char_lit);
+      SET_LVAL_NODE_TYPE (build_int_2 (char_lit, 0), char_type_node);
+      return CHAR_LIT_TK;
+    }
+
+  /* String literals */
+  if (c == '"')
+    {
+      int no_error;
+      char *string;
+
+      for (no_error = 1, c = java_get_unicode (); 
+          c != '"' && c != '\n'; c = java_get_unicode ())
+       {
+         if (c == '\\')
+           c = java_parse_escape_sequence ();
+         no_error &= (c != JAVA_CHAR_ERROR ? 1 : 0);
+         if (c)
+           java_unicode_2_utf8 (c);
+       }
+      if (c == '\n' || c == UEOF) /* ULT */
+       {
+         lineno--;             /* Refer to the line the terminator was seen */
+         java_lex_error ("String not terminated at end of line.", 0);
+         lineno++;
+       }
+
+      obstack_1grow (&temporary_obstack, '\0');
+      string = obstack_finish (&temporary_obstack);
+      if (!no_error || (c != '"'))
+       *string = '\0';         /* Silently turns the string to an empty one */
+      
+      JAVA_LEX_STR_LIT (string)
+      
+#ifndef JC1_LITE
+      if (*string)
+       {
+         extern struct obstack *expression_obstack;
+         tree s = make_node (STRING_CST);
+         TREE_STRING_LENGTH (s) = strlen (string);
+         TREE_STRING_POINTER (s) = 
+           obstack_alloc (expression_obstack, strlen (string));
+         strcpy (TREE_STRING_POINTER (s), string);
+         java_lval->node = s;
+       }
+      else
+       java_lval->node = error_mark_node;
+#endif
+      return STRING_LIT_TK;
+      
+    }
+
+  /* Separator */
+  switch (c)
+    {
+    case '(':
+      JAVA_LEX_SEP (c);
+      BUILD_OPERATOR (OP_TK);
+    case ')':
+      JAVA_LEX_SEP (c);
+      return CP_TK;
+    case '{':
+      JAVA_LEX_SEP (c);
+      if (ctxp->ccb_indent == 1)
+       ctxp->first_ccb_indent1 = lineno;
+      ctxp->ccb_indent++;
+      return OCB_TK;
+    case '}':
+      JAVA_LEX_SEP (c);
+      ctxp->ccb_indent--;
+      if (ctxp->ccb_indent == 1)
+        ctxp->last_ccb_indent1 = lineno;
+      return CCB_TK;
+    case '[':
+      JAVA_LEX_SEP (c);
+      BUILD_OPERATOR (OSB_TK);
+    case ']':
+      JAVA_LEX_SEP (c);
+      return CSB_TK;
+    case ';':
+      JAVA_LEX_SEP (c);
+      return SC_TK;
+    case ',':
+      JAVA_LEX_SEP (c);
+      return C_TK;
+    case '.':
+      JAVA_LEX_SEP (c);
+      BUILD_OPERATOR (DOT_TK);
+      /*      return DOT_TK; */
+    }
+
+  /* Operators */
+  switch (c)
+    {
+    case '=':
+      if ((c = java_get_unicode ()) == '=')
+       {
+         BUILD_OPERATOR (EQ_TK);
+       }
+      else
+       {
+         /* Equals is used in two different locations. In the 
+            variable_declarator: rule, it has to be seen as '=' as opposed
+            to being seen as an ordinary assignment operator in
+            assignment_operators: rule.  */
+         java_unget_unicode (c);
+         BUILD_OPERATOR (ASSIGN_TK);
+       }
+      
+    case '>':
+      switch ((c = java_get_unicode ()))
+       {
+       case '=':
+         BUILD_OPERATOR (GTE_TK);
+       case '>':
+         switch ((c = java_get_unicode ()))
+           {
+           case '>':
+             if ((c = java_get_unicode ()) == '=')
+               {
+                 BUILD_OPERATOR2 (ZRS_ASSIGN_TK);
+               }
+             else
+               {
+                 java_unget_unicode (c);
+                 BUILD_OPERATOR (ZRS_TK);
+               }
+           case '=':
+             BUILD_OPERATOR2 (SRS_ASSIGN_TK);
+           default:
+             java_unget_unicode (c);
+             BUILD_OPERATOR (SRS_TK);
+           }
+       default:
+         java_unget_unicode (c);
+         BUILD_OPERATOR (GT_TK);
+       }
+       
+    case '<':
+      switch ((c = java_get_unicode ()))
+       {
+       case '=':
+         BUILD_OPERATOR (LTE_TK);
+       case '<':
+         if ((c = java_get_unicode ()) == '=')
+           {
+             BUILD_OPERATOR2 (LS_ASSIGN_TK);
+           }
+         else
+           {
+             java_unget_unicode (c);
+             BUILD_OPERATOR (LS_TK);
+           }
+       default:
+         java_unget_unicode (c);
+         BUILD_OPERATOR (LT_TK);
+       }
+
+    case '&':
+      switch ((c = java_get_unicode ()))
+       {
+       case '&':
+         BUILD_OPERATOR (BOOL_AND_TK);
+       case '=':
+         BUILD_OPERATOR2 (AND_ASSIGN_TK);
+       default:
+         java_unget_unicode (c);
+         BUILD_OPERATOR (AND_TK);
+       }
+
+    case '|':
+      switch ((c = java_get_unicode ()))
+       {
+       case '|':
+         BUILD_OPERATOR (BOOL_OR_TK);
+       case '=':
+         BUILD_OPERATOR2 (OR_ASSIGN_TK);
+       default:
+         java_unget_unicode (c);
+         BUILD_OPERATOR (OR_TK);
+       }
+
+    case '+':
+      switch ((c = java_get_unicode ()))
+       {
+       case '+':
+         BUILD_OPERATOR (INCR_TK);
+       case '=':
+         BUILD_OPERATOR2 (PLUS_ASSIGN_TK);
+       default:
+         java_unget_unicode (c);
+         BUILD_OPERATOR (PLUS_TK);
+       }
+
+    case '-':
+      switch ((c = java_get_unicode ()))
+       {
+       case '-':
+         BUILD_OPERATOR (DECR_TK);
+       case '=':
+         BUILD_OPERATOR2 (MINUS_ASSIGN_TK);
+       default:
+         java_unget_unicode (c);
+         ctxp->minus_seen = 1;
+         BUILD_OPERATOR (MINUS_TK);
+       }
+
+    case '*':
+      if ((c = java_get_unicode ()) == '=')
+       {
+         BUILD_OPERATOR2 (MULT_ASSIGN_TK);
+       }
+      else
+       {
+         java_unget_unicode (c);
+         BUILD_OPERATOR (MULT_TK);
+       }
+
+    case '/':
+      if ((c = java_get_unicode ()) == '=')
+       {
+         BUILD_OPERATOR2 (DIV_ASSIGN_TK);
+       }
+      else
+       {
+         java_unget_unicode (c);
+         BUILD_OPERATOR (DIV_TK);
+       }
+
+    case '^':
+      if ((c = java_get_unicode ()) == '=')
+       {
+         BUILD_OPERATOR2 (XOR_ASSIGN_TK);
+       }
+      else
+       {
+         java_unget_unicode (c);
+         BUILD_OPERATOR (XOR_TK);
+       }
+
+    case '%':
+      if ((c = java_get_unicode ()) == '=')
+       {
+         BUILD_OPERATOR2 (REM_ASSIGN_TK);
+       }
+      else
+       {
+         java_unget_unicode (c);
+         BUILD_OPERATOR (REM_TK);
+       }
+
+    case '!':
+      if ((c = java_get_unicode()) == '=')
+       {
+         BUILD_OPERATOR (NEQ_TK);
+       }
+      else
+       {
+         java_unget_unicode (c);
+         BUILD_OPERATOR (NEG_TK);
+       }
+         
+    case '?':
+      JAVA_LEX_OP ("?");
+      BUILD_OPERATOR (REL_QM_TK);
+    case ':':
+      JAVA_LEX_OP (":");
+      BUILD_OPERATOR (REL_CL_TK);
+    case '~':
+      BUILD_OPERATOR (NOT_TK);
+    }
+  
+  /* Keyword, boolean literal or null literal */
+  for (first_unicode = c, all_ascii = 1, ascii_index = 0; 
+       JAVA_ID_CHAR_P (c); c = java_get_unicode ())
+    {
+      java_unicode_2_utf8 (c);
+      if (all_ascii && c >= 128)
+        all_ascii = 0;
+      ascii_index++;
+    }
+
+  obstack_1grow (&temporary_obstack, '\0');
+  string = obstack_finish (&temporary_obstack);
+  java_unget_unicode (c);
+
+  /* If we have something all ascii, we consider a keyword, a boolean
+     literal, a null literal or an all ASCII identifier.  Otherwise,
+     this is an identifier (possibly not respecting formation rule).  */
+  if (all_ascii)
+    {
+      struct java_keyword *kw;
+      if ((kw=java_keyword (string, ascii_index)))
+       {
+         JAVA_LEX_KW (string);
+         switch (kw->token)
+           {
+           case PUBLIC_TK:       case PROTECTED_TK: case STATIC_TK:
+           case ABSTRACT_TK:     case FINAL_TK:     case NATIVE_TK:
+           case SYNCHRONIZED_TK: case TRANSIENT_TK: case VOLATILE_TK:
+           case PRIVATE_TK:
+             SET_MODIFIER_CTX (kw->token);
+             return MODIFIER_TK;
+           case FLOAT_TK:
+             SET_LVAL_NODE (float_type_node);
+             return FP_TK;
+           case DOUBLE_TK:
+             SET_LVAL_NODE (double_type_node);
+             return FP_TK;
+           case BOOLEAN_TK:
+             SET_LVAL_NODE (boolean_type_node);
+             return BOOLEAN_TK;
+           case BYTE_TK:
+             SET_LVAL_NODE (byte_type_node);
+             return INTEGRAL_TK;
+           case SHORT_TK:
+             SET_LVAL_NODE (short_type_node);
+             return INTEGRAL_TK;
+           case INT_TK:
+             SET_LVAL_NODE (int_type_node);
+             return INTEGRAL_TK;
+           case LONG_TK:
+             SET_LVAL_NODE (long_type_node);
+             return INTEGRAL_TK;
+           case CHAR_TK:
+             SET_LVAL_NODE (char_type_node);
+             return INTEGRAL_TK;
+
+             /* Keyword based literals */
+           case TRUE_TK:
+           case FALSE_TK:
+             SET_LVAL_NODE ((kw->token == TRUE_TK ? 
+                             boolean_true_node : boolean_false_node));
+             return BOOL_LIT_TK;
+           case NULL_TK:
+             SET_LVAL_NODE (null_pointer_node);
+             return NULL_TK;
+
+             /* We build an operator for SUPER, so we can keep its position */
+           case SUPER_TK:
+           case THIS_TK:
+           case RETURN_TK:
+           case BREAK_TK:
+           case CONTINUE_TK:
+             BUILD_OPERATOR (kw->token);
+
+           default:
+             return kw->token;
+           }
+       }
+    }
+  
+  /* We may have and ID here */
+  if (JAVA_ID_CHAR_P(first_unicode) && !JAVA_DIGIT_P (first_unicode))
+    {
+      JAVA_LEX_ID (string);
+      java_lval->node = BUILD_ID_WFL (GET_IDENTIFIER (string));
+      return ID_TK;
+    }
+
+  /* Everything else is an invalid character in the input */
+  {
+    char lex_error_buffer [128];
+    sprintf (lex_error_buffer, "Invalid character '%s' in input", 
+            java_sprint_unicode (ctxp->c_line, ctxp->c_line->current));
+    java_lex_error (lex_error_buffer, 1);
+  }
+  return 0;
+}
+
+static void
+java_unicode_2_utf8 (unicode)
+    unicode_t unicode;
+{
+  if (RANGE (unicode, 0x01, 0x7f))
+    obstack_1grow (&temporary_obstack, (char)unicode);
+  else if (RANGE (unicode, 0x80, 0x7ff) || unicode == 0)
+    {
+      obstack_1grow (&temporary_obstack,
+                    (unsigned char)(0xc0 | ((0x7c0 & unicode) >> 6)));
+      obstack_1grow (&temporary_obstack,
+                    (unsigned char)(0x80 | (unicode & 0x3f)));
+    }
+  else                         /* Range 0x800-0xffff */
+    {
+      obstack_1grow (&temporary_obstack,
+                    (unsigned char)(0xe0 | (unicode & 0xf000) >> 12));
+      obstack_1grow (&temporary_obstack,
+                    (unsigned char)(0x80 | (unicode & 0x0fc0) >> 6));
+      obstack_1grow (&temporary_obstack,
+                    (unsigned char)(0x80 | (unicode & 0x003f) >> 12));
+    }
+}
+
+#ifndef JC1_LITE
+static tree
+build_wfl_node (node)
+     tree node;
+{
+  return build_expr_wfl (node, ctxp->filename, ctxp->elc.line, ctxp->elc.col);
+}
+#endif
+
+static void
+java_lex_error (msg, forward)
+     char *msg;
+     int forward;
+{
+#ifndef JC1_LITE
+  ctxp->elc.line = ctxp->c_line->lineno;
+  ctxp->elc.col = ctxp->c_line->char_col-1+forward;
+
+  /* Might be caught in the middle of some error report */
+  ctxp->java_error_flag = 0;
+  java_error (NULL);
+  java_error (msg);
+#endif
+}
+
+static int
+java_is_eol (fp, c)
+  FILE *fp;
+  int c;
+{
+  int next;
+  switch (c)
+    {
+    case '\n':
+      next = getc (fp);
+      if (next != '\r' && next != EOF)
+       ungetc (next, fp);
+      return 1;
+    case '\r':
+      return 1;
+    default:
+      return 0;
+    }  
+}
+
+char *
+java_get_line_col (filename, line, col)
+     char *filename;
+     int line, col;
+{
+#ifdef JC1_LITE
+  return 0;
+#else
+  /* Dumb implementation. Doesn't try to cache or optimize things. */
+  /* First line of the file is line 1, first column is 1 */
+
+  /* COL <= 0 means, at the CR/LF in LINE */
+
+  FILE *fp;
+  int c, ccol, cline = 1;
+  int current_line_col = 0;
+
+  if (!(fp = fopen (filename, "r")))
+    fatal ("Can't open file - java_display_line_col");
+
+  while (cline != line)
+    {
+      c = getc (fp);
+      if (c < 0)
+       {
+         static char msg[] = "<<file too short - unexpected EOF>>";
+         obstack_grow (&temporary_obstack, msg, sizeof(msg)-1);
+         goto have_line;
+       }
+      if (java_is_eol (fp, c))
+       cline++;
+    }
+
+  /* Gather the chars of the current line in a buffer */
+  for (;;)
+    {
+      c = getc (fp);
+      if (c < 0 || java_is_eol (fp, c))
+       break;
+      obstack_1grow (&temporary_obstack, c);
+      current_line_col++;
+    }
+ have_line:
+
+  obstack_1grow (&temporary_obstack, '\n');
+
+  if (col < 0)
+    col = current_line_col;
+
+  /* Place the '^' a the right position */
+  for (ccol = 1; ccol <= col; ccol++)
+    obstack_1grow (&temporary_obstack, ' ');
+  obstack_grow0 (&temporary_obstack, "^", 1);
+
+  fclose (fp);
+  return obstack_finish (&temporary_obstack);
+#endif
+}
diff --git a/gcc/java/lex.h b/gcc/java/lex.h
new file mode 100644 (file)
index 0000000..9116e42
--- /dev/null
@@ -0,0 +1,512 @@
+/* Language lexer definitions for the GNU compiler for the Java(TM) language.
+   Copyright (C) 1997, 1998 Free Software Foundation, Inc.
+   Contributed by Alexandre Petit-Bianco (apbianco@cygnus.com)
+
+This file is part of GNU CC.
+
+GNU CC is free software; you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation; either version 2, or (at your option)
+any later version.
+
+GNU CC is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with GNU CC; see the file COPYING.  If not, write to
+the Free Software Foundation, 59 Temple Place - Suite 330,
+Boston, MA 02111-1307, USA.
+
+Java and all Java-based marks are trademarks or registered trademarks
+of Sun Microsystems, Inc. in the United States and other countries.
+The Free Software Foundation is independent of Sun Microsystems, Inc.  */
+
+#ifndef JV_LEX_H
+#define JV_LEX_H
+
+/* Extern global variables declarations  */
+extern FILE *finput;
+extern int   lineno;
+extern char *classpath;
+
+/* A Unicode character, as read from the input file  */
+typedef unsigned short unicode_t;
+
+/* Function declaration  */
+static int java_lineterminator ();
+static char *java_sprint_unicode ();
+static void java_unicode_2_utf8 ();
+static void java_lex_error ();
+static void java_store_unicode ();
+
+/* Debug macro to print-out what we match  */
+#ifdef JAVA_LEX_DEBUG
+#ifdef JAVA_LEX_DEBUG_CHAR
+#define JAVA_LEX_CHAR(c)      printf ("java_lex:%d: char '%c'.%d\n",   \
+                                     lineno, (c < 128 ? c : '.'), c);
+#else
+#define JAVA_LEX_CHAR(c)
+#endif
+#define JAVA_LEX_KW(c)        printf ("java_lex:%d: keyword: '%s'\n", lineno,c)
+#define JAVA_LEX_ID(s)        printf ("java_lex:%d: ID: '%s'\n",       \
+                                     lineno,                           \
+                                     (all_ascii ? s : "<U>"))
+#define JAVA_LEX_LIT(s, r)    printf ("java_lex:%d: literal '%s'_%d\n",        \
+                                     lineno, s, r)
+#define JAVA_LEX_CHAR_LIT(s)  printf ("java_lex:%d: literal '%d'\n", lineno, s)
+#define JAVA_LEX_STR_LIT(s)   {                                                 \
+                                int i;                                  \
+                                printf ("java_lex:%d: literal '%s'\n",  \
+                                        lineno, s);                     \
+                              }
+#define JAVA_LEX_SEP(c)       printf ("java_lex:%d: separator '%c'\n",lineno,c)
+#define JAVA_LEX_OP(c)        printf ("java_lex:%d: operator '%s'\n", lineno,c)
+#else
+#define JAVA_LEX_CHAR(c)
+#define JAVA_LEX_KW(c)
+#define JAVA_LEX_ID(s)
+#define JAVA_LEX_LIT(s,r)
+#define JAVA_LEX_CHAR_LIT(s)
+#define JAVA_LEX_STR_LIT(s)
+#define JAVA_LEX_SEP(c)
+#define JAVA_LEX_OP(s)
+#endif
+
+/* Line information containers  */
+struct java_line {
+  unicode_t *line;             /* The line's unicode */
+  char      *unicode_escape_p; /* The maching char was a unicode escape */
+  unicode_t ahead[1];          /* Character ahead */
+  char unicode_escape_ahead_p; /* Character ahead is a unicode escape */
+  int max;                     /* buffer's max size */
+  int size;                    /* number of unicodes */
+  int current;                 /* Current position, unicode based */
+  int char_col;                        /* Current position, input char based */
+  int lineno;                  /* Its line number */
+  int white_space_only;                /* If it contains only white spaces */
+};
+#define JAVA_COLUMN_DELTA(p)                                           \
+  (ctxp->c_line->unicode_escape_p [ctxp->c_line->current+(p)] ? 6 :    \
+   (ctxp->c_line->line [ctxp->c_line->current+(p)] == '\t' ? 8 : 1))
+
+struct java_error {
+  struct java_line *line;
+  int error;
+};
+
+typedef struct _java_lc {
+  int line;
+  int col;
+} java_lc;
+
+
+#define JAVA_LINE_MAX 80
+
+/* Macro to read and unread chars */
+#define UNGETC(c) ctxp->unget_utf8_value = (c);
+#define GETC()    getc(finput)
+
+/* Build a location compound integer */
+#define BUILD_LOCATION() ((ctxp->elc.line << 12) | (ctxp->elc.col & 0xfff))
+
+/* Those macros are defined differently if we compile jc1-lite
+   (JC1_LITE defined) or jc1.  */
+#ifdef JC1_LITE
+
+#define DCONST0 0
+#define HOST_WIDE_INT long
+#define HOST_BITS_PER_WIDE_INT 64
+#define HOST_BITS_PER_CHAR 8
+#define REAL_VALUE_TYPE int
+#define SET_FLOAT_HANDLER(H)
+#define GET_IDENTIFIER(S) xstrdup ((S))
+#define REAL_VALUE_ATOF(LIT,MODE) 0
+#define REAL_VALUE_ISINF(VALUE)   0
+#define REAL_VALUE_ISNAN(VALUE)   0
+#define SET_REAL_VALUE_ATOF(TARGET,SOURCE)
+#define FLOAT_TYPE_NODE 0
+#define DOUBLE_TYPE_NODE 0
+#define SET_MODIFIER_CTX(TOKEN) java_lval->value = (TOKEN)
+#define GET_TYPE_PRECISION(NODE) 4
+#define BUILD_OPERATOR(TOKEN)  return TOKEN
+#define BUILD_OPERATOR2(TOKEN) return TOKEN
+#define SET_LVAL_NODE(NODE)
+#define SET_LVAL_NODE_TYPE(NODE, TYPE)
+#define BUILD_ID_WFL(EXP) (EXP)
+#define JAVA_FLOAT_RANGE_ERROR(S) {}
+#define JAVA_INTEGRAL_RANGE_ERROR(S) {}
+
+#else
+
+static tree build_wfl_node ();
+#define SET_FLOAT_HANDLER(H) set_float_handler ((H))
+#define DCONST0 dconst0
+#define GET_IDENTIFIER(S) get_identifier ((S))
+#define SET_REAL_VALUE_ATOF(TARGET,SOURCE) (TARGET) = (SOURCE)
+#define FLOAT_TYPE_NODE float_type_node
+#define DOUBLE_TYPE_NODE double_type_node
+/* Set modifier_ctx according to TOKEN */
+#define SET_MODIFIER_CTX(TOKEN)                                                   \
+  {                                                                       \
+    ctxp->modifier_ctx [(TOKEN)-PUBLIC_TK] = build_wfl_node (NULL_TREE); \
+    java_lval->value = (TOKEN)-PUBLIC_TK;                                 \
+  }
+/* Type precision for long */
+#define GET_TYPE_PRECISION(NODE) TYPE_PRECISION (long_type_node) / 8;
+/* Build an operator tree node and return TOKEN */
+#define BUILD_OPERATOR(TOKEN)                          \
+  {                                                    \
+    java_lval->operator.token = (TOKEN);               \
+    java_lval->operator.location = BUILD_LOCATION();   \
+    return (TOKEN);                                    \
+  }
+
+/* Build an operator tree node but return ASSIGN_ANY_TK */
+#define BUILD_OPERATOR2(TOKEN)                         \
+  {                                                    \
+    java_lval->operator.token = (TOKEN);               \
+    java_lval->operator.location = BUILD_LOCATION();   \
+    return ASSIGN_ANY_TK;                              \
+  }
+/* Set java_lval->node and TREE_TYPE(java_lval->node) in macros */
+#define SET_LVAL_NODE(NODE) java_lval->node = (NODE)
+#define SET_LVAL_NODE_TYPE(NODE,TYPE)          \
+  {                                            \
+    java_lval->node = (NODE);                  \
+    TREE_TYPE (java_lval->node) = (TYPE);      \
+  }
+/* Wrap identifier around a wfl */
+#define BUILD_ID_WFL(EXP) build_wfl_node ((EXP))
+/* Special ways to report error on numeric literals  */
+#define JAVA_FLOAT_RANGE_ERROR(m)                                        \
+  {                                                                      \
+    char msg [1024];                                                     \
+    int i = ctxp->c_line->current;                                       \
+    ctxp->c_line->current = number_beginning;                            \
+    sprintf (msg, "Floating pointer literal exceeds range of `%s'", (m)); \
+    java_lex_error (msg, 0);                                             \
+    ctxp->c_line->current = i;                                           \
+    value = dconst0;                                                     \
+  }
+#define JAVA_INTEGRAL_RANGE_ERROR(m)           \
+  {                                            \
+    int i = ctxp->c_line->current;             \
+    ctxp->c_line->current = number_beginning;  \
+    java_lex_error (m, 0);                     \
+    ctxp->c_line->current = i;                 \
+  }
+
+#endif /* Definitions for jc1 compilation only */
+
+/* Macros to decode character ranges */
+#define RANGE(c, l, h)           (((c) >= l && (c) <= h))
+#define JAVA_WHITE_SPACE_P(c) (c == ' ' || c == '\t' || c == '\f')
+#define JAVA_ID_CHAR_P(c)     ((c < 128 && (RANGE (c, 'A', 'Z') ||     \
+                                           RANGE (c, 'a', 'z') ||      \
+                                           RANGE (c, '0', '9') ||      \
+                                           c == '_'         ||         \
+                                           c == '$'))       ||         \
+                              (c > 127 && java_letter_or_digit_p (c)))
+#define JAVA_ASCII_DIGIT(c)    RANGE(c,'0', '9')
+#define JAVA_ASCII_OCTDIGIT(c) RANGE(c,'0', '7')
+#define JAVA_ASCII_HEXDIGIT(c) (RANGE(c,'0', '9') ||   \
+                               RANGE(c,'a', 'f') ||    \
+                               RANGE(c,'A', 'F'))
+#define JAVA_ASCII_FPCHAR(c)   (RANGE(c,'d', 'f') || RANGE(c,'D', 'F') || \
+                               c == '.' || JAVA_ASCII_DIGIT (c))
+#define JAVA_FP_SUFFIX(c)      (c == 'D' || c == 'd' || c == 'f' || c == 'F')
+#define JAVA_FP_EXP(c)         (c == 'E' || c == 'F')
+#define JAVA_FP_PM(c)          (c == '-' || c == '+')
+#define JAVA_ASCII_LETTER(c)   (RANGE(c,'a', 'z') || RANGE(c,'A', 'Z'))
+#define JAVA_DIGIT_P(c)                                                              \
+   (RANGE (c, 0x030, 0x039) || /* ISO-Latin-1 (and ASCII) digits ('0'-'9') */ \
+    RANGE (c, 0x660, 0x669) || /* Arabic-Indic digits */                     \
+    RANGE (c, 0x6F0, 0x6F9) || /* Eastern Arabic-Indic digits */             \
+    RANGE (c, 0x966, 0x96F) || /* Devanagari digits */                       \
+    RANGE (c, 0x9E6, 0x9EF) || /* Bengali digits */                          \
+    RANGE (c, 0xA66, 0xA6F) || /* Gurmukhi digits */                         \
+    RANGE (c, 0xAE6, 0xAEF) || /* Gujarati digits */                         \
+    RANGE (c, 0xB66, 0xB6F) || /* Oriya digits */                            \
+    RANGE (c, 0xBE7, 0xBEF) || /* Tamil digits */                            \
+    RANGE (c, 0xC66, 0xC6F) || /* Telugu digits */                           \
+    RANGE (c, 0xCE6, 0xCEF) || /* Kannada digits */                          \
+    RANGE (c, 0xD66, 0xD6F) || /* Malayalam digits */                        \
+    RANGE (c, 0xE50, 0xE59) || /* Thai digits */                             \
+    RANGE (c, 0xED0, 0xED9))   /* Lao digits */
+
+/* This is not to be used as a stand alone macro. Use JAVA_ID_CHAR_P()
+   or the forcoming JAVA_LETTER_OR_DIGIT_P() instead.
+   It need to be split by region. FIXME.  */
+#define _JAVA_LETTER_OR_DIGIT_P(c)             \
+   (RANGE (c, 0x00C0, 0x00D6) ||               \
+    RANGE (c, 0x00D8, 0x00F6) ||               \
+    RANGE (c, 0x00F8, 0x01F5) ||               \
+    RANGE (c, 0x01FA, 0x0217) ||               \
+    RANGE (c, 0x0250, 0x02A8) ||               \
+    RANGE (c, 0x02B0, 0x02DE) ||               \
+    RANGE (c, 0x02E0, 0x02E9) ||               \
+    RANGE (c, 0x0300, 0x0345) ||               \
+    RANGE (c, 0x0360, 0x0361) ||               \
+    RANGE (c, 0x0374, 0x0375) ||               \
+    c == 0x037A            ||                  \
+    c == 0x037E            ||                  \
+    RANGE (c, 0x0384, 0x038A) ||               \
+    c == 0x038C            ||                  \
+    c == 0x038E            ||                  \
+    RANGE (c, 0x038F, 0x03A1) ||               \
+    RANGE (c, 0x03A3, 0x03CE) ||               \
+    RANGE (c, 0x03D0, 0x03D6) ||               \
+    RANGE (c, 0x03DA, 0x03E2) ||               \
+    c == 0x03DA            ||                  \
+    c == 0x03DC            ||                  \
+    c == 0x03DE            ||                  \
+    c == 0x03E0            ||                  \
+    RANGE (c, 0x03E2, 0x03F3) ||               \
+    RANGE (c, 0x0401, 0x040C) ||               \
+    RANGE (c, 0x040E, 0x044F) ||               \
+    RANGE (c, 0x0451, 0x045C) ||               \
+    RANGE (c, 0x045E, 0x0486) ||               \
+    RANGE (c, 0x0490, 0x04C4) ||               \
+    RANGE (c, 0x04C7, 0x04C8) ||               \
+    RANGE (c, 0x04CB, 0x04CC) ||               \
+    RANGE (c, 0x04D0, 0x04EB) ||               \
+    RANGE (c, 0x04EE, 0x04F5) ||               \
+    RANGE (c, 0x04F8, 0x04F9) ||               \
+    RANGE (c, 0x0531, 0x0556) ||               \
+    RANGE (c, 0x0559, 0x055F) ||               \
+    RANGE (c, 0x0561, 0x0587) ||               \
+    c == 0x0589            ||                  \
+    RANGE (c, 0x05B0, 0x05B9) ||               \
+    RANGE (c, 0x05BB, 0x05C3) ||               \
+    RANGE (c, 0x05D0, 0x05EA) ||               \
+    RANGE (c, 0x05F0, 0x05F4) ||               \
+    c == 0x060C            ||                  \
+    c == 0x061B            ||                  \
+    c == 0x061F            ||                  \
+    c == 0x0621            ||                  \
+    RANGE (c, 0x0622, 0x063A) ||               \
+    RANGE (c, 0x0640, 0x0652) ||               \
+    RANGE (c, 0x0660, 0x066D) ||               \
+    RANGE (c, 0x0670, 0x06B7) ||               \
+    RANGE (c, 0x06BA, 0x06BE) ||               \
+    RANGE (c, 0x06C0, 0x06CE) ||               \
+    RANGE (c, 0x06D0, 0x06ED) ||               \
+    RANGE (c, 0x06F0, 0x06F9) ||               \
+    RANGE (c, 0x0901, 0x0903) ||               \
+    RANGE (c, 0x0905, 0x0939) ||               \
+    RANGE (c, 0x093C, 0x094D) ||               \
+    RANGE (c, 0x0950, 0x0954) ||               \
+    RANGE (c, 0x0958, 0x0970) ||               \
+    RANGE (c, 0x0981, 0x0983) ||               \
+    RANGE (c, 0x0985, 0x098C) ||               \
+    RANGE (c, 0x098F, 0x0990) ||               \
+    RANGE (c, 0x0993, 0x09A8) ||               \
+    RANGE (c, 0x09AA, 0x09B0) ||               \
+    c == 0x09B2            ||                  \
+    RANGE (c, 0x09B6, 0x09B9) ||               \
+    c == 0x09BC            ||                  \
+    c == 0x09BE            ||                  \
+    RANGE (c, 0x09BF, 0x09C4) ||               \
+    RANGE (c, 0x09C7, 0x09C8) ||               \
+    RANGE (c, 0x09CB, 0x09CD) ||               \
+    c == 0x09D7            ||                  \
+    RANGE (c, 0x09DC, 0x09DD) ||               \
+    RANGE (c, 0x09DF, 0x09E3) ||               \
+    RANGE (c, 0x09E6, 0x09FA) ||               \
+    c == 0x0A02            ||                  \
+    RANGE (c, 0x0A05, 0x0A0A) ||               \
+    RANGE (c, 0x0A0F, 0x0A10) ||               \
+    RANGE (c, 0x0A13, 0x0A28) ||               \
+    RANGE (c, 0x0A2A, 0x0A30) ||               \
+    RANGE (c, 0x0A32, 0x0A33) ||               \
+    RANGE (c, 0x0A35, 0x0A36) ||               \
+    RANGE (c, 0x0A38, 0x0A39) ||               \
+    c == 0x0A3C            ||                  \
+    c == 0x0A3E            ||                  \
+    RANGE (c, 0x0A3F, 0x0A42) ||               \
+    RANGE (c, 0x0A47, 0x0A48) ||               \
+    RANGE (c, 0x0A4B, 0x0A4D) ||               \
+    RANGE (c, 0x0A59, 0x0A5C) ||               \
+    c == 0x0A5E            ||                  \
+    RANGE (c, 0x0A66, 0x0A74) ||               \
+    RANGE (c, 0x0A81, 0x0A83) ||               \
+    RANGE (c, 0x0A85, 0x0A8B) ||               \
+    c == 0x0A8D            ||                  \
+    c == 0x0A8F            ||                  \
+    RANGE (c, 0x0A90, 0x0A91) ||               \
+    RANGE (c, 0x0A93, 0x0AA8) ||               \
+    RANGE (c, 0x0AAA, 0x0AB0) ||               \
+    RANGE (c, 0x0AB2, 0x0AB3) ||               \
+    RANGE (c, 0x0AB5, 0x0AB9) ||               \
+    RANGE (c, 0x0ABC, 0x0AC5) ||               \
+    RANGE (c, 0x0AC7, 0x0AC9) ||               \
+    RANGE (c, 0x0ACB, 0x0ACD) ||               \
+    c == 0x0AD0            ||                  \
+    c == 0x0AE0            ||                  \
+    RANGE (c, 0x0AE6, 0x0AEF) ||               \
+    RANGE (c, 0x0B01, 0x0B03) ||               \
+    RANGE (c, 0x0B05, 0x0B0C) ||               \
+    RANGE (c, 0x0B0F, 0x0B10) ||               \
+    RANGE (c, 0x0B13, 0x0B28) ||               \
+    RANGE (c, 0x0B2A, 0x0B30) ||               \
+    RANGE (c, 0x0B32, 0x0B33) ||               \
+    RANGE (c, 0x0B36, 0x0B39) ||               \
+    RANGE (c, 0x0B3C, 0x0B43) ||               \
+    RANGE (c, 0x0B47, 0x0B48) ||               \
+    RANGE (c, 0x0B4B, 0x0B4D) ||               \
+    RANGE (c, 0x0B56, 0x0B57) ||               \
+    RANGE (c, 0x0B5C, 0x0B5D) ||               \
+    RANGE (c, 0x0B5F, 0x0B61) ||               \
+    RANGE (c, 0x0B66, 0x0B70) ||               \
+    RANGE (c, 0x0B82, 0x0B83) ||               \
+    RANGE (c, 0x0B85, 0x0B8A) ||               \
+    RANGE (c, 0x0B8E, 0x0B90) ||               \
+    RANGE (c, 0x0B92, 0x0B95) ||               \
+    RANGE (c, 0x0B99, 0x0B9A) ||               \
+    c == 0x0B9C            ||                  \
+    c == 0x0B9E            ||                  \
+    c == 0x0B9F            ||                  \
+    RANGE (c, 0x0BA3, 0x0BA4) ||               \
+    RANGE (c, 0x0BA8, 0x0BAA) ||               \
+    RANGE (c, 0x0BAE, 0x0BB5) ||               \
+    RANGE (c, 0x0BB7, 0x0BB9) ||               \
+    RANGE (c, 0x0BBE, 0x0BC2) ||               \
+    RANGE (c, 0x0BC6, 0x0BC8) ||               \
+    RANGE (c, 0x0BCA, 0x0BCD) ||               \
+    c == 0x0BD7            ||                  \
+    RANGE (c, 0x0BE7, 0x0BF2) ||               \
+    RANGE (c, 0x0C01, 0x0C03) ||               \
+    RANGE (c, 0x0C05, 0x0C0C) ||               \
+    RANGE (c, 0x0C0E, 0x0C10) ||               \
+    RANGE (c, 0x0C12, 0x0C28) ||               \
+    RANGE (c, 0x0C2A, 0x0C33) ||               \
+    RANGE (c, 0x0C35, 0x0C39) ||               \
+    RANGE (c, 0x0C3E, 0x0C44) ||               \
+    RANGE (c, 0x0C46, 0x0C48) ||               \
+    RANGE (c, 0x0C4A, 0x0C4D) ||               \
+    RANGE (c, 0x0C55, 0x0C56) ||               \
+    RANGE (c, 0x0C60, 0x0C61) ||               \
+    RANGE (c, 0x0C66, 0x0C6F) ||               \
+    RANGE (c, 0x0C82, 0x0C83) ||               \
+    RANGE (c, 0x0C85, 0x0C8C) ||               \
+    RANGE (c, 0x0C8E, 0x0C90) ||               \
+    RANGE (c, 0x0C92, 0x0CA8) ||               \
+    RANGE (c, 0x0CAA, 0x0CB3) ||               \
+    RANGE (c, 0x0CB5, 0x0CB9) ||               \
+    RANGE (c, 0x0CBE, 0x0CC4) ||               \
+    RANGE (c, 0x0CC6, 0x0CC8) ||               \
+    RANGE (c, 0x0CCA, 0x0CCD) ||               \
+    RANGE (c, 0x0CD5, 0x0CD6) ||               \
+    c == 0x0CDE            ||                  \
+    c == 0x0CE0            ||                  \
+    c == 0x0CE1            ||                  \
+    RANGE (c, 0x0CE6, 0x0CEF) ||               \
+    RANGE (c, 0x0D02, 0x0D03) ||               \
+    RANGE (c, 0x0D05, 0x0D0C) ||               \
+    RANGE (c, 0x0D0E, 0x0D10) ||               \
+    RANGE (c, 0x0D12, 0x0D28) ||               \
+    RANGE (c, 0x0D2A, 0x0D39) ||               \
+    RANGE (c, 0x0D3E, 0x0D43) ||               \
+    RANGE (c, 0x0D46, 0x0D48) ||               \
+    RANGE (c, 0x0D4A, 0x0D4D) ||               \
+    c == 0x0D57            ||                  \
+    RANGE (c, 0x0D60, 0x0D61) ||               \
+    RANGE (c, 0x0D66, 0x0D6F) ||               \
+    RANGE (c, 0x0E01, 0x0E3A) ||               \
+    RANGE (c, 0x0E3F, 0x0E5B) ||               \
+    RANGE (c, 0x0E81, 0x0E82) ||               \
+    c == 0x0E84            ||                  \
+    RANGE (c, 0x0E87, 0x0E88) ||               \
+    c == 0x0E8A            ||                  \
+    c == 0x0E8D            ||                  \
+    RANGE (c, 0x0E94, 0x0E97) ||               \
+    RANGE (c, 0x0E99, 0x0E9F) ||               \
+    RANGE (c, 0x0EA1, 0x0EA3) ||               \
+    c == 0x0EA5            ||                  \
+    c == 0x0EA7            ||                  \
+    RANGE (c, 0x0EAA, 0x0EAB) ||               \
+    RANGE (c, 0x0EAD, 0x0EB9) ||               \
+    RANGE (c, 0x0EBB, 0x0EBD) ||               \
+    RANGE (c, 0x0EC0, 0x0EC4) ||               \
+    c == 0x0EC6            ||                  \
+    c == 0x0EC8            ||                  \
+    RANGE (c, 0x0EC9, 0x0ECD) ||               \
+    RANGE (c, 0x0ED0, 0x0ED9) ||               \
+    RANGE (c, 0x0EDC, 0x0EDD) ||               \
+    RANGE (c, 0x10A0, 0x10C5) ||               \
+    RANGE (c, 0x10D0, 0x10F6) ||               \
+    c == 0x10FB            ||                  \
+    RANGE (c, 0x1100, 0x1159) ||               \
+    RANGE (c, 0x115F, 0x11A2) ||               \
+    RANGE (c, 0x11A8, 0x11F9) ||               \
+    RANGE (c, 0x1E00, 0x1E9A) ||               \
+    RANGE (c, 0x1EA0, 0x1EF9) ||               \
+    RANGE (c, 0x1F00, 0x1F15) ||               \
+    RANGE (c, 0x1F18, 0x1F1D) ||               \
+    RANGE (c, 0x1F20, 0x1F45) ||               \
+    RANGE (c, 0x1F48, 0x1F4D) ||               \
+    RANGE (c, 0x1F50, 0x1F57) ||               \
+    c == 0x1F59            ||                  \
+    c == 0x1F5B            ||                  \
+    c == 0x1F5D            ||                  \
+    RANGE (c, 0x1F5F, 0x1F7D) ||               \
+    RANGE (c, 0x1F80, 0x1FB4) ||               \
+    RANGE (c, 0x1FB6, 0x1FC4) ||               \
+    RANGE (c, 0x1FC6, 0x1FD3) ||               \
+    RANGE (c, 0x1FD6, 0x1FDB) ||               \
+    RANGE (c, 0x1FDD, 0x1FEF) ||               \
+    RANGE (c, 0x1FF2, 0x1FF4) ||               \
+    RANGE (c, 0x1FF6, 0x1FFE) ||               \
+    RANGE (c, 0x3041, 0x3094) ||               \
+    RANGE (c, 0x3099, 0x309E) ||               \
+    RANGE (c, 0x30A1, 0x30FE) ||               \
+    RANGE (c, 0x3105, 0x312C) ||               \
+    RANGE (c, 0x3131, 0x318E) ||               \
+    RANGE (c, 0x3190, 0x319F) ||               \
+    RANGE (c, 0x3200, 0x321C) ||               \
+    RANGE (c, 0x3220, 0x3243) ||               \
+    RANGE (c, 0x3260, 0x327B) ||               \
+    RANGE (c, 0x327F, 0x32B0) ||               \
+    RANGE (c, 0x32C0, 0x32CB) ||               \
+    RANGE (c, 0x32D0, 0x32FE) ||               \
+    RANGE (c, 0x3300, 0x3376) ||               \
+    RANGE (c, 0x337B, 0x33DD) ||               \
+    RANGE (c, 0x33E0, 0x33FE) ||               \
+    RANGE (c, 0x3400, 0x9FA5) ||               \
+    RANGE (c, 0xF900, 0xFA2D) ||               \
+    RANGE (c, 0xFB00, 0xFB06) ||               \
+    RANGE (c, 0xFB13, 0xFB17) ||               \
+    RANGE (c, 0xFB1E, 0xFB36) ||               \
+    RANGE (c, 0xFB38, 0xFB3C) ||               \
+    c == 0xFB3E            ||                  \
+    c == 0xFB40            ||                  \
+    c == 0xFB41            ||                  \
+    c == 0xFB43            ||                  \
+    c == 0xFB44            ||                  \
+    c == 0xFB46            ||                  \
+    RANGE (c, 0xFB47, 0xFBB1) ||               \
+    RANGE (c, 0xFBD3, 0xFD3F) ||               \
+    RANGE (c, 0xFD50, 0xFD8F) ||               \
+    RANGE (c, 0xFD92, 0xFDC7) ||               \
+    RANGE (c, 0xFDF0, 0xFDFB) ||               \
+    RANGE (c, 0xFE70, 0xFE72) ||               \
+    c == 0xFE74            ||                  \
+    c == 0xFE76            ||                  \
+    RANGE (c, 0xFE77, 0xFEFC) ||               \
+    RANGE (c, 0xFF10, 0xFF19) ||               \
+    RANGE (c, 0xFF21, 0xFF3A) ||               \
+    RANGE (c, 0xFF41, 0xFF5A) ||               \
+    RANGE (c, 0xFF66, 0xFFBE) ||               \
+    RANGE (c, 0xFFC2, 0xFFC7) ||               \
+    RANGE (c, 0xFFCA, 0xFFCF) ||               \
+    RANGE (c, 0xFFD2, 0xFFD7) ||               \
+    RANGE (c, 0xFFDA, 0xFFDC))
+
+/* Constants  */
+#define JAVA_CHAR_ERROR 0xFFC1 /* This is an illegal unicode!?! FIXME */
+#define JAVA_READ_BUFFER 256
+#define UEOF (unicode_t)0xffff
+
+#endif
diff --git a/gcc/java/mangle.c b/gcc/java/mangle.c
new file mode 100644 (file)
index 0000000..f303c92
--- /dev/null
@@ -0,0 +1,163 @@
+/* Functions related to mangling class names for the GNU compiler
+   for the Java(TM) language.
+   Copyright (C) 1998 Free Software Foundation, Inc.
+
+This file is part of GNU CC.
+
+GNU CC is free software; you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation; either version 2, or (at your option)
+any later version.
+
+GNU CC is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with GNU CC; see the file COPYING.  If not, write to
+the Free Software Foundation, 59 Temple Place - Suite 330,
+Boston, MA 02111-1307, USA. 
+
+Java and all Java-based marks are trademarks or registered trademarks
+of Sun Microsystems, Inc. in the United States and other countries.
+The Free Software Foundation is independent of Sun Microsystems, Inc.  */
+
+/* Written by Per Bothner <bothner@cygnus.com> */
+
+#include <stdio.h>
+#include <string.h>
+#include "config.h"
+#include "jcf.h"
+#include "obstack.h"
+
+/* Assuming (NAME, LEN) is a Utf8-encoding string, calculate
+   the length of the string as mangled (a la g++) including Unicode escapes.
+   If no escapes are needed, return 0. */
+
+int
+unicode_mangling_length (name, len)
+     char *name; 
+     int len; 
+{
+  unsigned char *ptr;
+  unsigned char *limit = (unsigned char *)name + len;
+  int need_escapes = 0;
+  int num_chars = 0;
+  int underscores = 0;
+  for (ptr = (unsigned char *) name;  ptr < limit;  )
+    {
+      int ch = UTF8_GET(ptr, limit);
+      if (ch < 0)
+       error ("internal error - invalid Utf8 name");
+      if (ch >= '0' && ch <= '9')
+       need_escapes += num_chars == 0;
+      else if (ch == '_')
+       underscores++;
+      else if ((ch < 'a' || ch > 'z') && (ch < 'A' || ch > 'Z'))
+       need_escapes++;
+      num_chars++;
+    }
+  if (need_escapes)
+    return num_chars + 4 * (need_escapes + underscores);
+  else
+    return 0;
+}
+
+/* Assuming (NAME, LEN) is a Utf8-encoding string, emit the string
+   appropriately mangled (with Unicode escapes) to OBSTACK. */
+
+void
+emit_unicode_mangled_name (obstack, name, len)
+     struct obstack *obstack;
+     char *name;
+{
+  unsigned char *ptr;
+  unsigned char *limit = (unsigned char *)name + len;
+  for (ptr = (unsigned char *) name;  ptr < limit;  )
+    {
+      int ch = UTF8_GET(ptr, limit);
+      int emit_escape;
+      if (ch < 0)
+       {
+         error ("internal error - bad Utf8 string");
+         break;
+       }
+      if (ch >= '0' && ch <= '9')
+       emit_escape = (ptr == (unsigned char*) name);
+      else
+       emit_escape = (ch < 'a' || ch > 'z') && (ch < 'A' || ch > 'Z');
+      if (emit_escape)
+       {
+         char buf[6];
+         sprintf (buf, "_%04x", ch);
+         obstack_grow (obstack, buf, 5);
+       }
+      else
+       {
+         obstack_1grow (obstack, ch);
+       }
+    }
+}
+
+/* Assuming (NAME, LEN) is a Utf8-encoding string, emit the string
+   appropriately mangled (with Unicode escapes if needed) to OBSTACK. */
+
+void
+append_gpp_mangled_name (obstack, name, len)
+     struct obstack *obstack;
+     char *name;
+     int len;
+{
+  int encoded_len = unicode_mangling_length (name, len);
+  int needs_escapes = encoded_len > 0;
+  char buf[6];
+  if (needs_escapes)
+    {
+      sprintf (buf, "U%d", encoded_len);
+      obstack_grow (obstack, buf, strlen(buf));
+      emit_unicode_mangled_name (obstack, name, len);
+    }
+  else
+    {
+      sprintf (buf, "%d", len);
+      obstack_grow (obstack, buf, strlen(buf));
+      obstack_grow (obstack, name, len);
+    }
+}
+
+/* Append the mangled name of a class named CLASSNAME onto OBSTACK. */
+
+void
+append_gpp_mangled_classtype (obstack, class_name)
+     struct obstack *obstack;
+     char *class_name;
+{
+  char *ptr;
+  int qualifications = 0;
+
+  for (ptr = class_name; *ptr != '\0'; ptr++)
+    {
+      if (*ptr == '.')
+       qualifications++;
+    }
+  if (qualifications)
+    {
+      char buf[8];
+      if (qualifications >= 9)
+       sprintf (buf, "Q_%d_", qualifications + 1);
+      else
+       sprintf (buf, "Q%d", qualifications + 1);
+      obstack_grow (obstack, buf, strlen (buf));
+    }
+  for (ptr = class_name; ; ptr++)
+    {
+      if (ptr[0] == '.' || ptr[0] == '\0')
+       {
+         append_gpp_mangled_name (obstack, class_name, ptr - class_name);
+         if (ptr[0] == '\0')
+           break;
+         class_name = ptr + 1;
+       }
+    }
+}
diff --git a/gcc/java/parse-scan.y b/gcc/java/parse-scan.y
new file mode 100644 (file)
index 0000000..11c9978
--- /dev/null
@@ -0,0 +1,1162 @@
+/* Parser grammar for quick source code scan of Java(TM) language programs.
+   Copyright (C) 1998 Free Software Foundation, Inc.
+   Contributed by Alexandre Petit-Bianco (apbianco@cygnus.com)
+
+This file is part of GNU CC.
+
+GNU CC is free software; you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation; either version 2, or (at your option)
+any later version.
+
+GNU CC is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with GNU CC; see the file COPYING.  If not, write to
+the Free Software Foundation, 59 Temple Place - Suite 330,
+Boston, MA 02111-1307, USA.
+
+Java and all Java-based marks are trademarks or registered trademarks
+of Sun Microsystems, Inc. in the United States and other countries.
+The Free Software Foundation is independent of Sun Microsystems, Inc.  */
+
+/* This file parses Java source code. Action can be further completed
+to achieve a desired behavior. This file isn't part of the Java
+language gcc front end.
+
+The grammar conforms to the Java grammar described in "The Java(TM)
+Language Specification. J. Gosling, B. Joy, G. Steele. Addison Wesley
+1996, ISBN 0-201-63451-1"
+
+Some rules have been modified to support JDK1.1 inner classes
+definitions and other extensions.  */
+
+%{
+#define JC1_LITE
+
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+
+/* Definitions for PROTO and VPROTO macros */
+#include "gansidecl.h"
+#include "obstack.h"
+
+extern char *input_filename;
+extern FILE *finput, *out;
+
+/* Obstack for the lexer.  */
+struct obstack temporary_obstack;
+
+/* The current parser context.  */
+static struct parser_ctxt *ctxp;
+
+/* Error and warning counts, current line number, because they're used
+   elsewhere  */
+int java_error_count;
+int java_warning_count;
+int lineno;
+
+/* Tweak default rules when necessary.  */
+static int absorber;
+#define USE_ABSORBER absorber = 0
+
+/* Keep track of the current class name and package name.  */
+static char *current_class;
+static char *package_name;
+
+/* Keep track of whether things have be listed before.  */
+static int previous_output;
+
+/* Record modifier uses  */
+static int modifier_value;
+
+/* Record a method declaration  */
+struct method_declarator {
+  char *method_name;
+  char *args;
+};
+#define NEW_METHOD_DECLARATOR(D,N,A)                                        \
+{                                                                           \
+  (D) =                                                                     \
+    (struct method_declarator *)xmalloc (sizeof (struct method_declarator)); \
+  (D)->method_name = (N);                                                   \
+  (D)->args = (A);                                                          \
+}
+
+/* Two actions for this grammar */
+static void report_class_declaration PROTO ((char *));
+static void report_main_declaration PROTO ((struct method_declarator *));
+
+/* Other extern functions */
+char *xmalloc PROTO ((unsigned));
+char *xstrdup PROTO ((char *));
+
+#include "lex.h"
+#include "parse.h"
+%}
+
+%union {
+  char *node;
+  struct method_declarator *declarator;
+  int value;                   /* For modifiers */
+}
+
+%pure_parser
+
+/* Things defined here have to match the order of what's in the
+   binop_lookup table.  */
+
+%token   PLUS_TK         MINUS_TK        MULT_TK         DIV_TK    REM_TK
+%token   LS_TK           SRS_TK          ZRS_TK
+%token   AND_TK          XOR_TK          OR_TK
+%token   BOOL_AND_TK BOOL_OR_TK 
+%token   EQ_TK NEQ_TK GT_TK GTE_TK LT_TK LTE_TK
+
+/* This maps to the same binop_lookup entry than the token above */
+
+%token   PLUS_ASSIGN_TK  MINUS_ASSIGN_TK MULT_ASSIGN_TK DIV_ASSIGN_TK
+%token   REM_ASSIGN_TK   
+%token   LS_ASSIGN_TK    SRS_ASSIGN_TK   ZRS_ASSIGN_TK
+%token   AND_ASSIGN_TK   XOR_ASSIGN_TK   OR_ASSIGN_TK
+
+
+/* Modifier TOKEN have to be kept in this order. Don't scramble it */
+
+%token   PUBLIC_TK       PRIVATE_TK         PROTECTED_TK
+%token   STATIC_TK       FINAL_TK           SYNCHRONIZED_TK
+%token   VOLATILE_TK     TRANSIENT_TK       NATIVE_TK
+%token   PAD_TK          ABSTRACT_TK        MODIFIER_TK
+
+/* Keep those two in order, too */
+%token   DECR_TK INCR_TK
+
+/* From now one, things can be in any order */
+
+%token   DEFAULT_TK      IF_TK              THROW_TK
+%token   BOOLEAN_TK      DO_TK              IMPLEMENTS_TK
+%token   THROWS_TK       BREAK_TK           IMPORT_TK       
+%token   ELSE_TK         INSTANCEOF_TK      RETURN_TK
+%token   VOID_TK         CATCH_TK           INTERFACE_TK
+%token   CASE_TK         EXTENDS_TK         FINALLY_TK
+%token   SUPER_TK        WHILE_TK           CLASS_TK
+%token   SWITCH_TK       CONST_TK           TRY_TK
+%token   FOR_TK          NEW_TK             CONTINUE_TK
+%token   GOTO_TK         PACKAGE_TK         THIS_TK
+
+%token   BYTE_TK         SHORT_TK           INT_TK            LONG_TK
+%token   CHAR_TK         INTEGRAL_TK
+
+%token   FLOAT_TK        DOUBLE_TK          FP_TK
+
+%token   ID_TK
+
+%token   REL_QM_TK         REL_CL_TK NOT_TK  NEG_TK
+
+%token   ASSIGN_ANY_TK   ASSIGN_TK
+%token   OP_TK  CP_TK  OCB_TK  CCB_TK  OSB_TK  CSB_TK  SC_TK  C_TK DOT_TK
+
+%token   STRING_LIT_TK   CHAR_LIT_TK        INT_LIT_TK        FP_LIT_TK
+%token   TRUE_TK         FALSE_TK           BOOL_LIT_TK       NULL_TK
+
+%type <node> ID_TK identifier name simple_name qualified_name type
+            primitive_type reference_type array_type formal_parameter_list
+            formal_parameter class_or_interface_type class_type interface_type
+%type <declarator> method_declarator
+%type <value>      MODIFIER_TK
+
+%%
+/* 19.2 Production from 2.3: The Syntactic Grammar  */
+goal:
+       compilation_unit
+;
+
+/* 19.3 Productions from 3: Lexical structure  */
+literal:
+       INT_LIT_TK
+|      FP_LIT_TK
+|      BOOL_LIT_TK
+|      CHAR_LIT_TK
+|       STRING_LIT_TK
+|       NULL_TK
+;
+
+/* 19.4 Productions from 4: Types, Values and Variables  */
+type:
+       primitive_type
+|      reference_type
+;
+
+primitive_type:
+       INTEGRAL_TK
+               {
+                 /* use preset global here. FIXME */
+                 $$ = xstrdup ("int");
+               }
+|      FP_TK
+               {
+                 /* use preset global here. FIXME */
+                 $$ = xstrdup ("double");
+               }
+|      BOOLEAN_TK
+               {
+                 /* use preset global here. FIXME */
+                 $$ = xstrdup ("boolean");
+               }
+;
+
+reference_type:
+       class_or_interface_type
+|      array_type
+;
+
+class_or_interface_type:
+       name
+;
+
+class_type:
+       class_or_interface_type /* Default rule */
+;
+
+interface_type:
+        class_or_interface_type
+;
+
+array_type:
+       primitive_type OSB_TK CSB_TK
+|      name OSB_TK CSB_TK
+               {
+                 char *n = xmalloc (strlen ($1)+2);
+                 n [0] = '[';
+                 strcpy (n+1, $1);
+                 $$ = n;
+               }
+|      array_type OSB_TK CSB_TK
+               {       
+                 char *n = xmalloc (strlen ($1)+2);
+                 n [0] = '[';
+                 strcpy (n+1, $1);
+                 $$ = n;
+               }
+;
+
+/* 19.5 Productions from 6: Names  */
+name:
+       simple_name             /* Default rule */
+|      qualified_name          /* Default rule */
+;
+
+simple_name:
+       identifier              /* Default rule */
+;
+
+qualified_name:
+       name DOT_TK identifier
+               { 
+                 char *n = xmalloc (strlen ($1)+strlen ($3)+2);
+                 sprintf (n, "%s.s", $1, $3);
+                 $$ = n;
+               }
+;
+
+identifier:
+       ID_TK
+;
+
+/* 19.6: Production from 7: Packages  */
+compilation_unit:
+|      package_declaration
+|      import_declarations
+|      type_declarations
+|       package_declaration import_declarations
+|       package_declaration type_declarations
+|       import_declarations type_declarations
+|       package_declaration import_declarations type_declarations
+;
+
+import_declarations:
+       import_declaration
+|      import_declarations import_declaration
+;
+
+type_declarations:
+       type_declaration
+|      type_declarations type_declaration
+;
+
+package_declaration:
+       PACKAGE_TK name SC_TK
+               { package_name = $2; }
+;
+
+import_declaration:
+       single_type_import_declaration
+|      type_import_on_demand_declaration
+;
+
+single_type_import_declaration:
+       IMPORT_TK name SC_TK
+;
+
+type_import_on_demand_declaration:
+       IMPORT_TK name DOT_TK MULT_TK SC_TK
+;
+
+type_declaration:
+       class_declaration
+|      interface_declaration
+|      SC_TK
+;
+
+/* 19.7 Shortened from the original:
+   modifiers: modifier | modifiers modifier
+   modifier: any of public...  */
+modifiers:
+       MODIFIER_TK
+               { 
+                 if ($1 == PUBLIC_TK)
+                   modifier_value++;
+                  if ($1 == STATIC_TK)
+                    modifier_value++;
+                 USE_ABSORBER;
+               }       
+|      modifiers MODIFIER_TK
+               { 
+                 if ($2 == PUBLIC_TK)
+                   modifier_value++;
+                  if ($2 == STATIC_TK)
+                    modifier_value++;
+                 USE_ABSORBER;
+               }       
+;
+
+/* 19.8.1 Production from $8.1: Class Declaration */
+class_declaration:
+       modifiers CLASS_TK identifier super interfaces 
+               { 
+                 report_class_declaration($3);
+                 modifier_value = 0;
+                }
+       class_body
+|      CLASS_TK identifier super interfaces 
+               { report_class_declaration($2); }
+       class_body
+;
+
+super:
+|      EXTENDS_TK class_type
+;
+
+interfaces:
+|      IMPLEMENTS_TK interface_type_list
+;
+
+interface_type_list:
+       interface_type
+               { USE_ABSORBER; }
+|      interface_type_list C_TK interface_type
+               { USE_ABSORBER; }
+;
+
+class_body:
+       OCB_TK CCB_TK
+|      OCB_TK class_body_declarations CCB_TK
+;
+
+class_body_declarations:
+       class_body_declaration
+|      class_body_declarations class_body_declaration
+;
+
+class_body_declaration:
+       class_member_declaration
+|      static_initializer
+|      constructor_declaration
+|      block                   /* Added, JDK1.1, instance initializer */
+;
+
+class_member_declaration:
+       field_declaration
+|      method_declaration
+|      class_declaration       /* Added, JDK1.1 inner classes */
+|      interface_declaration   /* Added, JDK1.1 inner classes */
+;
+
+/* 19.8.2 Productions from 8.3: Field Declarations  */
+field_declaration:
+       type variable_declarators SC_TK
+               { USE_ABSORBER; }
+|      modifiers type variable_declarators SC_TK
+               { modifier_value = 0; }
+;
+
+variable_declarators:
+       /* Should we use build_decl_list () instead ? FIXME */
+       variable_declarator     /* Default rule */
+|      variable_declarators C_TK variable_declarator
+;
+
+variable_declarator:
+       variable_declarator_id
+|      variable_declarator_id ASSIGN_TK variable_initializer
+;
+
+variable_declarator_id:
+       identifier
+               { USE_ABSORBER; }
+|      variable_declarator_id OSB_TK CSB_TK
+;
+
+variable_initializer:
+       expression
+|      array_initializer
+;
+
+/* 19.8.3 Productions from 8.4: Method Declarations  */
+method_declaration:
+       method_header method_body
+;
+
+method_header: 
+       type method_declarator throws
+               { USE_ABSORBER; }
+|      VOID_TK method_declarator throws
+|      modifiers type method_declarator throws
+               { modifier_value = 0; }
+|      modifiers VOID_TK method_declarator throws
+               { 
+                  report_main_declaration ($3);
+                 modifier_value = 0;
+               }
+;
+
+method_declarator:
+       identifier OP_TK CP_TK
+               { 
+                 struct method_declarator *d;
+                 NEW_METHOD_DECLARATOR (d, $1, NULL);
+                 $$ = d;
+               }
+|      identifier OP_TK formal_parameter_list CP_TK
+               { 
+                 struct method_declarator *d;
+                 NEW_METHOD_DECLARATOR (d, $1, $3);
+                 $$ = d;
+               }
+|      method_declarator OSB_TK CSB_TK
+;
+
+formal_parameter_list:
+       formal_parameter
+|      formal_parameter_list C_TK formal_parameter
+               {
+                 char *n = xmalloc (strlen ($1)+strlen($3)+2);
+                 sprintf (n, "%s,%s", $1, $3);
+                 $$ = n;
+               }
+;
+
+formal_parameter:
+       type variable_declarator_id
+               { 
+                 USE_ABSORBER;
+                 $$ = $1;
+               }
+|      modifiers type variable_declarator_id /* Added, JDK1.1 final locals */
+               { $$ = $2; }
+;
+
+throws:
+|      THROWS_TK class_type_list
+;
+
+class_type_list:
+       class_type
+               { USE_ABSORBER; }
+|      class_type_list C_TK class_type
+               { USE_ABSORBER; }
+;
+
+method_body:
+       block
+|      block SC_TK
+|      SC_TK
+;
+
+/* 19.8.4 Productions from 8.5: Static Initializers  */
+static_initializer:
+       static block
+|      static block SC_TK      /* Shouldn't be here. FIXME */
+;
+
+static:                                /* Test lval.sub_token here */
+       MODIFIER_TK
+               { USE_ABSORBER; }
+;
+
+/* 19.8.5 Productions from 8.6: Constructor Declarations  */
+/* NOTE FOR FURTHER WORK ON CONSTRUCTORS:
+   - If a forbidded modifier is found, the the error is either the use of
+     a forbidded modifier for a constructor OR bogus attempt to declare a
+     method without having specified the return type. FIXME */
+constructor_declaration:
+       constructor_declarator throws constructor_body
+|      modifiers constructor_declarator throws constructor_body
+               { modifier_value = 0; }
+/* extra SC_TK, FIXME */
+|      constructor_declarator throws constructor_body SC_TK
+/* extra SC_TK, FIXME */
+|      modifiers constructor_declarator throws constructor_body SC_TK
+               { modifier_value = 0; }
+/* I'm not happy with the SC_TK addition. It isn't in the grammer and should
+   probably be matched by and empty statement. But it doesn't work. FIXME */
+;
+
+constructor_declarator:
+       simple_name OP_TK CP_TK
+               { USE_ABSORBER; }
+|      simple_name OP_TK formal_parameter_list CP_TK
+               { USE_ABSORBER; }
+;
+
+constructor_body:
+       OCB_TK CCB_TK
+|      OCB_TK explicit_constructor_invocation CCB_TK
+|      OCB_TK block_statements CCB_TK
+|       OCB_TK explicit_constructor_invocation block_statements CCB_TK
+;
+
+/* Error recovery for that rule moved down expression_statement: rule.  */
+explicit_constructor_invocation:
+       this_or_super OP_TK CP_TK SC_TK
+|      this_or_super OP_TK argument_list CP_TK SC_TK
+        /* Added, JDK1.1 inner classes. Modified because the rule
+          'primary' couldn't work.  */
+|      name DOT_TK SUPER_TK OP_TK argument_list CP_TK SC_TK
+               { USE_ABSORBER; }
+|      name DOT_TK SUPER_TK OP_TK CP_TK SC_TK
+               { USE_ABSORBER; }
+;
+
+this_or_super:                 /* Added, simplifies error diagnostics */
+       THIS_TK
+|      SUPER_TK
+;
+
+/* 19.9 Productions from 9: Interfaces  */
+/* 19.9.1 Productions from 9.1: Interfaces Declarations  */
+interface_declaration:
+       INTERFACE_TK identifier interface_body
+|      modifiers INTERFACE_TK identifier interface_body
+               { modifier_value = 0; }
+|      INTERFACE_TK identifier extends_interfaces interface_body
+|      modifiers INTERFACE_TK identifier extends_interfaces interface_body
+               { modifier_value = 0; }
+;
+
+extends_interfaces:
+       EXTENDS_TK interface_type
+|      extends_interfaces C_TK interface_type
+;
+
+interface_body:
+       OCB_TK CCB_TK
+|      OCB_TK interface_member_declarations CCB_TK
+;
+
+interface_member_declarations:
+       interface_member_declaration
+|      interface_member_declarations interface_member_declaration
+;
+
+interface_member_declaration:
+       constant_declaration
+|      abstract_method_declaration
+|      class_declaration       /* Added, JDK1.1 inner classes */
+|      interface_declaration   /* Added, JDK1.1 inner classes */
+;
+
+constant_declaration:
+       field_declaration
+;
+
+abstract_method_declaration:
+       method_header SC_TK
+;
+
+/* 19.10 Productions from 10: Arrays  */
+array_initializer:
+       OCB_TK CCB_TK
+|      OCB_TK variable_initializers CCB_TK
+|      OCB_TK C_TK CCB_TK
+|      OCB_TK variable_initializers C_TK CCB_TK
+;
+
+variable_initializers:
+       variable_initializer
+|      variable_initializers C_TK variable_initializer
+;
+
+/* 19.11 Production from 14: Blocks and Statements  */
+block:
+       OCB_TK CCB_TK
+|      OCB_TK block_statements CCB_TK
+;
+
+block_statements:
+       block_statement
+|      block_statements block_statement
+;
+
+block_statement:
+       local_variable_declaration_statement
+|      statement
+|      class_declaration       /* Added, JDK1.1 inner classes */
+;
+
+local_variable_declaration_statement:
+       local_variable_declaration SC_TK /* Can't catch missing ';' here */
+;
+
+local_variable_declaration:
+       type variable_declarators
+               { USE_ABSORBER; }
+|      modifiers type variable_declarators /* Added, JDK1.1 final locals */
+               { modifier_value = 0; }
+;
+
+statement:
+       statement_without_trailing_substatement
+|      labeled_statement
+|      if_then_statement
+|      if_then_else_statement
+|      while_statement
+|      for_statement
+;
+
+statement_nsi:
+       statement_without_trailing_substatement
+|      labeled_statement_nsi
+|      if_then_else_statement_nsi
+|      while_statement_nsi
+|      for_statement_nsi
+;
+
+statement_without_trailing_substatement:
+       block
+|      empty_statement
+|      expression_statement
+|      switch_statement
+|      do_statement
+|      break_statement
+|      continue_statement
+|      return_statement
+|      synchronized_statement
+|      throw_statement
+|      try_statement
+;
+
+empty_statement:
+       SC_TK
+;
+
+label_decl:
+       identifier REL_CL_TK
+               { USE_ABSORBER; }
+;
+
+labeled_statement:
+       label_decl statement
+;
+
+labeled_statement_nsi:
+       label_decl statement_nsi
+;
+
+/* We concentrate here a bunch of error handling rules that we couldn't write
+   earlier, because expression_statement catches a missing ';'.  */
+expression_statement:
+       statement_expression SC_TK
+;
+
+statement_expression: 
+       assignment
+|      pre_increment_expression
+|      pre_decrement_expression
+|      post_increment_expression
+|      post_decrement_expression
+|      method_invocation
+|      class_instance_creation_expression
+;
+
+if_then_statement:
+       IF_TK OP_TK expression CP_TK statement
+;
+
+if_then_else_statement:
+       IF_TK OP_TK expression CP_TK statement_nsi ELSE_TK statement
+;
+
+if_then_else_statement_nsi:
+       IF_TK OP_TK expression CP_TK statement_nsi ELSE_TK statement_nsi
+;
+
+switch_statement:
+       SWITCH_TK OP_TK expression CP_TK switch_block
+;
+
+switch_block:
+       OCB_TK CCB_TK
+|      OCB_TK switch_labels CCB_TK
+|      OCB_TK switch_block_statement_groups CCB_TK
+|      OCB_TK switch_block_statement_groups switch_labels CCB_TK
+;
+
+switch_block_statement_groups: 
+       switch_block_statement_group
+|      switch_block_statement_groups switch_block_statement_group
+;
+
+switch_block_statement_group:
+       switch_labels block_statements
+;
+
+
+switch_labels:
+       switch_label
+|      switch_labels switch_label
+;
+
+switch_label:
+       CASE_TK constant_expression REL_CL_TK
+|      DEFAULT_TK REL_CL_TK
+;
+
+while_expression:
+       WHILE_TK OP_TK expression CP_TK
+;
+
+while_statement:
+       while_expression statement
+;
+
+while_statement_nsi:
+       while_expression statement_nsi
+;
+
+do_statement_begin:
+       DO_TK
+;
+
+do_statement: 
+       do_statement_begin statement WHILE_TK OP_TK expression CP_TK SC_TK
+;
+
+for_statement:
+       for_begin SC_TK expression SC_TK for_update CP_TK statement
+|      for_begin SC_TK SC_TK for_update CP_TK statement
+;
+
+for_statement_nsi:
+       for_begin SC_TK expression SC_TK for_update CP_TK statement_nsi
+|      for_begin SC_TK SC_TK for_update CP_TK statement_nsi
+;
+
+for_header:
+       FOR_TK OP_TK
+;
+
+for_begin:
+       for_header for_init
+;
+for_init:                      /* Can be empty */
+|      statement_expression_list
+|      local_variable_declaration
+;
+
+for_update:                    /* Can be empty */
+|      statement_expression_list
+;
+
+statement_expression_list:
+       statement_expression
+|      statement_expression_list C_TK statement_expression
+;
+
+break_statement:
+       BREAK_TK SC_TK
+|      BREAK_TK identifier SC_TK
+;
+
+continue_statement:
+       CONTINUE_TK SC_TK
+|       CONTINUE_TK identifier SC_TK
+;
+
+return_statement:
+       RETURN_TK SC_TK
+|      RETURN_TK expression SC_TK
+;
+
+throw_statement:
+       THROW_TK expression SC_TK
+;
+
+synchronized_statement:
+       synchronized OP_TK expression CP_TK block
+|      synchronized OP_TK expression CP_TK error
+;
+
+synchronized:                  /* Test lval.sub_token here */
+       MODIFIER_TK
+               { USE_ABSORBER; }
+;
+
+try_statement:
+       TRY_TK block catches
+|      TRY_TK block finally
+|      TRY_TK block catches finally
+;
+
+catches:
+       catch_clause
+|      catches catch_clause
+;
+
+catch_clause:
+       CATCH_TK OP_TK formal_parameter CP_TK block
+;
+
+finally:
+       FINALLY_TK block
+;
+
+/* 19.12 Production from 15: Expressions  */
+primary:
+       primary_no_new_array
+|      array_creation_expression
+;
+
+primary_no_new_array:
+       literal
+|      THIS_TK
+|      OP_TK expression CP_TK
+|      class_instance_creation_expression
+|      field_access
+|      method_invocation
+|      array_access
+       /* type DOT_TK CLASS_TK doens't work. So we split the rule
+          'type' into its components. Missing is something for array,
+          which will complete the reference_type part. FIXME */
+|      name DOT_TK CLASS_TK           /* Added, JDK1.1 class literals */
+               { USE_ABSORBER; }
+|      primitive_type DOT_TK CLASS_TK /* Added, JDK1.1 class literals */
+               { USE_ABSORBER; }
+|      VOID_TK DOT_TK CLASS_TK        /* Added, JDK1.1 class literals */
+        /* Added, JDK1.1 inner classes. Documentation is wrong
+           refering to a 'ClassName' (class_name) rule that doesn't
+           exist. Used name instead.  */
+|      name DOT_TK THIS_TK
+               { USE_ABSORBER; }
+;
+
+class_instance_creation_expression:
+       NEW_TK class_type OP_TK argument_list CP_TK
+|      NEW_TK class_type OP_TK CP_TK
+        /* Added, JDK1.1 inner classes but modified to use
+           'class_type' instead of 'TypeName' (type_name) mentionned
+           in the documentation but doesn't exist. */
+|      NEW_TK class_type OP_TK argument_list CP_TK class_body
+|      NEW_TK class_type OP_TK CP_TK class_body         
+        /* Added, JDK1.1 inner classes, modified to use name or
+          primary instead of primary solely which couldn't work in
+          all situations.  */
+|      something_dot_new identifier OP_TK CP_TK
+|      something_dot_new identifier OP_TK CP_TK class_body
+|      something_dot_new identifier OP_TK argument_list CP_TK
+|      something_dot_new identifier OP_TK argument_list CP_TK class_body
+;
+
+something_dot_new:             /* Added, not part of the specs. */
+       name DOT_TK NEW_TK
+               { USE_ABSORBER; }
+|      primary DOT_TK NEW_TK
+;
+
+argument_list:
+       expression
+|      argument_list C_TK expression
+|      argument_list C_TK error
+;
+
+array_creation_expression:
+       NEW_TK primitive_type dim_exprs
+|      NEW_TK class_or_interface_type dim_exprs
+|      NEW_TK primitive_type dim_exprs dims
+|      NEW_TK class_or_interface_type dim_exprs dims
+        /* Added, JDK1.1 anonymous array. Initial documentation rule
+           modified */
+|      NEW_TK class_or_interface_type dims array_initializer
+|      NEW_TK primitive_type dims array_initializer
+;
+
+dim_exprs:
+       dim_expr
+|      dim_exprs dim_expr
+;
+
+dim_expr:
+       OSB_TK expression CSB_TK
+;
+
+dims:                          
+       OSB_TK CSB_TK
+|      dims OSB_TK CSB_TK
+;
+
+field_access:
+       primary DOT_TK identifier
+|      SUPER_TK DOT_TK identifier
+;
+
+method_invocation:
+       name OP_TK CP_TK
+               { USE_ABSORBER; }
+|      name OP_TK argument_list CP_TK
+               { USE_ABSORBER; }
+|      primary DOT_TK identifier OP_TK CP_TK
+|      primary DOT_TK identifier OP_TK argument_list CP_TK
+|      SUPER_TK DOT_TK identifier OP_TK CP_TK
+|      SUPER_TK DOT_TK identifier OP_TK argument_list CP_TK
+;
+
+array_access:
+       name OSB_TK expression CSB_TK
+               { USE_ABSORBER; }
+|      primary_no_new_array OSB_TK expression CSB_TK
+;
+
+postfix_expression:
+       primary
+|      name
+               { USE_ABSORBER; }
+|      post_increment_expression
+|      post_decrement_expression
+;
+
+post_increment_expression:
+       postfix_expression INCR_TK
+;
+
+post_decrement_expression:
+       postfix_expression DECR_TK
+;
+
+unary_expression:
+       pre_increment_expression
+|      pre_decrement_expression
+|      PLUS_TK unary_expression
+|      MINUS_TK unary_expression
+|      unary_expression_not_plus_minus
+;
+
+pre_increment_expression:
+       INCR_TK unary_expression
+;
+
+pre_decrement_expression:
+       DECR_TK unary_expression
+;
+
+unary_expression_not_plus_minus:
+       postfix_expression
+|      NOT_TK unary_expression
+|      NEG_TK unary_expression
+|      cast_expression
+;
+
+cast_expression:               /* Error handling here is potentially weak */
+       OP_TK primitive_type dims CP_TK unary_expression
+|      OP_TK primitive_type CP_TK unary_expression
+|      OP_TK expression CP_TK unary_expression_not_plus_minus
+|      OP_TK name dims CP_TK unary_expression_not_plus_minus
+;
+
+multiplicative_expression:
+       unary_expression
+|      multiplicative_expression MULT_TK unary_expression
+|      multiplicative_expression DIV_TK unary_expression
+|      multiplicative_expression REM_TK unary_expression
+;
+
+additive_expression:
+       multiplicative_expression
+|      additive_expression PLUS_TK multiplicative_expression
+|      additive_expression MINUS_TK multiplicative_expression
+;
+
+shift_expression:
+       additive_expression
+|      shift_expression LS_TK additive_expression
+|      shift_expression SRS_TK additive_expression
+|      shift_expression ZRS_TK additive_expression
+;
+
+relational_expression:
+       shift_expression
+|      relational_expression LT_TK shift_expression
+|      relational_expression GT_TK shift_expression
+|      relational_expression LTE_TK shift_expression
+|      relational_expression GTE_TK shift_expression
+|      relational_expression INSTANCEOF_TK reference_type
+;
+
+equality_expression:
+       relational_expression
+|      equality_expression EQ_TK relational_expression
+|      equality_expression NEQ_TK relational_expression
+;
+
+and_expression:
+       equality_expression
+|      and_expression AND_TK equality_expression
+;
+
+exclusive_or_expression:
+       and_expression
+|      exclusive_or_expression XOR_TK and_expression
+;
+
+inclusive_or_expression:
+       exclusive_or_expression
+|      inclusive_or_expression OR_TK exclusive_or_expression
+;
+
+conditional_and_expression:
+       inclusive_or_expression
+|      conditional_and_expression BOOL_AND_TK inclusive_or_expression
+;
+
+conditional_or_expression:
+       conditional_and_expression
+|      conditional_or_expression BOOL_OR_TK conditional_and_expression
+;
+
+conditional_expression:                /* Error handling here is weak */
+       conditional_or_expression
+|      conditional_or_expression REL_QM_TK expression REL_CL_TK conditional_expression
+;
+
+assignment_expression:
+       conditional_expression
+|      assignment
+;
+
+assignment:
+       left_hand_side assignment_operator assignment_expression
+;
+
+left_hand_side:
+       name
+               { USE_ABSORBER; }
+|      field_access
+|      array_access
+;
+
+assignment_operator:
+       ASSIGN_ANY_TK
+|      ASSIGN_TK
+;
+
+expression:
+       assignment_expression
+;
+
+constant_expression:
+       expression
+;
+
+%%
+\f
+#include "lex.c"
+
+/* Create a new parser context */
+
+void
+java_push_parser_context ()
+{
+  struct parser_ctxt *new = 
+    (struct parser_ctxt *)xmalloc(sizeof (struct parser_ctxt));
+
+  bzero (new, sizeof (struct parser_ctxt));
+  new->next = ctxp;
+  ctxp = new;
+}  
+
+/* Actions defined here */
+
+static void
+report_class_declaration (name)
+     char * name;
+{
+  extern int flag_dump_class, flag_list_filename;
+
+  if (flag_dump_class)
+    {
+      if (!previous_output)
+       {
+         if (flag_list_filename)
+           fprintf (out, "%s: ", input_filename);
+         previous_output = 1;
+       }
+       
+      if (package_name)
+       fprintf (out, "%s.%s ", package_name, name);
+      else
+       fprintf (out, "%s ", name);
+    }
+      
+  current_class = name;
+}
+
+static void
+report_main_declaration (declarator)
+     struct method_declarator *declarator;
+{
+  extern int flag_find_main;
+
+  if (flag_find_main
+      && modifier_value == 2
+      && !strcmp (declarator->method_name, "main") 
+      && declarator->args 
+      && declarator->args [0] == '[' 
+      && !strcmp( declarator->args+1, "String")
+      && current_class)
+    {
+      if (!previous_output)
+       {
+         if (package_name)
+           fprintf (out, "%s.%s ", package_name, current_class);
+         else
+           fprintf (out, current_class);
+         previous_output = 1;
+       }
+    }
+}
+
+/* Reset global status used by the report functions.  */
+
+void reset_report ()
+{
+  previous_output = 0;
+  current_class = package_name = NULL;
+}
+
+void
+yyerror (msg)
+     char *msg;
+{
+}
diff --git a/gcc/java/parse.c b/gcc/java/parse.c
new file mode 100644 (file)
index 0000000..224ae38
--- /dev/null
@@ -0,0 +1,10648 @@
+
+/*  A Bison parser, made from /nfs/hoser/beer/java/egcs/gcc/java/parse.y
+ by  GNU Bison version 1.25
+  */
+
+#define YYBISON 1  /* Identify Bison output.  */
+
+#define yyparse java_parse
+#define yylex java_lex
+#define yyerror java_error
+#define yylval java_lval
+#define yychar java_char
+#define yydebug java_debug
+#define yynerrs java_nerrs
+#define        PLUS_TK 258
+#define        MINUS_TK        259
+#define        MULT_TK 260
+#define        DIV_TK  261
+#define        REM_TK  262
+#define        LS_TK   263
+#define        SRS_TK  264
+#define        ZRS_TK  265
+#define        AND_TK  266
+#define        XOR_TK  267
+#define        OR_TK   268
+#define        BOOL_AND_TK     269
+#define        BOOL_OR_TK      270
+#define        EQ_TK   271
+#define        NEQ_TK  272
+#define        GT_TK   273
+#define        GTE_TK  274
+#define        LT_TK   275
+#define        LTE_TK  276
+#define        PLUS_ASSIGN_TK  277
+#define        MINUS_ASSIGN_TK 278
+#define        MULT_ASSIGN_TK  279
+#define        DIV_ASSIGN_TK   280
+#define        REM_ASSIGN_TK   281
+#define        LS_ASSIGN_TK    282
+#define        SRS_ASSIGN_TK   283
+#define        ZRS_ASSIGN_TK   284
+#define        AND_ASSIGN_TK   285
+#define        XOR_ASSIGN_TK   286
+#define        OR_ASSIGN_TK    287
+#define        PUBLIC_TK       288
+#define        PRIVATE_TK      289
+#define        PROTECTED_TK    290
+#define        STATIC_TK       291
+#define        FINAL_TK        292
+#define        SYNCHRONIZED_TK 293
+#define        VOLATILE_TK     294
+#define        TRANSIENT_TK    295
+#define        NATIVE_TK       296
+#define        PAD_TK  297
+#define        ABSTRACT_TK     298
+#define        MODIFIER_TK     299
+#define        DECR_TK 300
+#define        INCR_TK 301
+#define        DEFAULT_TK      302
+#define        IF_TK   303
+#define        THROW_TK        304
+#define        BOOLEAN_TK      305
+#define        DO_TK   306
+#define        IMPLEMENTS_TK   307
+#define        THROWS_TK       308
+#define        BREAK_TK        309
+#define        IMPORT_TK       310
+#define        ELSE_TK 311
+#define        INSTANCEOF_TK   312
+#define        RETURN_TK       313
+#define        VOID_TK 314
+#define        CATCH_TK        315
+#define        INTERFACE_TK    316
+#define        CASE_TK 317
+#define        EXTENDS_TK      318
+#define        FINALLY_TK      319
+#define        SUPER_TK        320
+#define        WHILE_TK        321
+#define        CLASS_TK        322
+#define        SWITCH_TK       323
+#define        CONST_TK        324
+#define        TRY_TK  325
+#define        FOR_TK  326
+#define        NEW_TK  327
+#define        CONTINUE_TK     328
+#define        GOTO_TK 329
+#define        PACKAGE_TK      330
+#define        THIS_TK 331
+#define        BYTE_TK 332
+#define        SHORT_TK        333
+#define        INT_TK  334
+#define        LONG_TK 335
+#define        CHAR_TK 336
+#define        INTEGRAL_TK     337
+#define        FLOAT_TK        338
+#define        DOUBLE_TK       339
+#define        FP_TK   340
+#define        ID_TK   341
+#define        REL_QM_TK       342
+#define        REL_CL_TK       343
+#define        NOT_TK  344
+#define        NEG_TK  345
+#define        ASSIGN_ANY_TK   346
+#define        ASSIGN_TK       347
+#define        OP_TK   348
+#define        CP_TK   349
+#define        OCB_TK  350
+#define        CCB_TK  351
+#define        OSB_TK  352
+#define        CSB_TK  353
+#define        SC_TK   354
+#define        C_TK    355
+#define        DOT_TK  356
+#define        STRING_LIT_TK   357
+#define        CHAR_LIT_TK     358
+#define        INT_LIT_TK      359
+#define        FP_LIT_TK       360
+#define        TRUE_TK 361
+#define        FALSE_TK        362
+#define        BOOL_LIT_TK     363
+#define        NULL_TK 364
+
+#line 49 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+#include <dirent.h>
+#ifdef __STDC__
+#include <stdarg.h>
+#else
+#include <varargs.h>
+#endif
+
+#include "config.h"
+#include "tree.h"
+#include "rtl.h"
+#include "obstack.h"
+#include "flags.h"
+#include "java-tree.h"
+#include "jcf.h"
+#include "lex.h"
+#include "parse.h"
+#include "zipfile.h"
+
+/* Number of error found so far. */
+int java_error_count; 
+/* Number of warning found so far. */
+int java_warning_count;
+
+/* The current parser context */
+static struct parser_ctxt *ctxp;
+
+/* binop_lookup maps token to tree_code. It is used where binary
+   operations are involved and required by the parser. RDIV_EXPR
+   covers both integral/floating point division. The code is changed
+   once the type of both operator is worked out.  */
+
+static enum tree_code binop_lookup[19] = 
+  { 
+    PLUS_EXPR, MINUS_EXPR, MULT_EXPR, RDIV_EXPR, TRUNC_MOD_EXPR,
+    LSHIFT_EXPR, RSHIFT_EXPR, URSHIFT_EXPR, 
+    BIT_AND_EXPR, BIT_XOR_EXPR, BIT_IOR_EXPR,
+    TRUTH_ANDIF_EXPR, TRUTH_ORIF_EXPR,
+    EQ_EXPR, NE_EXPR, GT_EXPR, GE_EXPR, LT_EXPR, LE_EXPR,
+   };
+#define BINOP_LOOKUP(VALUE)                                            \
+  binop_lookup [((VALUE) - PLUS_TK)%                                   \
+               (sizeof (binop_lookup) / sizeof (binop_lookup[0]))]
+
+/* Fake WFL used to report error message. It is initialized once if
+   needed and reused with it's location information is overriden.  */
+static tree wfl_operator = NULL_TREE;
+
+/* The "$L" identifier we use to create labels.  */
+static tree label_id;
+
+#line 104 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+typedef union {
+  tree node;
+  int sub_token;
+  struct {
+    int token;
+    int location;
+  } operator;
+  int value;
+} YYSTYPE;
+#ifndef YYDEBUG
+#define YYDEBUG 1
+#endif
+
+#include <stdio.h>
+
+#ifndef __cplusplus
+#ifndef __STDC__
+#define const
+#endif
+#endif
+
+
+
+#define        YYFINAL         772
+#define        YYFLAG          -32768
+#define        YYNTBASE        110
+
+#define YYTRANSLATE(x) ((unsigned)(x) <= 364 ? yytranslate[x] : 259)
+
+static const char yytranslate[] = {     0,
+     2,     2,     2,     2,     2,     2,     2,     2,     2,     2,
+     2,     2,     2,     2,     2,     2,     2,     2,     2,     2,
+     2,     2,     2,     2,     2,     2,     2,     2,     2,     2,
+     2,     2,     2,     2,     2,     2,     2,     2,     2,     2,
+     2,     2,     2,     2,     2,     2,     2,     2,     2,     2,
+     2,     2,     2,     2,     2,     2,     2,     2,     2,     2,
+     2,     2,     2,     2,     2,     2,     2,     2,     2,     2,
+     2,     2,     2,     2,     2,     2,     2,     2,     2,     2,
+     2,     2,     2,     2,     2,     2,     2,     2,     2,     2,
+     2,     2,     2,     2,     2,     2,     2,     2,     2,     2,
+     2,     2,     2,     2,     2,     2,     2,     2,     2,     2,
+     2,     2,     2,     2,     2,     2,     2,     2,     2,     2,
+     2,     2,     2,     2,     2,     2,     2,     2,     2,     2,
+     2,     2,     2,     2,     2,     2,     2,     2,     2,     2,
+     2,     2,     2,     2,     2,     2,     2,     2,     2,     2,
+     2,     2,     2,     2,     2,     2,     2,     2,     2,     2,
+     2,     2,     2,     2,     2,     2,     2,     2,     2,     2,
+     2,     2,     2,     2,     2,     2,     2,     2,     2,     2,
+     2,     2,     2,     2,     2,     2,     2,     2,     2,     2,
+     2,     2,     2,     2,     2,     2,     2,     2,     2,     2,
+     2,     2,     2,     2,     2,     2,     2,     2,     2,     2,
+     2,     2,     2,     2,     2,     2,     2,     2,     2,     2,
+     2,     2,     2,     2,     2,     2,     2,     2,     2,     2,
+     2,     2,     2,     2,     2,     2,     2,     2,     2,     2,
+     2,     2,     2,     2,     2,     2,     2,     2,     2,     2,
+     2,     2,     2,     2,     2,     1,     2,     3,     4,     5,
+     6,     7,     8,     9,    10,    11,    12,    13,    14,    15,
+    16,    17,    18,    19,    20,    21,    22,    23,    24,    25,
+    26,    27,    28,    29,    30,    31,    32,    33,    34,    35,
+    36,    37,    38,    39,    40,    41,    42,    43,    44,    45,
+    46,    47,    48,    49,    50,    51,    52,    53,    54,    55,
+    56,    57,    58,    59,    60,    61,    62,    63,    64,    65,
+    66,    67,    68,    69,    70,    71,    72,    73,    74,    75,
+    76,    77,    78,    79,    80,    81,    82,    83,    84,    85,
+    86,    87,    88,    89,    90,    91,    92,    93,    94,    95,
+    96,    97,    98,    99,   100,   101,   102,   103,   104,   105,
+   106,   107,   108,   109
+};
+
+#if YYDEBUG != 0
+static const short yyprhs[] = {     0,
+     0,     2,     4,     6,     8,    10,    12,    14,    16,    18,
+    20,    22,    24,    26,    28,    30,    32,    34,    38,    42,
+    46,    50,    54,    56,    58,    60,    64,    66,    67,    69,
+    71,    73,    76,    79,    82,    86,    88,    91,    93,    96,
+   100,   103,   107,   109,   111,   115,   118,   122,   128,   133,
+   139,   141,   143,   145,   147,   149,   152,   153,   161,   162,
+   169,   173,   176,   180,   185,   186,   189,   193,   196,   197,
+   200,   203,   205,   209,   213,   216,   220,   222,   225,   227,
+   229,   231,   233,   235,   237,   239,   241,   245,   250,   252,
+   256,   260,   262,   266,   270,   275,   277,   281,   284,   288,
+   292,   294,   296,   297,   301,   304,   308,   312,   317,   322,
+   325,   329,   332,   336,   339,   343,   348,   352,   356,   360,
+   362,   366,   370,   373,   377,   380,   384,   385,   388,   391,
+   393,   397,   401,   403,   406,   408,   411,   415,   417,   421,
+   426,   431,   437,   441,   446,   449,   453,   457,   462,   467,
+   473,   481,   488,   490,   492,   493,   498,   499,   505,   506,
+   512,   513,   520,   524,   529,   532,   536,   539,   543,   546,
+   550,   552,   555,   557,   559,   561,   563,   565,   568,   571,
+   574,   578,   582,   587,   589,   593,   597,   600,   601,   606,
+   608,   611,   613,   615,   617,   620,   623,   627,   629,   631,
+   633,   635,   637,   639,   641,   643,   645,   647,   649,   651,
+   653,   655,   657,   659,   661,   663,   665,   667,   669,   671,
+   673,   676,   679,   682,   685,   688,   691,   694,   697,   701,
+   706,   711,   717,   722,   728,   735,   743,   750,   752,   754,
+   756,   758,   760,   762,   764,   770,   773,   777,   782,   790,
+   798,   804,   807,   811,   817,   820,   824,   828,   833,   835,
+   838,   841,   843,   846,   850,   853,   856,   860,   863,   868,
+   871,   874,   878,   883,   886,   888,   896,   904,   911,   915,
+   921,   926,   934,   941,   944,   947,   951,   954,   955,   957,
+   959,   962,   963,   965,   967,   971,   975,   978,   982,   985,
+   989,   992,   996,   999,  1003,  1006,  1010,  1013,  1017,  1021,
+  1024,  1028,  1034,  1040,  1043,  1048,  1052,  1054,  1058,  1062,
+  1067,  1070,  1072,  1075,  1081,  1084,  1089,  1093,  1096,  1099,
+  1101,  1103,  1105,  1107,  1111,  1113,  1115,  1117,  1119,  1123,
+  1127,  1131,  1135,  1139,  1143,  1147,  1151,  1157,  1162,  1169,
+  1175,  1180,  1186,  1192,  1199,  1203,  1207,  1212,  1218,  1221,
+  1225,  1229,  1233,  1235,  1239,  1243,  1247,  1251,  1256,  1261,
+  1266,  1271,  1275,  1279,  1281,  1284,  1288,  1292,  1295,  1298,
+  1302,  1306,  1310,  1314,  1317,  1321,  1326,  1332,  1339,  1345,
+  1352,  1357,  1362,  1367,  1372,  1376,  1381,  1385,  1390,  1392,
+  1394,  1396,  1398,  1401,  1404,  1406,  1408,  1411,  1414,  1416,
+  1419,  1422,  1425,  1428,  1431,  1434,  1436,  1439,  1442,  1444,
+  1447,  1450,  1456,  1461,  1466,  1472,  1477,  1480,  1486,  1491,
+  1497,  1499,  1503,  1507,  1511,  1515,  1519,  1523,  1525,  1529,
+  1533,  1537,  1541,  1543,  1547,  1551,  1555,  1559,  1563,  1567,
+  1569,  1573,  1577,  1581,  1585,  1589,  1593,  1597,  1601,  1605,
+  1609,  1611,  1615,  1619,  1623,  1627,  1629,  1633,  1637,  1639,
+  1643,  1647,  1649,  1653,  1657,  1659,  1663,  1667,  1669,  1673,
+  1677,  1679,  1685,  1690,  1694,  1700,  1702,  1704,  1708,  1712,
+  1714,  1716,  1718,  1720,  1722,  1724
+};
+
+static const short yyrhs[] = {   123,
+     0,   104,     0,   105,     0,   108,     0,   103,     0,   102,
+     0,   109,     0,   113,     0,   114,     0,    82,     0,    85,
+     0,    50,     0,   115,     0,   118,     0,   119,     0,   115,
+     0,   115,     0,   113,    97,    98,     0,   119,    97,    98,
+     0,   118,    97,    98,     0,   113,    97,     1,     0,   118,
+    97,     1,     0,   120,     0,   121,     0,   122,     0,   119,
+   101,   122,     0,    86,     0,     0,   126,     0,   124,     0,
+   125,     0,   126,   124,     0,   126,   125,     0,   124,   125,
+     0,   126,   124,   125,     0,   127,     0,   124,   127,     0,
+   130,     0,   125,   130,     0,    75,   119,    99,     0,    75,
+     1,     0,    75,   119,     1,     0,   128,     0,   129,     0,
+    55,   119,    99,     0,    55,     1,     0,    55,   119,     1,
+     0,    55,   119,   101,     5,    99,     0,    55,   119,   101,
+     1,     0,    55,   119,   101,     5,     1,     0,   132,     0,
+   163,     0,    99,     0,     1,     0,    44,     0,   131,    44,
+     0,     0,   131,    67,   122,   135,   136,   133,   138,     0,
+     0,    67,   122,   135,   136,   134,   138,     0,   131,    67,
+     1,     0,    67,     1,     0,    67,   122,     1,     0,   131,
+    67,   122,     1,     0,     0,    63,   116,     0,    63,   116,
+     1,     0,    63,     1,     0,     0,    52,   137,     0,    52,
+     1,     0,   117,     0,   137,   100,   117,     0,   137,   100,
+     1,     0,    95,    96,     0,    95,   139,    96,     0,   140,
+     0,   139,   140,     0,   141,     0,   156,     0,   158,     0,
+   176,     0,   142,     0,   147,     0,   132,     0,   163,     0,
+   112,   143,    99,     0,   131,   112,   143,    99,     0,   144,
+     0,   143,   100,   144,     0,   143,   100,     1,     0,   145,
+     0,   145,    92,   146,     0,   145,    92,     1,     0,   145,
+    92,   146,     1,     0,   122,     0,   145,    97,    98,     0,
+   122,     1,     0,   145,    97,     1,     0,   145,    98,     1,
+     0,   257,     0,   174,     0,     0,   149,   148,   155,     0,
+   149,     1,     0,   112,   150,   153,     0,    59,   150,   153,
+     0,   131,   112,   150,   153,     0,   131,    59,   150,   153,
+     0,   112,     1,     0,   131,   112,     1,     0,    59,     1,
+     0,   131,    59,     1,     0,   131,     1,     0,   122,    93,
+    94,     0,   122,    93,   151,    94,     0,   150,    97,    98,
+     0,   122,    93,     1,     0,   150,    97,     1,     0,   152,
+     0,   151,   100,   152,     0,   151,   100,     1,     0,   112,
+   145,     0,   131,   112,   145,     0,   112,     1,     0,   131,
+   112,     1,     0,     0,    53,   154,     0,    53,     1,     0,
+   116,     0,   154,   100,   116,     0,   154,   100,     1,     0,
+   176,     0,   176,    99,     0,    99,     0,   157,   176,     0,
+   157,   176,    99,     0,    44,     0,   159,   153,   160,     0,
+   131,   159,   153,   160,     0,   159,   153,   160,    99,     0,
+   131,   159,   153,   160,    99,     0,   120,    93,    94,     0,
+   120,    93,   151,    94,     0,    95,    96,     0,    95,   161,
+    96,     0,    95,   178,    96,     0,    95,   161,   178,    96,
+     0,   162,    93,    94,    99,     0,   162,    93,   226,    94,
+    99,     0,   119,   101,    65,    93,   226,    94,    99,     0,
+   119,   101,    65,    93,    94,    99,     0,    76,     0,    65,
+     0,     0,    61,   122,   164,   169,     0,     0,   131,    61,
+   122,   165,   169,     0,     0,    61,   122,   168,   166,   169,
+     0,     0,   131,    61,   122,   168,   167,   169,     0,    61,
+   122,     1,     0,   131,    61,   122,     1,     0,    63,   117,
+     0,   168,   100,   117,     0,    63,     1,     0,   168,   100,
+     1,     0,    95,    96,     0,    95,   170,    96,     0,   171,
+     0,   170,   171,     0,   172,     0,   173,     0,   132,     0,
+   163,     0,   142,     0,   149,    99,     0,   149,     1,     0,
+    95,    96,     0,    95,   175,    96,     0,    95,   100,    96,
+     0,    95,   175,   100,    96,     0,   146,     0,   175,   100,
+   146,     0,   175,   100,     1,     0,    95,    96,     0,     0,
+    95,   177,   178,    96,     0,   179,     0,   178,   179,     0,
+   180,     0,   182,     0,   132,     0,   181,    99,     0,   112,
+   143,     0,   131,   112,   143,     0,   184,     0,   187,     0,
+   191,     0,   192,     0,   201,     0,   205,     0,   184,     0,
+   188,     0,   193,     0,   202,     0,   206,     0,   176,     0,
+   185,     0,   189,     0,   194,     0,   204,     0,   212,     0,
+   213,     0,   214,     0,   216,     0,   215,     0,   218,     0,
+    99,     0,   122,    88,     0,   186,   182,     0,   122,     1,
+     0,   186,   183,     0,   190,    99,     0,     1,    99,     0,
+     1,    95,     0,     1,    96,     0,   162,    93,     1,     0,
+   162,    93,    94,     1,     0,   162,    93,   226,     1,     0,
+   162,    93,   226,    94,     1,     0,   119,   101,    65,     1,
+     0,   119,   101,    65,    93,     1,     0,   119,   101,    65,
+    93,   226,     1,     0,   119,   101,    65,    93,   226,    94,
+     1,     0,   119,   101,    65,    93,    94,     1,     0,   254,
+     0,   238,     0,   239,     0,   235,     0,   236,     0,   232,
+     0,   224,     0,    48,    93,   257,    94,   182,     0,    48,
+     1,     0,    48,    93,     1,     0,    48,    93,   257,     1,
+     0,    48,    93,   257,    94,   183,    56,   182,     0,    48,
+    93,   257,    94,   183,    56,   183,     0,    68,    93,   257,
+    94,   195,     0,    68,     1,     0,    68,    93,     1,     0,
+    68,    93,   257,    94,     1,     0,    95,    96,     0,    95,
+   198,    96,     0,    95,   196,    96,     0,    95,   196,   198,
+    96,     0,   197,     0,   196,   197,     0,   198,   178,     0,
+   199,     0,   198,   199,     0,    62,   258,    88,     0,    47,
+    88,     0,    62,     1,     0,    62,   258,     1,     0,    47,
+     1,     0,    66,    93,   257,    94,     0,   200,   182,     0,
+    66,     1,     0,    66,    93,     1,     0,    66,    93,   257,
+     1,     0,   200,   183,     0,    51,     0,   203,   182,    66,
+    93,   257,    94,    99,     0,   208,    99,   257,    99,   210,
+    94,   182,     0,   208,    99,    99,   210,    94,   182,     0,
+   208,    99,     1,     0,   208,    99,   257,    99,     1,     0,
+   208,    99,    99,     1,     0,   208,    99,   257,    99,   210,
+    94,   183,     0,   208,    99,    99,   210,    94,   183,     0,
+    71,    93,     0,    71,     1,     0,    71,    93,     1,     0,
+   207,   209,     0,     0,   211,     0,   181,     0,   211,     1,
+     0,     0,   211,     0,   190,     0,   211,   100,   190,     0,
+   211,   100,     1,     0,    54,    99,     0,    54,   122,    99,
+     0,    54,     1,     0,    54,   122,     1,     0,    73,    99,
+     0,    73,   122,    99,     0,    73,     1,     0,    73,   122,
+     1,     0,    58,    99,     0,    58,   257,    99,     0,    58,
+     1,     0,    58,   257,     1,     0,    49,   257,    99,     0,
+    49,     1,     0,    49,   257,     1,     0,   217,    93,   257,
+    94,   176,     0,   217,    93,   257,    94,     1,     0,   217,
+     1,     0,   217,    93,     1,    94,     0,   217,    93,     1,
+     0,    44,     0,    70,   176,   219,     0,    70,   176,   221,
+     0,    70,   176,   219,   221,     0,    70,     1,     0,   220,
+     0,   219,   220,     0,    60,    93,   152,    94,   176,     0,
+    60,     1,     0,    60,    93,     1,    94,     0,    60,    93,
+     1,     0,    64,   176,     0,    64,     1,     0,   223,     0,
+   227,     0,   111,     0,    76,     0,    93,   257,    94,     0,
+   224,     0,   231,     0,   232,     0,   233,     0,   119,   101,
+    67,     0,   113,   101,    67,     0,    59,   101,    67,     0,
+   119,   101,    76,     0,    93,   257,     1,     0,   119,   101,
+     1,     0,   113,   101,     1,     0,    59,   101,     1,     0,
+    72,   116,    93,   226,    94,     0,    72,   116,    93,    94,
+     0,    72,   116,    93,   226,    94,   138,     0,    72,   116,
+    93,    94,   138,     0,   225,   122,    93,    94,     0,   225,
+   122,    93,    94,   138,     0,   225,   122,    93,   226,    94,
+     0,   225,   122,    93,   226,    94,   138,     0,    72,     1,
+    99,     0,    72,   116,     1,     0,    72,   116,    93,     1,
+     0,    72,   116,    93,   226,     1,     0,   225,     1,     0,
+   225,   122,     1,     0,   119,   101,    72,     0,   222,   101,
+    72,     0,   257,     0,   226,   100,   257,     0,   226,   100,
+     1,     0,    72,   113,   228,     0,    72,   115,   228,     0,
+    72,   113,   228,   230,     0,    72,   115,   228,   230,     0,
+    72,   115,   230,   174,     0,    72,   113,   230,   174,     0,
+    72,     1,    98,     0,    72,     1,    97,     0,   229,     0,
+   228,   229,     0,    97,   257,    98,     0,    97,   257,     1,
+     0,    97,     1,     0,    97,    98,     0,   230,    97,    98,
+     0,   230,    97,     1,     0,   222,   101,   122,     0,    65,
+   101,   122,     0,    65,     1,     0,   119,    93,    94,     0,
+   119,    93,   226,    94,     0,   222,   101,   122,    93,    94,
+     0,   222,   101,   122,    93,   226,    94,     0,    65,   101,
+   122,    93,    94,     0,    65,   101,   122,    93,   226,    94,
+     0,    65,   101,     1,    94,     0,    65,   101,     1,   101,
+     0,   119,    97,   257,    98,     0,   223,    97,   257,    98,
+     0,   119,    97,     1,     0,   119,    97,   257,     1,     0,
+   223,    97,     1,     0,   223,    97,   257,     1,     0,   222,
+     0,   119,     0,   235,     0,   236,     0,   234,    46,     0,
+   234,    45,     0,   238,     0,   239,     0,     3,   237,     0,
+     4,   237,     0,   240,     0,     3,     1,     0,     4,     1,
+     0,    46,   237,     0,    46,     1,     0,    45,   237,     0,
+    45,     1,     0,   234,     0,    89,   237,     0,    90,   237,
+     0,   241,     0,    89,     1,     0,    90,     1,     0,    93,
+   113,   230,    94,   237,     0,    93,   113,    94,   237,     0,
+    93,   257,    94,   240,     0,    93,   119,   230,    94,   240,
+     0,    93,   113,    97,     1,     0,    93,     1,     0,    93,
+   113,   230,    94,     1,     0,    93,   113,    94,     1,     0,
+    93,   119,   230,    94,     1,     0,   237,     0,   242,     5,
+   237,     0,   242,     6,   237,     0,   242,     7,   237,     0,
+   242,     5,     1,     0,   242,     6,     1,     0,   242,     7,
+     1,     0,   242,     0,   243,     3,   242,     0,   243,     4,
+   242,     0,   243,     3,     1,     0,   243,     4,     1,     0,
+   243,     0,   244,     8,   243,     0,   244,     9,   243,     0,
+   244,    10,   243,     0,   244,     8,     1,     0,   244,     9,
+     1,     0,   244,    10,     1,     0,   244,     0,   245,    20,
+   244,     0,   245,    18,   244,     0,   245,    21,   244,     0,
+   245,    19,   244,     0,   245,    57,   114,     0,   245,    20,
+     1,     0,   245,    18,     1,     0,   245,    21,     1,     0,
+   245,    19,     1,     0,   245,    57,     1,     0,   245,     0,
+   246,    16,   245,     0,   246,    17,   245,     0,   246,    16,
+     1,     0,   246,    17,     1,     0,   246,     0,   247,    11,
+   246,     0,   247,    11,     1,     0,   247,     0,   248,    12,
+   247,     0,   248,    12,     1,     0,   248,     0,   249,    13,
+   248,     0,   249,    13,     1,     0,   249,     0,   250,    14,
+   249,     0,   250,    14,     1,     0,   250,     0,   251,    15,
+   250,     0,   251,    15,     1,     0,   251,     0,   251,    87,
+   257,    88,   252,     0,   251,    87,    88,     1,     0,   251,
+    87,     1,     0,   251,    87,   257,    88,     1,     0,   252,
+     0,   254,     0,   255,   256,   253,     0,   255,   256,     1,
+     0,   119,     0,   231,     0,   233,     0,    91,     0,    92,
+     0,   253,     0,   257,     0
+};
+
+#endif
+
+#if YYDEBUG != 0
+static const short yyrline[] = { 0,
+   251,   257,   259,   260,   261,   262,   263,   267,   269,   272,
+   274,   275,   278,   280,   283,   287,   291,   295,   301,   303,
+   305,   307,   312,   314,   317,   321,   326,   331,   333,   334,
+   335,   336,   337,   338,   339,   342,   347,   353,   355,   358,
+   361,   363,   367,   369,   372,   402,   404,   408,   421,   423,
+   427,   433,   434,   436,   446,   451,   466,   470,   473,   476,
+   479,   481,   483,   485,   489,   491,   493,   495,   499,   501,
+   503,   510,   516,   521,   525,   528,   532,   534,   537,   539,
+   540,   541,   544,   546,   547,   548,   552,   555,   567,   570,
+   572,   576,   579,   586,   592,   600,   602,   606,   608,   610,
+   614,   616,   621,   628,   640,   644,   647,   649,   651,   653,
+   655,   657,   659,   661,   668,   671,   673,   678,   680,   684,
+   689,   694,   698,   703,   708,   710,   717,   718,   719,   723,
+   725,   726,   730,   732,   733,   738,   743,   749,   761,   766,
+   772,   777,   786,   788,   791,   793,   794,   795,   799,   801,
+   804,   805,   809,   816,   826,   830,   833,   836,   839,   842,
+   845,   848,   851,   853,   857,   863,   868,   870,   874,   877,
+   881,   883,   886,   888,   889,   890,   893,   897,   903,   908,
+   913,   917,   921,   927,   929,   930,   935,   938,   941,   948,
+   950,   953,   955,   957,   960,   964,   967,   971,   973,   975,
+   977,   979,   981,   991,   993,   995,   997,   999,  1003,  1006,
+  1008,  1010,  1012,  1014,  1016,  1018,  1019,  1021,  1023,  1027,
+  1032,  1043,  1050,  1054,  1065,  1075,  1081,  1087,  1093,  1095,
+  1097,  1099,  1101,  1103,  1105,  1107,  1109,  1113,  1115,  1119,
+  1123,  1127,  1131,  1132,  1138,  1141,  1143,  1145,  1149,  1154,
+  1159,  1161,  1163,  1165,  1169,  1171,  1172,  1173,  1176,  1178,
+  1181,  1186,  1188,  1191,  1193,  1194,  1196,  1198,  1202,  1210,
+  1213,  1215,  1217,  1221,  1226,  1235,  1240,  1243,  1250,  1252,
+  1254,  1258,  1261,  1270,  1277,  1279,  1283,  1296,  1298,  1304,
+  1310,  1314,  1316,  1320,  1323,  1325,  1329,  1332,  1334,  1336,
+  1340,  1343,  1345,  1347,  1351,  1354,  1356,  1358,  1362,  1364,
+  1366,  1370,  1372,  1374,  1376,  1378,  1382,  1389,  1391,  1392,
+  1393,  1397,  1399,  1402,  1404,  1406,  1408,  1412,  1414,  1419,
+  1421,  1424,  1426,  1428,  1430,  1431,  1432,  1433,  1437,  1438,
+  1439,  1443,  1444,  1446,  1448,  1450,  1454,  1460,  1468,  1470,
+  1475,  1476,  1477,  1478,  1479,  1481,  1483,  1485,  1487,  1489,
+  1493,  1495,  1498,  1504,  1509,  1513,  1516,  1518,  1520,  1524,
+  1526,  1528,  1530,  1534,  1537,  1541,  1547,  1549,  1557,  1560,
+  1562,  1566,  1569,  1576,  1580,  1583,  1585,  1590,  1595,  1603,
+  1615,  1617,  1621,  1624,  1626,  1631,  1636,  1641,  1648,  1650,
+  1651,  1652,  1655,  1660,  1665,  1667,  1668,  1670,  1672,  1673,
+  1675,  1679,  1682,  1686,  1689,  1693,  1695,  1697,  1699,  1700,
+  1702,  1706,  1714,  1716,  1718,  1730,  1732,  1738,  1740,  1742,
+  1746,  1748,  1753,  1758,  1763,  1765,  1767,  1771,  1773,  1778,
+  1783,  1785,  1789,  1791,  1796,  1801,  1806,  1808,  1810,  1814,
+  1816,  1821,  1826,  1831,  1836,  1837,  1839,  1841,  1843,  1845,
+  1849,  1851,  1856,  1861,  1863,  1867,  1869,  1874,  1878,  1880,
+  1885,  1889,  1891,  1896,  1900,  1902,  1907,  1911,  1913,  1918,
+  1922,  1924,  1925,  1931,  1933,  1937,  1939,  1942,  1945,  1953,
+  1955,  1956,  1959,  1961,  1964,  1968
+};
+#endif
+
+
+#if YYDEBUG != 0 || defined (YYERROR_VERBOSE)
+
+static const char * const yytname[] = {   "$","error","$undefined.","PLUS_TK",
+"MINUS_TK","MULT_TK","DIV_TK","REM_TK","LS_TK","SRS_TK","ZRS_TK","AND_TK","XOR_TK",
+"OR_TK","BOOL_AND_TK","BOOL_OR_TK","EQ_TK","NEQ_TK","GT_TK","GTE_TK","LT_TK",
+"LTE_TK","PLUS_ASSIGN_TK","MINUS_ASSIGN_TK","MULT_ASSIGN_TK","DIV_ASSIGN_TK",
+"REM_ASSIGN_TK","LS_ASSIGN_TK","SRS_ASSIGN_TK","ZRS_ASSIGN_TK","AND_ASSIGN_TK",
+"XOR_ASSIGN_TK","OR_ASSIGN_TK","PUBLIC_TK","PRIVATE_TK","PROTECTED_TK","STATIC_TK",
+"FINAL_TK","SYNCHRONIZED_TK","VOLATILE_TK","TRANSIENT_TK","NATIVE_TK","PAD_TK",
+"ABSTRACT_TK","MODIFIER_TK","DECR_TK","INCR_TK","DEFAULT_TK","IF_TK","THROW_TK",
+"BOOLEAN_TK","DO_TK","IMPLEMENTS_TK","THROWS_TK","BREAK_TK","IMPORT_TK","ELSE_TK",
+"INSTANCEOF_TK","RETURN_TK","VOID_TK","CATCH_TK","INTERFACE_TK","CASE_TK","EXTENDS_TK",
+"FINALLY_TK","SUPER_TK","WHILE_TK","CLASS_TK","SWITCH_TK","CONST_TK","TRY_TK",
+"FOR_TK","NEW_TK","CONTINUE_TK","GOTO_TK","PACKAGE_TK","THIS_TK","BYTE_TK","SHORT_TK",
+"INT_TK","LONG_TK","CHAR_TK","INTEGRAL_TK","FLOAT_TK","DOUBLE_TK","FP_TK","ID_TK",
+"REL_QM_TK","REL_CL_TK","NOT_TK","NEG_TK","ASSIGN_ANY_TK","ASSIGN_TK","OP_TK",
+"CP_TK","OCB_TK","CCB_TK","OSB_TK","CSB_TK","SC_TK","C_TK","DOT_TK","STRING_LIT_TK",
+"CHAR_LIT_TK","INT_LIT_TK","FP_LIT_TK","TRUE_TK","FALSE_TK","BOOL_LIT_TK","NULL_TK",
+"goal","literal","type","primitive_type","reference_type","class_or_interface_type",
+"class_type","interface_type","array_type","name","simple_name","qualified_name",
+"identifier","compilation_unit","import_declarations","type_declarations","package_declaration",
+"import_declaration","single_type_import_declaration","type_import_on_demand_declaration",
+"type_declaration","modifiers","class_declaration","@1","@2","super","interfaces",
+"interface_type_list","class_body","class_body_declarations","class_body_declaration",
+"class_member_declaration","field_declaration","variable_declarators","variable_declarator",
+"variable_declarator_id","variable_initializer","method_declaration","@3","method_header",
+"method_declarator","formal_parameter_list","formal_parameter","throws","class_type_list",
+"method_body","static_initializer","static","constructor_declaration","constructor_declarator",
+"constructor_body","explicit_constructor_invocation","this_or_super","interface_declaration",
+"@4","@5","@6","@7","extends_interfaces","interface_body","interface_member_declarations",
+"interface_member_declaration","constant_declaration","abstract_method_declaration",
+"array_initializer","variable_initializers","block","@8","block_statements",
+"block_statement","local_variable_declaration_statement","local_variable_declaration",
+"statement","statement_nsi","statement_without_trailing_substatement","empty_statement",
+"label_decl","labeled_statement","labeled_statement_nsi","expression_statement",
+"statement_expression","if_then_statement","if_then_else_statement","if_then_else_statement_nsi",
+"switch_statement","switch_block","switch_block_statement_groups","switch_block_statement_group",
+"switch_labels","switch_label","while_expression","while_statement","while_statement_nsi",
+"do_statement_begin","do_statement","for_statement","for_statement_nsi","for_header",
+"for_begin","for_init","for_update","statement_expression_list","break_statement",
+"continue_statement","return_statement","throw_statement","synchronized_statement",
+"synchronized","try_statement","catches","catch_clause","finally","primary",
+"primary_no_new_array","class_instance_creation_expression","something_dot_new",
+"argument_list","array_creation_expression","dim_exprs","dim_expr","dims","field_access",
+"method_invocation","array_access","postfix_expression","post_increment_expression",
+"post_decrement_expression","unary_expression","pre_increment_expression","pre_decrement_expression",
+"unary_expression_not_plus_minus","cast_expression","multiplicative_expression",
+"additive_expression","shift_expression","relational_expression","equality_expression",
+"and_expression","exclusive_or_expression","inclusive_or_expression","conditional_and_expression",
+"conditional_or_expression","conditional_expression","assignment_expression",
+"assignment","left_hand_side","assignment_operator","expression","constant_expression", NULL
+};
+#endif
+
+static const short yyr1[] = {     0,
+   110,   111,   111,   111,   111,   111,   111,   112,   112,   113,
+   113,   113,   114,   114,   115,   116,   117,   118,   118,   118,
+   118,   118,   119,   119,   120,   121,   122,   123,   123,   123,
+   123,   123,   123,   123,   123,   124,   124,   125,   125,   126,
+   126,   126,   127,   127,   128,   128,   128,   129,   129,   129,
+   130,   130,   130,   130,   131,   131,   133,   132,   134,   132,
+   132,   132,   132,   132,   135,   135,   135,   135,   136,   136,
+   136,   137,   137,   137,   138,   138,   139,   139,   140,   140,
+   140,   140,   141,   141,   141,   141,   142,   142,   143,   143,
+   143,   144,   144,   144,   144,   145,   145,   145,   145,   145,
+   146,   146,   148,   147,   147,   149,   149,   149,   149,   149,
+   149,   149,   149,   149,   150,   150,   150,   150,   150,   151,
+   151,   151,   152,   152,   152,   152,   153,   153,   153,   154,
+   154,   154,   155,   155,   155,   156,   156,   157,   158,   158,
+   158,   158,   159,   159,   160,   160,   160,   160,   161,   161,
+   161,   161,   162,   162,   164,   163,   165,   163,   166,   163,
+   167,   163,   163,   163,   168,   168,   168,   168,   169,   169,
+   170,   170,   171,   171,   171,   171,   172,   173,   173,   174,
+   174,   174,   174,   175,   175,   175,   176,   177,   176,   178,
+   178,   179,   179,   179,   180,   181,   181,   182,   182,   182,
+   182,   182,   182,   183,   183,   183,   183,   183,   184,   184,
+   184,   184,   184,   184,   184,   184,   184,   184,   184,   185,
+   186,   187,   187,   188,   189,   189,   189,   189,   189,   189,
+   189,   189,   189,   189,   189,   189,   189,   190,   190,   190,
+   190,   190,   190,   190,   191,   191,   191,   191,   192,   193,
+   194,   194,   194,   194,   195,   195,   195,   195,   196,   196,
+   197,   198,   198,   199,   199,   199,   199,   199,   200,   201,
+   201,   201,   201,   202,   203,   204,   205,   205,   205,   205,
+   205,   206,   206,   207,   207,   207,   208,   209,   209,   209,
+   209,   210,   210,   211,   211,   211,   212,   212,   212,   212,
+   213,   213,   213,   213,   214,   214,   214,   214,   215,   215,
+   215,   216,   216,   216,   216,   216,   217,   218,   218,   218,
+   218,   219,   219,   220,   220,   220,   220,   221,   221,   222,
+   222,   223,   223,   223,   223,   223,   223,   223,   223,   223,
+   223,   223,   223,   223,   223,   223,   224,   224,   224,   224,
+   224,   224,   224,   224,   224,   224,   224,   224,   224,   224,
+   225,   225,   226,   226,   226,   227,   227,   227,   227,   227,
+   227,   227,   227,   228,   228,   229,   229,   229,   230,   230,
+   230,   231,   231,   231,   232,   232,   232,   232,   232,   232,
+   232,   232,   233,   233,   233,   233,   233,   233,   234,   234,
+   234,   234,   235,   236,   237,   237,   237,   237,   237,   237,
+   237,   238,   238,   239,   239,   240,   240,   240,   240,   240,
+   240,   241,   241,   241,   241,   241,   241,   241,   241,   241,
+   242,   242,   242,   242,   242,   242,   242,   243,   243,   243,
+   243,   243,   244,   244,   244,   244,   244,   244,   244,   245,
+   245,   245,   245,   245,   245,   245,   245,   245,   245,   245,
+   246,   246,   246,   246,   246,   247,   247,   247,   248,   248,
+   248,   249,   249,   249,   250,   250,   250,   251,   251,   251,
+   252,   252,   252,   252,   252,   253,   253,   254,   254,   255,
+   255,   255,   256,   256,   257,   258
+};
+
+static const short yyr2[] = {     0,
+     1,     1,     1,     1,     1,     1,     1,     1,     1,     1,
+     1,     1,     1,     1,     1,     1,     1,     3,     3,     3,
+     3,     3,     1,     1,     1,     3,     1,     0,     1,     1,
+     1,     2,     2,     2,     3,     1,     2,     1,     2,     3,
+     2,     3,     1,     1,     3,     2,     3,     5,     4,     5,
+     1,     1,     1,     1,     1,     2,     0,     7,     0,     6,
+     3,     2,     3,     4,     0,     2,     3,     2,     0,     2,
+     2,     1,     3,     3,     2,     3,     1,     2,     1,     1,
+     1,     1,     1,     1,     1,     1,     3,     4,     1,     3,
+     3,     1,     3,     3,     4,     1,     3,     2,     3,     3,
+     1,     1,     0,     3,     2,     3,     3,     4,     4,     2,
+     3,     2,     3,     2,     3,     4,     3,     3,     3,     1,
+     3,     3,     2,     3,     2,     3,     0,     2,     2,     1,
+     3,     3,     1,     2,     1,     2,     3,     1,     3,     4,
+     4,     5,     3,     4,     2,     3,     3,     4,     4,     5,
+     7,     6,     1,     1,     0,     4,     0,     5,     0,     5,
+     0,     6,     3,     4,     2,     3,     2,     3,     2,     3,
+     1,     2,     1,     1,     1,     1,     1,     2,     2,     2,
+     3,     3,     4,     1,     3,     3,     2,     0,     4,     1,
+     2,     1,     1,     1,     2,     2,     3,     1,     1,     1,
+     1,     1,     1,     1,     1,     1,     1,     1,     1,     1,
+     1,     1,     1,     1,     1,     1,     1,     1,     1,     1,
+     2,     2,     2,     2,     2,     2,     2,     2,     3,     4,
+     4,     5,     4,     5,     6,     7,     6,     1,     1,     1,
+     1,     1,     1,     1,     5,     2,     3,     4,     7,     7,
+     5,     2,     3,     5,     2,     3,     3,     4,     1,     2,
+     2,     1,     2,     3,     2,     2,     3,     2,     4,     2,
+     2,     3,     4,     2,     1,     7,     7,     6,     3,     5,
+     4,     7,     6,     2,     2,     3,     2,     0,     1,     1,
+     2,     0,     1,     1,     3,     3,     2,     3,     2,     3,
+     2,     3,     2,     3,     2,     3,     2,     3,     3,     2,
+     3,     5,     5,     2,     4,     3,     1,     3,     3,     4,
+     2,     1,     2,     5,     2,     4,     3,     2,     2,     1,
+     1,     1,     1,     3,     1,     1,     1,     1,     3,     3,
+     3,     3,     3,     3,     3,     3,     5,     4,     6,     5,
+     4,     5,     5,     6,     3,     3,     4,     5,     2,     3,
+     3,     3,     1,     3,     3,     3,     3,     4,     4,     4,
+     4,     3,     3,     1,     2,     3,     3,     2,     2,     3,
+     3,     3,     3,     2,     3,     4,     5,     6,     5,     6,
+     4,     4,     4,     4,     3,     4,     3,     4,     1,     1,
+     1,     1,     2,     2,     1,     1,     2,     2,     1,     2,
+     2,     2,     2,     2,     2,     1,     2,     2,     1,     2,
+     2,     5,     4,     4,     5,     4,     2,     5,     4,     5,
+     1,     3,     3,     3,     3,     3,     3,     1,     3,     3,
+     3,     3,     1,     3,     3,     3,     3,     3,     3,     1,
+     3,     3,     3,     3,     3,     3,     3,     3,     3,     3,
+     1,     3,     3,     3,     3,     1,     3,     3,     1,     3,
+     3,     1,     3,     3,     1,     3,     3,     1,     3,     3,
+     1,     5,     4,     3,     5,     1,     1,     3,     3,     1,
+     1,     1,     1,     1,     1,     1
+};
+
+static const short yydefact[] = {     0,
+    54,    55,     0,     0,     0,     0,    53,     1,     0,     0,
+     0,    36,    43,    44,    38,     0,    51,    52,    46,    27,
+     0,    23,    24,    25,     0,    62,     0,    41,     0,     0,
+    37,    39,     0,     0,    56,     0,     0,    47,    45,     0,
+   163,     0,     0,   159,    63,     0,    69,    42,    40,     0,
+     0,     0,    61,     0,    49,     0,    26,   167,    17,   165,
+    15,     0,   156,     0,     0,    68,    16,     0,     0,    59,
+   164,     0,   161,    64,    69,    50,    48,    12,     0,    10,
+    11,   169,     0,     8,     9,    13,    14,    15,     0,   175,
+   177,     0,   176,     0,   171,   173,   174,   168,   166,   160,
+    67,    71,    72,    70,     0,   158,     0,    57,   112,     0,
+   127,   110,     0,     0,    89,    92,   127,     0,     0,     0,
+   114,     0,     0,   179,   178,   170,   172,     0,     0,    60,
+   162,     0,     0,     0,     0,   107,    98,    87,     0,     0,
+     0,     0,   106,    21,    18,    22,    20,    19,   113,   127,
+   111,     0,   127,    74,    73,    55,   188,    75,    23,     0,
+    85,     0,    77,    79,    83,    84,     0,    80,     0,    81,
+   127,    86,    82,    58,   118,   115,     0,     0,     0,   120,
+   129,   130,   128,   119,   117,    91,     0,    90,    94,     0,
+     0,     0,     0,     0,     0,     0,   333,     0,     0,     0,
+     0,     6,     5,     2,     3,     4,     7,   332,     0,   400,
+     0,   102,   399,   330,   335,     0,   331,   336,   337,   338,
+   416,   401,   402,   431,   405,   406,   409,   419,   438,   443,
+   450,   461,   466,   469,   472,   475,   478,   481,   486,   495,
+   487,     0,   101,    99,    97,   100,   109,    88,   108,   187,
+     0,     0,   127,    76,    78,   105,     0,   136,     0,   125,
+   123,     0,   116,     0,     0,   410,   400,   336,   338,   407,
+   411,   408,   415,   414,   413,   412,     0,   384,     0,     0,
+     0,    16,     0,   420,   417,   421,   418,   427,     0,   400,
+     0,   180,     0,   184,     0,     0,     0,     0,     0,    95,
+     0,     0,   359,     0,   404,   403,     0,     0,     0,     0,
+     0,     0,     0,     0,     0,     0,     0,     0,     0,     0,
+     0,     0,     0,     0,     0,     0,     0,   493,   494,     0,
+     0,    55,     0,     0,   275,     0,     0,     0,     0,     0,
+     0,     0,     0,   333,     0,   220,     0,     8,   400,     0,
+     0,   194,     0,   209,     0,   190,   192,     0,   193,   198,
+   210,     0,   199,   211,     0,   200,   201,   212,     0,   202,
+     0,   213,   203,   288,     0,   214,   215,   216,   218,   217,
+     0,   219,   244,   243,     0,   241,   242,   239,   240,   238,
+   143,     0,     0,   135,   104,   133,   137,     0,   139,   126,
+   124,   122,   121,   132,   131,   346,   341,     0,   383,   373,
+   372,   355,     0,   366,   374,     0,   367,     0,   356,     0,
+     0,     0,     0,     0,     0,   343,   334,   182,   181,     0,
+   345,   340,   385,     0,   363,   395,     0,   344,   339,   361,
+   342,   362,   382,   397,     0,   360,     0,   435,   432,   436,
+   433,   437,   434,   441,   439,   442,   440,   447,   444,   448,
+   445,   449,   446,   457,   452,   459,   454,   456,   451,   458,
+   453,   460,     0,   455,   464,   462,   465,   463,   468,   467,
+   471,   470,   474,   473,   477,   476,   480,   479,   484,     0,
+     0,   489,   488,   227,   228,   226,   246,     0,   310,     0,
+   299,   297,     0,   307,   305,     0,   271,     0,   252,     0,
+   321,     0,   285,     0,   303,   301,     0,     0,   196,     0,
+     0,   223,   221,     0,     0,   189,   191,   195,   317,   400,
+   222,   225,   270,     0,   400,     0,   290,   294,   287,     0,
+     0,   314,     0,   144,   140,   134,   145,   400,     0,     0,
+     0,   141,   391,   392,     0,   378,   379,     0,   375,   368,
+     0,   371,   369,   370,   357,   348,     0,   429,   423,   426,
+     0,     0,   424,   186,   183,   185,   386,     0,   396,   393,
+     0,   398,   394,   351,     0,   483,     0,   247,     0,   311,
+   309,   300,   298,   308,   306,   272,     0,   253,     0,     0,
+     0,   318,   322,   319,   286,   304,   302,   334,     0,   197,
+   229,     0,     0,     0,   291,     0,   279,     0,     0,   316,
+     0,   142,     0,   146,     0,     0,   147,   389,     0,   377,
+   376,   381,   380,   350,   358,   347,   428,   422,   430,   425,
+   365,   364,   387,     0,   352,   353,   485,   482,   248,     0,
+   273,   269,     0,   325,     0,   329,   328,   323,   320,   233,
+     0,   230,   231,     0,     0,   296,   295,   281,     0,   293,
+     0,   315,     0,     0,   148,     0,     0,   390,   349,   388,
+   354,     0,   245,     0,   198,     0,   205,   206,     0,   207,
+   208,     0,   254,     0,   251,   327,     0,   234,     0,     0,
+   232,     0,     0,   280,     0,   313,   312,     0,   149,     0,
+     0,     0,   224,   274,     0,     0,     0,   255,     0,   259,
+     0,   262,   326,     0,   237,   235,     0,     0,   278,     0,
+     0,     0,   150,     0,   249,     0,     0,   268,   265,   266,
+   496,     0,   257,   260,     0,   256,     0,   263,   324,   236,
+   276,   277,   152,     0,     0,     0,     0,   267,   264,   258,
+   151,     0,     0,     0,     0,   283,     0,   250,   282,     0,
+     0,     0
+};
+
+static const short yydefgoto[] = {   770,
+   208,   347,   209,    85,    86,    68,    60,    87,   210,    22,
+    23,    24,     8,     9,    10,    11,    12,    13,    14,    15,
+   351,   352,   132,   105,    47,    70,   104,   130,   162,   163,
+   164,    91,   114,   115,   116,   211,   166,   257,    92,   111,
+   179,   180,   136,   183,   395,   168,   169,   170,   171,   399,
+   549,   353,    18,    43,    72,    65,   107,    44,    63,    94,
+    95,    96,    97,   212,   295,   354,   251,   747,   356,   357,
+   358,   359,   684,   360,   361,   362,   363,   687,   364,   365,
+   366,   367,   688,   368,   695,   719,   720,   721,   722,   369,
+   370,   690,   371,   372,   373,   691,   374,   375,   539,   669,
+   670,   376,   377,   378,   379,   380,   381,   382,   602,   603,
+   604,   213,   214,   215,   216,   434,   217,   414,   415,   416,
+   218,   219,   220,   221,   222,   223,   224,   225,   226,   227,
+   228,   229,   230,   231,   232,   233,   234,   235,   236,   237,
+   238,   239,   240,   241,   242,   330,   435,   742
+};
+
+static const short yypact[] = {   353,
+-32768,-32768,   325,   -41,   376,   398,-32768,-32768,   164,   528,
+   436,-32768,-32768,-32768,-32768,   494,-32768,-32768,-32768,-32768,
+    10,-32768,-32768,-32768,    18,-32768,   297,-32768,    11,   576,
+-32768,-32768,   445,   598,-32768,   -41,   428,-32768,-32768,   690,
+-32768,   470,   -13,   -46,-32768,   495,    82,-32768,-32768,   -41,
+   612,   328,-32768,   381,-32768,    28,-32768,-32768,-32768,-32768,
+   -36,  1021,-32768,   531,   -13,-32768,-32768,   232,   577,-32768,
+-32768,   -13,   -46,-32768,    82,-32768,-32768,-32768,   590,-32768,
+-32768,-32768,   592,     2,-32768,-32768,   170,    -7,   916,-32768,
+-32768,    92,-32768,  1055,-32768,-32768,-32768,-32768,-32768,-32768,
+-32768,-32768,-32768,   172,   202,-32768,   -13,-32768,-32768,   262,
+    -1,-32768,   424,   -75,-32768,   504,    -1,     9,    86,   278,
+-32768,   615,   617,-32768,-32768,-32768,-32768,   637,   936,-32768,
+-32768,   202,   729,   653,   117,-32768,-32768,-32768,   660,  2333,
+   132,   384,-32768,-32768,-32768,-32768,-32768,-32768,-32768,    -1,
+-32768,   655,    -1,-32768,-32768,   337,   377,-32768,   417,   916,
+-32768,  1001,-32768,-32768,-32768,-32768,    47,-32768,   406,-32768,
+   472,-32768,-32768,-32768,-32768,-32768,   667,   903,   551,-32768,
+-32768,-32768,   442,-32768,-32768,-32768,   460,-32768,-32768,  3080,
+  3145,  3196,  3261,   466,    21,   497,-32768,  3312,  3377,  3428,
+  5442,-32768,-32768,-32768,-32768,-32768,-32768,-32768,   543,   915,
+    99,-32768,   548,   560,-32768,   712,-32768,   770,-32768,   836,
+   888,-32768,-32768,-32768,-32768,-32768,-32768,-32768,   619,   945,
+  1019,   937,   935,   669,   676,   694,   714,   258,-32768,-32768,
+-32768,   872,-32768,-32768,-32768,-32768,-32768,-32768,-32768,-32768,
+  2125,   731,   472,-32768,-32768,-32768,   128,   634,   668,-32768,
+   873,   719,-32768,   585,   721,-32768,   673,-32768,-32768,-32768,
+-32768,-32768,-32768,-32768,-32768,-32768,    25,-32768,   724,   992,
+   643,   643,   290,-32768,-32768,-32768,-32768,-32768,   746,   962,
+   208,-32768,   654,-32768,   358,   331,  5509,  3493,   622,-32768,
+   427,  3544,-32768,   393,-32768,-32768,  3609,  3660,  3725,  3776,
+  3841,  3892,  3957,  4008,  4073,  4124,  4189,  4240,   525,  4305,
+  4356,  4421,  4472,  4537,  4588,  4653,  2384,-32768,-32768,  4704,
+   510,   415,   416,  4769,-32768,   182,  2449,    13,   418,   426,
+    26,   450,   204,   679,  5741,-32768,   -41,   618,   844,   448,
+   827,-32768,   684,-32768,  1785,-32768,-32768,   685,-32768,-32768,
+-32768,  2193,-32768,-32768,   689,-32768,-32768,-32768,  2193,-32768,
+  2193,-32768,-32768,  5792,   697,-32768,-32768,-32768,-32768,-32768,
+   469,-32768,   702,   711,   888,   947,   965,-32768,-32768,-32768,
+-32768,   759,   668,-32768,-32768,   728,-32768,  1853,   732,-32768,
+   873,-32768,-32768,-32768,-32768,-32768,-32768,   123,   740,-32768,
+-32768,-32768,  2500,   643,-32768,   493,   643,   493,-32768,  2565,
+  4820,   166,   -30,  2616,   308,-32768,  1180,-32768,-32768,  1645,
+-32768,-32768,-32768,   781,-32768,-32768,   239,-32768,-32768,-32768,
+-32768,-32768,   772,-32768,   244,-32768,  5560,-32768,-32768,-32768,
+-32768,-32768,-32768,-32768,   619,-32768,   619,-32768,   945,-32768,
+   945,-32768,   945,-32768,  1019,-32768,  1019,-32768,  1019,-32768,
+  1019,-32768,     2,-32768,-32768,   937,-32768,   937,-32768,   935,
+-32768,   669,-32768,   676,-32768,   694,-32768,   714,-32768,   766,
+   784,-32768,-32768,-32768,-32768,-32768,-32768,  4885,-32768,   143,
+-32768,-32768,   153,-32768,-32768,   180,-32768,  4936,-32768,  5001,
+-32768,   681,-32768,  5350,-32768,-32768,   235,   242,   783,  2681,
+   620,-32768,-32768,   -41,  2732,-32768,-32768,-32768,-32768,  1070,
+-32768,-32768,-32768,   821,   984,   903,-32768,-32768,-32768,   104,
+  2797,-32768,  5052,-32768,   802,-32768,-32768,  1037,  1921,   812,
+  1989,-32768,-32768,-32768,  5625,-32768,-32768,   259,-32768,   826,
+   295,-32768,   826,-32768,-32768,   202,    61,-32768,-32768,-32768,
+  5117,   817,-32768,-32768,-32768,-32768,-32768,  5168,-32768,-32768,
+  5676,-32768,-32768,   202,   785,-32768,  5233,-32768,   252,-32768,
+-32768,-32768,-32768,-32768,-32768,-32768,   253,-32768,   845,   471,
+    46,   681,-32768,-32768,-32768,-32768,-32768,-32768,   473,   783,
+-32768,   941,   122,   851,-32768,  5484,-32768,  5375,   855,   865,
+   867,-32768,   645,-32768,  2057,  2848,-32768,-32768,   792,-32768,
+-32768,-32768,-32768,-32768,-32768,   202,-32768,-32768,-32768,-32768,
+-32768,-32768,-32768,   814,-32768,   202,-32768,-32768,-32768,  2259,
+-32768,-32768,    87,-32768,   788,-32768,-32768,-32768,-32768,-32768,
+  2913,-32768,-32768,   972,  5741,-32768,-32768,-32768,   885,   884,
+  5417,-32768,   206,   478,-32768,   245,   134,-32768,-32768,-32768,
+-32768,   481,-32768,   931,   934,  2259,-32768,-32768,  2259,-32768,
+-32768,   897,-32768,   733,-32768,   905,   910,-32768,  1016,   138,
+-32768,   929,  2193,-32768,   939,-32768,-32768,  2964,-32768,   257,
+  4885,  2193,-32768,-32768,  3029,   167,  5284,-32768,   739,-32768,
+  1484,-32768,-32768,   406,-32768,-32768,  1024,   944,-32768,  2193,
+   273,   150,-32768,   269,-32768,  5375,   948,-32768,-32768,-32768,
+-32768,   212,-32768,-32768,  1571,-32768,  1717,-32768,-32768,-32768,
+-32768,-32768,-32768,   276,  2259,   946,  5417,-32768,-32768,-32768,
+-32768,   974,  2259,   955,  2259,-32768,  2259,-32768,-32768,  1038,
+  1052,-32768
+};
+
+static const short yypgoto[] = {-32768,
+-32768,   -58,    14,   737,     4,   -99,   674,-32768,    -3,   677,
+-32768,    79,-32768,  1047,   760,-32768,   266,-32768,-32768,   881,
+     7,    23,-32768,-32768,  1007,   997,-32768,  -131,-32768,   912,
+-32768,   281,  -121,   928,   431,  -195,-32768,-32768,   408,   509,
+   832,  -259,   -73,-32768,-32768,-32768,-32768,-32768,   918,   686,
+-32768,   700,   -34,-32768,-32768,-32768,-32768,  1040,   508,-32768,
+  1006,-32768,-32768,   500,-32768,  -120,-32768,  -221,  -340,-32768,
+   727,  -299,   101,  -526,-32768,  -325,-32768,-32768,-32768,  -336,
+-32768,-32768,-32768,-32768,-32768,-32768,   389,   391,  -666,  -300,
+-32768,-32768,-32768,-32768,-32768,-32768,-32768,  -280,-32768,  -297,
+   741,-32768,-32768,-32768,-32768,-32768,-32768,-32768,-32768,   511,
+   517,-32768,-32768,   -31,-32768,  -369,-32768,   813,   219,  -269,
+  1157,   133,  1187,   279,   423,   486,   -20,   569,   607,  -419,
+-32768,   703,   819,   534,   699,   799,   801,   803,   800,   816,
+-32768,   552,   815,   695,-32768,-32768,   -51,-32768
+};
+
+
+#define        YYLAST          5901
+
+
+static const short yytable[] = {    21,
+   174,   152,    29,    83,   403,   294,    16,   573,   173,   144,
+    38,    48,   418,   278,   527,    16,    16,    16,    41,   423,
+   425,   278,    17,   138,   139,   406,   511,    93,    76,   355,
+   123,    17,    17,    17,   182,    83,    16,   538,    61,    16,
+    16,   173,    61,   143,    20,    59,   656,   256,   258,    67,
+   567,   134,    17,    64,   748,    17,    17,    16,    88,    93,
+    61,   635,   531,   571,    50,    61,   561,    59,    89,   533,
+    83,   534,    59,    17,   177,    84,   247,   585,   748,   249,
+    42,    62,    25,    27,    90,    88,   146,   693,   243,   120,
+    88,   407,   124,    50,   172,   135,   283,   259,   118,   300,
+    89,   123,    84,    83,   615,  -154,   145,    84,    39,    49,
+    40,    50,  -155,   279,    52,    54,    90,   184,    57,   262,
+   157,   279,   663,   685,    61,    88,    77,   172,    57,    88,
+    61,    59,   244,    69,   663,   160,   396,    67,   726,   178,
+   157,  -103,    84,   590,   560,  -103,    84,   563,   291,   243,
+   726,   161,   640,   592,   636,   613,    88,   110,    88,   685,
+   578,   113,   685,   -30,     1,   405,   570,   738,   160,   270,
+   272,   274,   276,    84,    88,    84,   551,   285,   287,   393,
+   594,   694,   501,   147,   161,   629,   267,   267,   267,   267,
+   125,    84,    61,   177,   267,   267,   290,   -93,   -93,   282,
+   110,   113,  -289,   616,   515,   177,   706,     2,   426,   281,
+   527,   644,   758,   289,   185,   664,   553,   187,     3,   383,
+   512,   578,   157,   554,     4,   519,   394,   710,   685,   245,
+     5,   727,   101,   578,   576,   606,   685,   578,   685,   579,
+   685,   591,   426,   754,   582,   662,   437,   349,    88,   578,
+   445,   593,   649,   651,   739,   187,   677,   701,   178,   630,
+    88,    61,     7,   557,   348,    84,   119,    20,    67,   649,
+   178,   128,   326,   725,    31,   491,   750,    84,   595,   667,
+   502,   538,   500,   -66,   527,   506,   449,   451,   453,    20,
+   419,   700,   524,   518,   304,   632,   129,    45,    31,   759,
+   157,   427,   516,   267,   267,   267,   267,   267,   267,   267,
+   267,   267,   267,   267,   267,    88,   267,   267,   267,   267,
+   267,   267,   267,   383,   686,    19,   -66,   625,    71,   350,
+   383,   431,   473,   607,   538,   608,   580,   383,   732,   383,
+   187,   583,   383,   709,   327,   650,   652,    88,   -65,   689,
+   683,   349,   -28,     1,   133,   733,   631,   409,   530,    46,
+   686,   558,   755,   686,    84,   530,   383,   530,   348,   692,
+   535,   753,   437,   705,   761,   148,    26,    57,   243,   443,
+   536,    74,   420,   384,   246,   689,   531,   348,   689,   533,
+    42,   -65,   633,   446,   548,   697,     2,   432,    28,   538,
+   569,   572,   610,   729,   561,   692,   527,     3,   692,   165,
+    20,   348,   735,     4,   503,  -317,   497,   267,   507,     5,
+   538,   517,  -157,   267,   137,   187,   509,     6,    53,   686,
+   752,  -138,   -65,   350,   634,   -29,     1,   686,   756,   686,
+   350,   686,   165,    46,   -32,     1,   589,   350,   522,   350,
+   513,     7,   645,   429,   689,   683,   597,   430,   599,   764,
+   137,    20,   689,   729,   689,   735,   689,   752,   437,   542,
+    58,   654,   250,   660,   692,   -65,   350,   524,   660,     2,
+   657,   497,   692,    20,   692,   447,   692,   384,     2,   619,
+     3,   621,   -25,   -25,   384,    66,     4,   280,   442,     3,
+   157,   384,     5,   384,   679,     4,   384,  -317,   498,   252,
+   508,     5,    20,    20,   681,   -96,   133,   383,   510,   383,
+   -96,   -96,   -96,   -96,   134,   472,   642,   -31,     1,   385,
+   384,    98,    88,   -25,     7,   523,   167,    35,   -25,   -25,
+   -25,   265,   514,     7,   -25,   349,    78,   349,   -25,    84,
+   638,   -96,   707,   -96,    36,    20,   -96,   -96,   -96,   -96,
+    37,   543,   348,   655,   348,   661,   277,   267,   267,   167,
+   708,     2,   100,   711,    78,   -34,     1,   102,    80,   106,
+    20,    81,    20,   267,   383,   402,   383,   201,     4,   561,
+   109,   117,   112,   383,     5,   140,   177,   -33,     1,    57,
+   141,   142,   187,   749,   494,   495,    80,   261,   496,    81,
+    20,   -35,     1,   702,   131,   149,    20,   151,   383,     2,
+   438,   349,   438,   307,   308,   309,     7,   350,     2,   350,
+   150,   153,   559,   385,    78,   559,     4,   154,   348,   383,
+   385,     2,     5,   296,   263,   438,   530,   385,   301,   385,
+   264,    88,   385,   181,   383,     2,   302,   383,     4,   734,
+   186,   178,    20,   737,     5,   741,    80,   260,    84,    81,
+    20,   383,     4,   386,     7,    20,   385,    20,     5,   322,
+   383,   384,   530,   384,   609,   530,   439,   323,   439,   383,
+    55,   440,   401,   440,    56,   441,     7,   441,   383,   530,
+    20,    57,    20,   350,   383,    20,   324,    20,   530,   674,
+     7,   439,   303,   383,   118,   383,   440,   349,   296,   400,
+   441,   404,    20,   383,   408,   383,   530,   325,   350,   175,
+    20,   383,   397,   383,   348,   383,   387,    99,    20,   413,
+   600,   349,   103,   349,   601,    20,  -335,  -335,   384,   428,
+   384,   530,    20,   248,   139,  -337,  -337,   384,   348,   530,
+   348,   530,   398,   530,   350,   297,   586,   350,    30,   298,
+    34,  -153,     2,   299,     2,    20,   525,   386,    78,   716,
+    78,   350,   384,   528,   386,   716,   713,   532,   696,   714,
+   350,   386,    51,   386,   717,   541,   386,    20,  -335,   350,
+   717,   155,  -335,   384,    20,   159,    20,  -337,   350,    20,
+    80,  -337,    80,    81,    20,    81,    20,   639,   384,   388,
+   386,   384,   176,   350,   391,   350,   546,   385,   718,   385,
+   552,     2,   555,   350,   743,   384,   159,    78,   159,   421,
+   387,   350,   422,   350,   384,   350,   296,   387,   465,   467,
+   469,   471,   544,   384,   387,   762,   387,   389,   264,   387,
+  -491,  -491,   384,   766,   581,   768,    78,   769,   384,    80,
+    35,   587,    81,    20,   577,   194,    78,   384,   646,   384,
+   578,   195,   139,   387,   578,   678,   614,   384,   196,   384,
+    32,   578,   197,    37,   385,   384,   385,   384,    80,   384,
+   622,    81,    20,   385,   626,   198,   199,   680,    80,   200,
+    32,    81,    20,   578,    32,   562,   121,   564,   202,   203,
+   204,   205,   561,   388,   206,   207,  -492,  -492,   385,   -15,
+   388,    32,   305,   306,  -490,  -490,   297,   388,   653,   388,
+   520,   662,   388,   665,   521,   390,    35,   310,   311,   385,
+   320,   321,    78,   671,   315,   316,   317,   318,   672,    35,
+   673,   389,   328,   329,   385,    78,   388,   385,   389,   141,
+   142,   386,   701,   386,   122,   389,    36,   389,   703,   156,
+   389,   385,    37,   616,    80,    78,   712,    81,    20,  -204,
+   385,  -401,  -401,   319,    79,   715,     4,    80,   723,   385,
+    81,    20,     5,   724,   389,  -490,  -490,   297,   385,  -402,
+  -402,   298,   455,   457,   385,   299,   725,    80,   476,   478,
+    81,    20,   728,   385,   750,   385,   312,   313,   314,   765,
+   157,   158,   730,   385,   387,   385,   387,   771,   386,   763,
+   386,   385,   751,   385,   156,   385,   757,   386,   767,   390,
+    78,   772,  -490,  -490,   297,   474,   390,    33,   424,    79,
+    75,     4,   299,   390,     2,   390,   188,     5,   390,   -15,
+    78,   108,   386,   255,  -490,  -490,   297,   253,   545,    79,
+   520,     4,    80,   392,   299,    81,    20,     5,   410,   411,
+   412,    73,   390,   386,   417,   157,   254,   550,     2,   127,
+   537,   387,    80,   387,    78,    81,    20,   744,   386,   745,
+   387,   386,   658,    79,   540,     4,    82,   388,   659,   388,
+   480,     5,   -15,   482,   486,   386,   484,  -490,  -490,   297,
+   459,   461,   463,   520,   386,   387,    80,   623,   648,    81,
+    20,   488,     0,   386,   493,     0,     0,     0,     0,     0,
+   126,     0,   386,     0,     0,   389,   387,   389,   386,     0,
+  -490,  -490,   297,     0,     0,     0,   298,   386,     0,   386,
+   521,   387,     0,     0,   387,     0,     0,   386,     0,   386,
+     0,     0,     0,     0,   388,   386,   388,   386,   387,   386,
+     0,     0,     0,   388,     0,     0,     0,   387,     0,     0,
+     0,     0,     0,     0,     0,     0,   387,     0,     0,     0,
+     0,     0,     0,     0,     0,   387,     0,     0,   388,     0,
+     0,   387,   389,     0,   389,     0,     0,     0,     0,    78,
+   387,   389,   387,     0,     0,     0,     0,     0,   194,   388,
+   387,     0,   387,   390,   195,   390,     0,     0,   387,     0,
+   387,   196,   387,     0,   388,   197,   389,   388,     0,     0,
+     0,    80,     0,     0,    81,    20,     0,     0,   198,   199,
+     0,   388,   200,     0,     0,     0,     0,   389,     0,     0,
+   388,   202,   203,   204,   205,     0,     0,   206,   207,   388,
+     0,     0,   389,     0,     0,   389,     0,     0,   388,     0,
+     0,     0,     0,     0,   388,     0,     0,     0,     0,   389,
+   390,     0,   390,   388,     0,   388,     0,     0,   389,   390,
+     0,     0,     0,   388,     0,   388,     0,   389,     0,     0,
+     0,   388,     0,   388,     0,   388,   389,     0,     0,     0,
+     0,     0,   389,     0,   390,     0,   268,   268,   268,   268,
+     0,   389,     0,   389,   268,   268,     0,     0,     0,     0,
+     0,   389,     0,   389,     0,   390,     0,     0,     0,   389,
+     0,   389,     0,   389,     0,     0,   269,   269,   269,   269,
+   390,     0,     0,   390,   269,   269,     0,     0,     0,     0,
+     0,     0,     0,     0,     0,     0,     0,   390,     0,     0,
+     0,     0,     0,     0,     0,     0,   390,     0,     0,     0,
+     0,     0,     0,     0,     0,   390,     0,     0,     0,     0,
+     0,     0,     0,     0,   390,     0,     0,     0,     0,     0,
+   390,     0,     0,     0,     0,     0,     0,     0,     0,   390,
+     0,   390,     0,     0,     0,     0,     0,     0,     0,   390,
+     0,   390,     0,     0,     0,     0,     0,   390,     0,   390,
+     0,   390,     0,   268,   268,   268,   268,   268,   268,   268,
+   268,   268,   268,   268,   268,     0,   268,   268,   268,   268,
+   268,   268,   268,     0,   331,     0,     0,     0,     0,     0,
+     0,     0,     0,   269,   269,   269,   269,   269,   269,   269,
+   269,   269,   269,   269,   269,     0,   269,   269,   269,   269,
+   269,   269,   269,     0,     0,     0,     0,     0,     0,     0,
+     0,     0,     0,     0,     0,     0,     0,   332,   192,   193,
+   716,   333,   334,    78,   335,     0,     0,   336,     0,     0,
+     0,   337,   194,     0,     0,   717,     0,     0,   338,   339,
+     5,   340,     0,   341,   342,   196,   343,     0,     0,   344,
+     0,     0,     0,     0,     0,    80,     0,     0,    81,    20,
+     0,   331,     0,     0,     0,     0,   345,   268,   157,   746,
+     0,     0,   346,   268,     0,   202,   203,   204,   205,     0,
+     0,   206,   207,     0,     0,     0,     0,     0,     0,     0,
+     0,     0,     0,     0,     0,     0,     0,   269,     0,     0,
+     0,     0,     0,   269,   332,   192,   193,   716,   333,   334,
+    78,   335,     0,     0,   336,     0,     0,     0,   337,   194,
+     0,     0,   717,     0,     0,   338,   339,     5,   340,     0,
+   341,   342,   196,   343,     0,   574,   344,   190,   191,     0,
+     0,     0,    80,     0,     0,    81,    20,     0,     0,     0,
+     0,     0,     0,   345,     0,   157,   760,     0,     0,   346,
+     0,     0,   202,   203,   204,   205,     0,     0,   206,   207,
+     0,     0,     0,     0,     0,     0,     0,     0,     0,   192,
+   193,     0,     0,     0,    78,     0,     0,     0,     0,     0,
+     0,     0,     0,   194,     0,     0,     0,     0,     0,   195,
+     0,     0,     0,     0,     0,     0,   196,   331,     0,     0,
+   197,     0,     0,     0,     0,     0,    80,   268,   268,    81,
+    20,     0,     0,   198,   199,     0,     0,   200,     0,   201,
+   575,     0,     0,   268,     0,     0,   202,   203,   204,   205,
+     0,     0,   206,   207,     0,     0,     0,   269,   269,     0,
+   332,   192,   193,  -261,   333,   334,    78,   335,     0,     0,
+   336,     0,     0,   269,   337,   194,     0,     0,  -261,     0,
+     0,   338,   339,     5,   340,   331,   341,   342,   196,   343,
+     0,     0,   344,     0,     0,     0,     0,     0,    80,     0,
+     0,    81,    20,     0,     0,     0,     0,     0,     0,   345,
+     0,   157,  -261,     0,     0,   346,     0,     0,   202,   203,
+   204,   205,     0,     0,   206,   207,     0,     0,   332,   192,
+   193,     0,   333,   334,    78,   335,     0,     0,   336,     0,
+     0,     0,   337,   194,     0,     0,     0,     0,     0,   338,
+   339,     5,   340,   331,   341,   342,   196,   343,     0,     0,
+   344,     0,     0,     0,     0,     0,    80,     0,     0,    81,
+    20,     0,     0,     0,     0,     0,     0,   345,     0,   157,
+   526,     0,     0,   346,     0,     0,   202,   203,   204,   205,
+     0,     0,   206,   207,     0,     0,   332,   192,   193,     0,
+   333,   334,    78,   335,     0,     0,   336,     0,     0,     0,
+   337,   194,     0,     0,     0,     0,     0,   338,   339,     5,
+   340,   331,   341,   342,   196,   343,     0,     0,   344,     0,
+     0,     0,     0,     0,    80,     0,     0,    81,    20,     0,
+     0,     0,     0,     0,     0,   345,     0,   157,   547,     0,
+     0,   346,     0,     0,   202,   203,   204,   205,     0,     0,
+   206,   207,     0,     0,   332,   192,   193,     0,   333,   334,
+    78,   335,     0,     0,   336,     0,     0,     0,   337,   194,
+     0,     0,     0,     0,     0,   338,   339,     5,   340,   331,
+   341,   342,   196,   343,     0,     0,   344,     0,     0,     0,
+     0,     0,    80,     0,     0,    81,    20,     0,     0,     0,
+     0,     0,     0,   345,     0,   157,   624,     0,     0,   346,
+     0,     0,   202,   203,   204,   205,     0,     0,   206,   207,
+     0,     0,   332,   192,   193,     0,   333,   334,    78,   335,
+     0,     0,   336,     0,     0,     0,   337,   194,     0,     0,
+     0,     0,     0,   338,   339,     5,   340,   331,   341,   342,
+   196,   343,     0,     0,   344,     0,     0,     0,     0,     0,
+    80,     0,     0,    81,    20,     0,     0,     0,     0,     0,
+     0,   345,     0,   157,   627,     0,     0,   346,     0,     0,
+   202,   203,   204,   205,     0,     0,   206,   207,     0,     0,
+   332,   192,   193,     0,   333,   334,    78,   335,     0,     0,
+   336,     0,     0,     0,   337,   194,     0,     0,     0,     0,
+     0,   338,   339,     5,   340,   331,   341,   342,   196,   343,
+     0,     0,   344,     0,     0,     0,     0,     0,    80,     0,
+     0,    81,    20,     0,     0,     0,     0,     0,     0,   345,
+     0,   157,   675,     0,     0,   346,     0,     0,   202,   203,
+   204,   205,     0,     0,   206,   207,     0,     0,   332,   192,
+   193,     0,   333,   334,    78,   335,     0,     0,   336,     0,
+     0,     0,   337,   194,     0,     0,     0,     0,     0,   338,
+   339,     5,   340,   331,   341,   342,   196,   343,     0,     0,
+   344,     0,     0,     0,     0,     0,    80,     0,     0,    81,
+    20,     0,     0,     0,     0,     0,     0,   345,     0,   157,
+     0,     0,     0,   346,     0,     0,   202,   203,   204,   205,
+     0,     0,   206,   207,     0,     0,   529,   192,   193,     0,
+   333,   334,    78,   335,     0,     0,   336,     0,     0,     0,
+   337,   194,     0,     0,     0,     0,     0,   338,   339,   331,
+   340,     0,   341,   342,   196,   343,     0,     0,   344,     0,
+     0,     0,     0,     0,    80,     0,     0,    81,    20,     0,
+     0,     0,     0,     0,     0,   345,     0,   157,     0,     0,
+     0,   346,     0,     0,   202,   203,   204,   205,     0,     0,
+   206,   207,   529,   192,   193,     0,   682,   334,    78,   335,
+     0,     0,   336,     0,     0,     0,   337,   194,     0,     0,
+     0,     0,     0,   338,   339,     0,   340,     0,   341,   342,
+   196,   343,     0,   189,   344,   190,   191,     0,     0,     0,
+    80,     0,     0,    81,    20,     0,     0,     0,     0,     0,
+     0,   345,     0,   157,     0,     0,     0,   346,     0,     0,
+   202,   203,   204,   205,     0,     0,   206,   207,     0,     0,
+     0,     0,     0,     0,     0,     0,     0,   192,   193,     0,
+     0,     0,    78,     0,   489,     0,   190,   191,     0,     0,
+     0,   194,     0,     0,     0,     0,     0,   195,     0,     0,
+     0,     0,     0,     0,   196,     0,     0,     0,   197,     0,
+     0,     0,     0,     0,    80,     0,     0,    81,    20,     0,
+     0,   198,   199,     0,     0,   200,     0,   201,   192,   193,
+     0,     0,     0,    78,   202,   203,   204,   205,     0,     0,
+   206,   207,   194,     0,     0,     0,     0,     0,   195,   504,
+     0,   190,   191,     0,     0,   196,     0,     0,     0,   197,
+     0,     0,     0,     0,     0,    80,     0,     0,    81,    20,
+     0,   490,   198,   199,     0,     0,   200,     0,     0,     0,
+     0,     0,     0,     0,     0,   202,   203,   204,   205,     0,
+     0,   206,   207,   192,   193,     0,     0,     0,    78,     0,
+   556,     0,   190,   191,     0,     0,     0,   194,     0,     0,
+     0,     0,     0,   195,     0,     0,     0,     0,     0,     0,
+   196,     0,     0,     0,   197,     0,     0,     0,     0,     0,
+    80,     0,     0,    81,    20,     0,     0,   198,   199,     0,
+     0,   200,     0,     0,   192,   193,     0,   505,     0,    78,
+   202,   203,   204,   205,     0,     0,   206,   207,   194,     0,
+     0,     0,     0,     0,   195,   565,     0,   190,   191,     0,
+     0,   196,     0,     0,     0,   197,     0,     0,     0,     0,
+     0,    80,     0,     0,    81,    20,     0,     0,   198,   199,
+     0,     0,   200,     0,     0,     0,     0,   557,     0,     0,
+     0,   202,   203,   204,   205,     0,     0,   206,   207,   192,
+   193,     0,     0,     0,    78,     0,   436,     0,   190,   191,
+     0,     0,     0,   194,     0,     0,     0,     0,     0,   195,
+     0,     0,     0,     0,     0,     0,   196,     0,     0,     0,
+   197,     0,     0,     0,     0,     0,    80,     0,     0,    81,
+    20,     0,     0,   198,   199,     0,     0,   200,   566,     0,
+   192,   193,     0,     0,     0,    78,   202,   203,   204,   205,
+     0,     0,   206,   207,   194,     0,     0,     0,     0,     0,
+   195,   436,     0,   190,   191,     0,     0,   196,     0,     0,
+     0,   197,     0,     0,     0,     0,     0,    80,     0,     0,
+    81,    20,     0,     0,   198,   199,     0,     0,   200,     0,
+     0,     0,     0,   557,     0,     0,     0,   202,   203,   204,
+   205,     0,     0,   206,   207,   192,   193,     0,     0,     0,
+    78,     0,   611,     0,   190,   191,     0,     0,     0,   194,
+     0,     0,     0,     0,     0,   195,     0,     0,     0,     0,
+     0,     0,   196,     0,     0,     0,   197,     0,     0,     0,
+     0,     0,    80,     0,     0,    81,    20,     0,     0,   198,
+   199,     0,     0,   200,     0,     0,   192,   193,   148,     0,
+     0,    78,   202,   203,   204,   205,     0,     0,   206,   207,
+   194,     0,     0,     0,     0,     0,   195,   617,     0,   190,
+   191,     0,     0,   196,     0,     0,     0,   197,     0,     0,
+     0,     0,     0,    80,     0,     0,    81,    20,     0,     0,
+   198,   199,     0,     0,   200,   612,     0,     0,     0,     0,
+     0,     0,     0,   202,   203,   204,   205,     0,     0,   206,
+   207,   192,   193,     0,     0,     0,    78,     0,   611,     0,
+   190,   191,     0,     0,     0,   194,     0,     0,     0,     0,
+     0,   195,     0,     0,     0,     0,     0,     0,   196,     0,
+     0,     0,   197,     0,     0,     0,     0,     0,    80,     0,
+     0,    81,    20,     0,     0,   198,   199,     0,     0,   200,
+     0,     0,   192,   193,     0,   618,     0,    78,   202,   203,
+   204,   205,     0,     0,   206,   207,   194,     0,     0,     0,
+     0,     0,   195,   698,     0,   190,   191,     0,     0,   196,
+     0,     0,     0,   197,     0,     0,     0,     0,     0,    80,
+     0,     0,    81,    20,     0,     0,   198,   199,     0,     0,
+   200,   676,     0,     0,     0,     0,     0,     0,     0,   202,
+   203,   204,   205,     0,     0,   206,   207,   192,   193,     0,
+     0,     0,    78,     0,   698,     0,   190,   191,     0,     0,
+     0,   194,     0,     0,     0,     0,     0,   195,     0,     0,
+     0,     0,     0,     0,   196,     0,     0,     0,   197,     0,
+     0,     0,     0,     0,    80,     0,     0,    81,    20,     0,
+     0,   198,   199,     0,     0,   200,   699,     0,   192,   193,
+     0,     0,     0,    78,   202,   203,   204,   205,     0,     0,
+   206,   207,   194,     0,     0,     0,     0,     0,   195,   617,
+     0,   190,   191,     0,     0,   196,     0,     0,     0,   197,
+     0,     0,     0,     0,     0,    80,     0,     0,    81,    20,
+     0,     0,   198,   199,     0,     0,   200,   731,     0,     0,
+     0,     0,     0,     0,     0,   202,   203,   204,   205,     0,
+     0,   206,   207,   192,   193,     0,     0,     0,    78,     0,
+   266,     0,   190,   191,     0,     0,     0,   194,     0,     0,
+     0,     0,     0,   195,     0,     0,     0,     0,     0,     0,
+   196,     0,     0,     0,   197,     0,     0,     0,     0,     0,
+    80,     0,     0,    81,    20,     0,     0,   198,   199,     0,
+     0,   200,     0,     0,   192,   193,     0,   736,     0,    78,
+   202,   203,   204,   205,     0,     0,   206,   207,   194,     0,
+     0,     0,     0,     0,   195,   271,     0,   190,   191,     0,
+     0,   196,     0,     0,     0,   197,     0,     0,     0,     0,
+     0,    80,     0,     0,    81,    20,     0,     0,   198,   199,
+     0,     0,   200,     0,     0,     0,     0,     0,     0,     0,
+     0,   202,   203,   204,   205,     0,     0,   206,   207,   192,
+   193,     0,     0,     0,    78,     0,   273,     0,   190,   191,
+     0,     0,     0,   194,     0,     0,     0,     0,     0,   195,
+     0,     0,     0,     0,     0,     0,   196,     0,     0,     0,
+   197,     0,     0,     0,     0,     0,    80,     0,     0,    81,
+    20,     0,     0,   198,   199,     0,     0,   200,     0,     0,
+   192,   193,     0,     0,     0,    78,   202,   203,   204,   205,
+     0,     0,   206,   207,   194,     0,     0,     0,     0,     0,
+   195,   275,     0,   190,   191,     0,     0,   196,     0,     0,
+     0,   197,     0,     0,     0,     0,     0,    80,     0,     0,
+    81,    20,     0,     0,   198,   199,     0,     0,   200,     0,
+     0,     0,     0,     0,     0,     0,     0,   202,   203,   204,
+   205,     0,     0,   206,   207,   192,   193,     0,     0,     0,
+    78,     0,   284,     0,   190,   191,     0,     0,     0,   194,
+     0,     0,     0,     0,     0,   195,     0,     0,     0,     0,
+     0,     0,   196,     0,     0,     0,   197,     0,     0,     0,
+     0,     0,    80,     0,     0,    81,    20,     0,     0,   198,
+   199,     0,     0,   200,     0,     0,   192,   193,     0,     0,
+     0,    78,   202,   203,   204,   205,     0,     0,   206,   207,
+   194,     0,     0,     0,     0,     0,   195,   286,     0,   190,
+   191,     0,     0,   196,     0,     0,     0,   197,     0,     0,
+     0,     0,     0,    80,     0,     0,    81,    20,     0,     0,
+   198,   199,     0,     0,   200,     0,     0,     0,     0,     0,
+     0,     0,     0,   202,   203,   204,   205,     0,     0,   206,
+   207,   192,   193,     0,     0,     0,    78,     0,   288,     0,
+   190,   191,     0,     0,     0,   194,     0,     0,     0,     0,
+     0,   195,     0,     0,     0,     0,     0,     0,   196,     0,
+     0,     0,   197,     0,     0,     0,     0,     0,    80,     0,
+     0,    81,    20,     0,     0,   198,   199,     0,     0,   200,
+     0,     0,   192,   193,     0,     0,     0,    78,   202,   203,
+   204,   205,     0,     0,   206,   207,   194,     0,     0,     0,
+     0,     0,   195,   436,     0,   190,   191,     0,     0,   196,
+     0,     0,     0,   197,     0,     0,     0,     0,     0,    80,
+     0,     0,    81,    20,     0,     0,   198,   199,     0,     0,
+   200,     0,     0,     0,     0,     0,     0,     0,     0,   202,
+   203,   204,   205,     0,     0,   206,   207,   192,   193,     0,
+     0,     0,    78,     0,   444,     0,   190,   191,     0,     0,
+     0,   194,     0,     0,     0,     0,     0,   195,     0,     0,
+     0,     0,     0,     0,   196,     0,     0,     0,   197,     0,
+     0,     0,     0,     0,    80,     0,     0,    81,    20,     0,
+     0,   198,   199,     0,     0,   200,     0,     0,   192,   193,
+     0,     0,     0,    78,   202,   203,   204,   205,     0,     0,
+   206,   207,   194,     0,     0,     0,     0,     0,   195,   448,
+     0,   190,   191,     0,     0,   196,     0,     0,     0,   197,
+     0,     0,     0,     0,     0,    80,     0,     0,    81,    20,
+     0,     0,   198,   199,     0,     0,   200,     0,     0,     0,
+     0,     0,     0,     0,     0,   202,   203,   204,   205,     0,
+     0,   206,   207,   192,   193,     0,     0,     0,    78,     0,
+   450,     0,   190,   191,     0,     0,     0,   194,     0,     0,
+     0,     0,     0,   195,     0,     0,     0,     0,     0,     0,
+   196,     0,     0,     0,   197,     0,     0,     0,     0,     0,
+    80,     0,     0,    81,    20,     0,     0,   198,   199,     0,
+     0,   200,     0,     0,   192,   193,     0,     0,     0,    78,
+   202,   203,   204,   205,     0,     0,   206,   207,   194,     0,
+     0,     0,     0,     0,   195,   452,     0,   190,   191,     0,
+     0,   196,     0,     0,     0,   197,     0,     0,     0,     0,
+     0,    80,     0,     0,    81,    20,     0,     0,   198,   199,
+     0,     0,   200,     0,     0,     0,     0,     0,     0,     0,
+     0,   202,   203,   204,   205,     0,     0,   206,   207,   192,
+   193,     0,     0,     0,    78,     0,   454,     0,   190,   191,
+     0,     0,     0,   194,     0,     0,     0,     0,     0,   195,
+     0,     0,     0,     0,     0,     0,   196,     0,     0,     0,
+   197,     0,     0,     0,     0,     0,    80,     0,     0,    81,
+    20,     0,     0,   198,   199,     0,     0,   200,     0,     0,
+   192,   193,     0,     0,     0,    78,   202,   203,   204,   205,
+     0,     0,   206,   207,   194,     0,     0,     0,     0,     0,
+   195,   456,     0,   190,   191,     0,     0,   196,     0,     0,
+     0,   197,     0,     0,     0,     0,     0,    80,     0,     0,
+    81,    20,     0,     0,   198,   199,     0,     0,   200,     0,
+     0,     0,     0,     0,     0,     0,     0,   202,   203,   204,
+   205,     0,     0,   206,   207,   192,   193,     0,     0,     0,
+    78,     0,   458,     0,   190,   191,     0,     0,     0,   194,
+     0,     0,     0,     0,     0,   195,     0,     0,     0,     0,
+     0,     0,   196,     0,     0,     0,   197,     0,     0,     0,
+     0,     0,    80,     0,     0,    81,    20,     0,     0,   198,
+   199,     0,     0,   200,     0,     0,   192,   193,     0,     0,
+     0,    78,   202,   203,   204,   205,     0,     0,   206,   207,
+   194,     0,     0,     0,     0,     0,   195,   460,     0,   190,
+   191,     0,     0,   196,     0,     0,     0,   197,     0,     0,
+     0,     0,     0,    80,     0,     0,    81,    20,     0,     0,
+   198,   199,     0,     0,   200,     0,     0,     0,     0,     0,
+     0,     0,     0,   202,   203,   204,   205,     0,     0,   206,
+   207,   192,   193,     0,     0,     0,    78,     0,   462,     0,
+   190,   191,     0,     0,     0,   194,     0,     0,     0,     0,
+     0,   195,     0,     0,     0,     0,     0,     0,   196,     0,
+     0,     0,   197,     0,     0,     0,     0,     0,    80,     0,
+     0,    81,    20,     0,     0,   198,   199,     0,     0,   200,
+     0,     0,   192,   193,     0,     0,     0,    78,   202,   203,
+   204,   205,     0,     0,   206,   207,   194,     0,     0,     0,
+     0,     0,   195,   464,     0,   190,   191,     0,     0,   196,
+     0,     0,     0,   197,     0,     0,     0,     0,     0,    80,
+     0,     0,    81,    20,     0,     0,   198,   199,     0,     0,
+   200,     0,     0,     0,     0,     0,     0,     0,     0,   202,
+   203,   204,   205,     0,     0,   206,   207,   192,   193,     0,
+     0,     0,    78,     0,   466,     0,   190,   191,     0,     0,
+     0,   194,     0,     0,     0,     0,     0,   195,     0,     0,
+     0,     0,     0,     0,   196,     0,     0,     0,   197,     0,
+     0,     0,     0,     0,    80,     0,     0,    81,    20,     0,
+     0,   198,   199,     0,     0,   200,     0,     0,   192,   193,
+     0,     0,     0,    78,   202,   203,   204,   205,     0,     0,
+   206,   207,   194,     0,     0,     0,     0,     0,   195,   468,
+     0,   190,   191,     0,     0,   196,     0,     0,     0,   197,
+     0,     0,     0,     0,     0,    80,     0,     0,    81,    20,
+     0,     0,   198,   199,     0,     0,   200,     0,     0,     0,
+     0,     0,     0,     0,     0,   202,   203,   204,   205,     0,
+     0,   206,   207,   192,   193,     0,     0,     0,    78,     0,
+   470,     0,   190,   191,     0,     0,     0,   194,     0,     0,
+     0,     0,     0,   195,     0,     0,     0,     0,     0,     0,
+   196,     0,     0,     0,   197,     0,     0,     0,     0,     0,
+    80,     0,     0,    81,    20,     0,     0,   198,   199,     0,
+     0,   200,     0,     0,   192,   193,     0,     0,     0,    78,
+   202,   203,   204,   205,     0,     0,   206,   207,   194,     0,
+     0,     0,     0,     0,   195,   475,     0,   190,   191,     0,
+     0,   196,     0,     0,     0,   197,     0,     0,     0,     0,
+     0,    80,     0,     0,    81,    20,     0,     0,   198,   199,
+     0,     0,   200,     0,     0,     0,     0,     0,     0,     0,
+     0,   202,   203,   204,   205,     0,     0,   206,   207,   192,
+   193,     0,     0,     0,    78,     0,   477,     0,   190,   191,
+     0,     0,     0,   194,     0,     0,     0,     0,     0,   195,
+     0,     0,     0,     0,     0,     0,   196,     0,     0,     0,
+   197,     0,     0,     0,     0,     0,    80,     0,     0,    81,
+    20,     0,     0,   198,   199,     0,     0,   200,     0,     0,
+   192,   193,     0,     0,     0,    78,   202,   203,   204,   205,
+     0,     0,   206,   207,   194,     0,     0,     0,     0,     0,
+   195,   479,     0,   190,   191,     0,     0,   196,     0,     0,
+     0,   197,     0,     0,     0,     0,     0,    80,     0,     0,
+    81,    20,     0,     0,   198,   199,     0,     0,   200,     0,
+     0,     0,     0,     0,     0,     0,     0,   202,   203,   204,
+   205,     0,     0,   206,   207,   192,   193,     0,     0,     0,
+    78,     0,   481,     0,   190,   191,     0,     0,     0,   194,
+     0,     0,     0,     0,     0,   195,     0,     0,     0,     0,
+     0,     0,   196,     0,     0,     0,   197,     0,     0,     0,
+     0,     0,    80,     0,     0,    81,    20,     0,     0,   198,
+   199,     0,     0,   200,     0,     0,   192,   193,     0,     0,
+     0,    78,   202,   203,   204,   205,     0,     0,   206,   207,
+   194,     0,     0,     0,     0,     0,   195,   483,     0,   190,
+   191,     0,     0,   196,     0,     0,     0,   197,     0,     0,
+     0,     0,     0,    80,     0,     0,    81,    20,     0,     0,
+   198,   199,     0,     0,   200,     0,     0,     0,     0,     0,
+     0,     0,     0,   202,   203,   204,   205,     0,     0,   206,
+   207,   192,   193,     0,     0,     0,    78,     0,   485,     0,
+   190,   191,     0,     0,     0,   194,     0,     0,     0,     0,
+     0,   195,     0,     0,     0,     0,     0,     0,   196,     0,
+     0,     0,   197,     0,     0,     0,     0,     0,    80,     0,
+     0,    81,    20,     0,     0,   198,   199,     0,     0,   200,
+     0,     0,   192,   193,     0,     0,     0,    78,   202,   203,
+   204,   205,     0,     0,   206,   207,   194,     0,     0,     0,
+     0,     0,   195,   487,     0,   190,   191,     0,     0,   196,
+     0,     0,     0,   197,     0,     0,     0,     0,     0,    80,
+     0,     0,    81,    20,     0,     0,   198,   199,     0,     0,
+   200,     0,     0,     0,     0,     0,     0,     0,     0,   202,
+   203,   204,   205,     0,     0,   206,   207,   192,   193,     0,
+     0,     0,    78,     0,   492,     0,   190,   191,     0,     0,
+     0,   194,     0,     0,     0,     0,     0,   195,     0,     0,
+     0,     0,     0,     0,   196,     0,     0,     0,   197,     0,
+     0,     0,     0,     0,    80,     0,     0,    81,    20,     0,
+     0,   198,   199,     0,     0,   200,     0,     0,   192,   193,
+     0,     0,     0,    78,   202,   203,   204,   205,     0,     0,
+   206,   207,   194,     0,     0,     0,     0,     0,   195,   499,
+     0,   190,   191,     0,     0,   196,     0,     0,     0,   197,
+     0,     0,     0,     0,     0,    80,     0,     0,    81,    20,
+     0,     0,   198,   199,     0,     0,   200,     0,     0,     0,
+     0,     0,     0,     0,     0,   202,   203,   204,   205,     0,
+     0,   206,   207,   192,   193,     0,     0,     0,    78,     0,
+   568,     0,   190,   191,     0,     0,     0,   194,     0,     0,
+     0,     0,     0,   195,     0,     0,     0,     0,     0,     0,
+   196,     0,     0,     0,   197,     0,     0,     0,     0,     0,
+    80,     0,     0,    81,    20,     0,     0,   198,   199,     0,
+     0,   200,     0,     0,   192,   193,     0,     0,     0,    78,
+   202,   203,   204,   205,     0,     0,   206,   207,   194,     0,
+     0,     0,     0,     0,   195,   588,     0,   190,   191,     0,
+     0,   196,     0,     0,     0,   197,     0,     0,     0,     0,
+     0,    80,     0,     0,    81,    20,     0,     0,   198,   199,
+     0,     0,   200,     0,     0,     0,     0,     0,     0,     0,
+     0,   202,   203,   204,   205,     0,     0,   206,   207,   192,
+   193,     0,     0,     0,    78,     0,   596,     0,   190,   191,
+     0,     0,     0,   194,     0,     0,     0,     0,     0,   195,
+     0,     0,     0,     0,     0,     0,   196,     0,     0,     0,
+   197,     0,     0,     0,     0,     0,    80,     0,     0,    81,
+    20,     0,     0,   198,   199,     0,     0,   200,     0,     0,
+   192,   193,     0,     0,     0,    78,   202,   203,   204,   205,
+     0,     0,   206,   207,   194,     0,     0,     0,     0,     0,
+   195,   598,     0,   190,   191,     0,     0,   196,     0,     0,
+     0,   197,     0,     0,     0,     0,     0,    80,     0,     0,
+    81,    20,     0,     0,   198,   199,     0,     0,   200,     0,
+     0,     0,     0,     0,     0,     0,     0,   202,   203,   204,
+   205,     0,     0,   206,   207,   192,   193,     0,     0,     0,
+    78,     0,   620,     0,   190,   191,     0,     0,     0,   194,
+     0,     0,     0,     0,     0,   195,     0,     0,     0,     0,
+     0,     0,   196,     0,     0,     0,   197,     0,     0,     0,
+     0,     0,    80,     0,     0,    81,    20,     0,     0,   198,
+   199,     0,     0,   200,     0,     0,   192,   193,     0,     0,
+     0,    78,   202,   203,   204,   205,     0,     0,   206,   207,
+   194,     0,     0,     0,     0,     0,   195,   637,     0,   190,
+   191,     0,     0,   196,     0,     0,     0,   197,     0,     0,
+     0,     0,     0,    80,     0,     0,    81,    20,     0,     0,
+   198,   199,     0,     0,   200,     0,     0,     0,     0,     0,
+     0,     0,     0,   202,   203,   204,   205,     0,     0,   206,
+   207,   192,   193,     0,     0,     0,    78,     0,   641,     0,
+   190,   191,     0,     0,     0,   194,     0,     0,     0,     0,
+     0,   195,     0,     0,     0,     0,     0,     0,   196,     0,
+     0,     0,   197,     0,     0,     0,     0,     0,    80,     0,
+     0,    81,    20,     0,     0,   198,   199,     0,     0,   200,
+     0,     0,   192,   193,     0,     0,     0,    78,   202,   203,
+   204,   205,     0,     0,   206,   207,   194,     0,     0,     0,
+     0,     0,   195,   647,     0,   190,   191,     0,     0,   196,
+     0,     0,     0,   197,     0,     0,     0,     0,     0,    80,
+     0,     0,    81,    20,     0,     0,   198,   199,     0,     0,
+   200,     0,     0,     0,     0,     0,     0,     0,     0,   202,
+   203,   204,   205,     0,     0,   206,   207,   192,   193,     0,
+     0,     0,    78,     0,   740,     0,   190,   191,     0,     0,
+     0,   194,     0,     0,     0,     0,     0,   195,     0,     0,
+     0,     0,     0,     0,   196,     0,     0,     0,   197,     0,
+     0,     0,     0,     0,    80,     0,     0,    81,    20,     0,
+     0,   198,   199,     0,     0,   200,     0,     0,   192,   193,
+     0,     0,     0,    78,   202,   203,   204,   205,     0,     0,
+   206,   207,   194,     0,     0,     0,     0,     0,   195,     0,
+   605,     0,     0,     0,     0,   196,     0,     0,     0,   197,
+     0,     0,     0,     0,     0,    80,     0,     0,    81,    20,
+     0,     0,   198,   199,     0,   668,   200,     0,     0,     0,
+     0,     0,     0,     0,     0,   202,   203,   204,   205,     0,
+     0,   206,   207,  -284,  -284,  -284,     0,     0,     0,  -284,
+     0,     0,     0,     0,     0,     0,     0,     0,  -284,     0,
+     0,     0,     0,     0,  -284,     0,     0,   704,     0,   192,
+   193,  -284,     0,     0,    78,  -284,     0,     0,     0,     0,
+     0,  -284,     0,   194,  -284,  -284,     0,     0,     0,   195,
+     0,     0,  -284,     0,   190,   191,   196,     0,  -284,     0,
+   197,  -284,  -284,  -284,  -284,     0,    80,  -284,  -284,    81,
+    20,   192,   193,     0,     0,     0,    78,   345,  -292,     0,
+     0,     0,     0,     0,     0,   194,   202,   203,   204,   205,
+     0,   195,   206,   207,   666,     0,   192,   193,   196,     0,
+     0,    78,   197,     0,     0,     0,     0,     0,    80,     0,
+   194,    81,    20,     0,     0,     0,   195,     0,     0,   345,
+  -292,   190,   191,   196,     0,     0,     0,   197,   202,   203,
+   204,   205,     0,    80,   206,   207,    81,    20,   192,   193,
+   198,   199,     0,    78,   200,     0,   201,   292,     0,     0,
+     0,   293,   194,   202,   203,   204,   205,     0,   195,   206,
+   207,     0,     0,   192,   193,   196,     0,     0,    78,   197,
+     0,     0,   190,   191,     0,    80,     0,   194,    81,    20,
+     0,     0,     0,   195,     0,     0,   345,     0,     0,     0,
+   196,     0,     0,     0,   197,   202,   203,   204,   205,     0,
+    80,   206,   207,    81,    20,     0,     0,   198,   199,     0,
+     0,   200,   433,     0,   192,   193,     0,     0,     0,    78,
+   202,   203,   204,   205,     0,     0,   206,   207,   194,     0,
+     0,     0,     0,     0,   195,     0,     0,   190,   191,     0,
+     0,   196,     0,     0,     0,   197,     0,     0,     0,     0,
+     0,    80,     0,     0,    81,    20,     0,     0,   198,   199,
+     0,     0,   200,   584,     0,     0,     0,     0,     0,     0,
+     0,   202,   203,   204,   205,     0,     0,   206,   207,   192,
+   193,     0,     0,     0,    78,     0,     0,     0,   190,   191,
+     0,     0,     0,   194,     0,     0,     0,     0,     0,   195,
+     0,     0,     0,     0,     0,     0,   196,     0,     0,     0,
+   197,     0,     0,     0,     0,     0,    80,     0,     0,    81,
+    20,     0,     0,   198,   199,     0,     0,   200,   628,     0,
+   192,   193,     0,     0,     0,    78,   202,   203,   204,   205,
+     0,     0,   206,   207,   194,     0,     0,     0,     0,     0,
+   195,     0,     0,   190,   191,     0,     0,   196,     0,     0,
+     0,   197,     0,     0,     0,     0,     0,    80,     0,     0,
+    81,    20,     0,     0,   198,   199,     0,     0,   200,   643,
+     0,     0,     0,     0,     0,     0,     0,   202,   203,   204,
+   205,     0,     0,   206,   207,   192,   193,     0,     0,     0,
+    78,     0,     0,     0,     0,     0,     0,     0,     0,   194,
+     0,     0,     0,     0,     0,   195,     0,     0,     0,     0,
+     0,     0,   196,     0,     0,     0,   197,     0,     0,     0,
+     0,     0,    80,     0,     0,    81,    20,     0,     0,   198,
+   199,     0,     0,   200,     0,     2,   192,   193,     0,     0,
+     0,    78,   202,   203,   204,   205,     0,     0,   206,   207,
+   194,     0,     0,     0,     0,     0,   195,     0,     0,     0,
+     0,     0,     0,   196,     0,     0,     0,   197,     0,     0,
+     0,     0,     0,    80,     0,     0,    81,    20,     0,     0,
+     0,     0,     0,     0,   345,     0,     0,     0,     0,     0,
+     0,     0,     0,   202,   203,   204,   205,     0,     0,   206,
+   207
+};
+
+static const short yycheck[] = {     3,
+   132,   123,     6,    62,   264,   201,     0,   427,   129,     1,
+     1,     1,   282,     1,   355,     9,    10,    11,     1,   289,
+   290,     1,     0,    99,   100,     1,     1,    62,     1,   251,
+    89,     9,    10,    11,   134,    94,    30,   374,    42,    33,
+    34,   162,    46,   117,    86,    42,     1,     1,   169,    46,
+   420,    53,    30,   100,   721,    33,    34,    51,    62,    94,
+    64,     1,   362,    94,   101,    69,    97,    64,    62,   369,
+   129,   371,    69,    51,   133,    62,   150,   447,   745,   153,
+    63,    95,     4,     5,    62,    89,     1,     1,   140,    97,
+    94,    67,     1,   101,   129,    97,   196,   171,    97,     1,
+    94,   160,    89,   162,     1,    93,    98,    94,    99,    99,
+   101,   101,    95,   101,    36,    37,    94,     1,    40,   178,
+    95,   101,     1,   650,   128,   129,    99,   162,    50,   133,
+   134,   128,     1,    52,     1,   129,   257,   134,     1,   133,
+    95,    95,   129,     1,   414,    99,   133,   417,   200,   201,
+     1,   129,   572,     1,    94,   525,   160,    79,   162,   686,
+   100,    83,   689,     0,     1,   265,     1,     1,   162,   190,
+   191,   192,   193,   160,   178,   162,   398,   198,   199,   253,
+     1,    95,     1,    98,   162,   555,   190,   191,   192,   193,
+    99,   178,   196,   252,   198,   199,   200,    99,   100,   196,
+   122,   123,    99,   100,     1,   264,     1,    44,     1,   196,
+   551,   581,     1,   200,    98,    94,    94,   139,    55,   251,
+   341,   100,    95,   101,    61,   347,    99,    94,   755,    98,
+    67,    94,     1,   100,   430,     1,   763,   100,   765,     1,
+   767,    99,     1,    94,     1,     1,   298,   251,   252,   100,
+   302,    99,     1,     1,    88,   177,   626,     1,   252,     1,
+   264,   265,    99,    98,   251,   252,    97,    86,   265,     1,
+   264,   100,    15,     1,     9,   327,     1,   264,    99,   616,
+    99,   618,   334,    52,   625,   337,   307,   308,   309,    86,
+     1,   661,   351,   345,   216,     1,    95,     1,    33,    88,
+    95,    94,    99,   307,   308,   309,   310,   311,   312,   313,
+   314,   315,   316,   317,   318,   319,   320,   321,   322,   323,
+   324,   325,   326,   355,   650,     1,    95,   549,     1,   251,
+   362,     1,   319,    99,   671,    94,    98,   369,   708,   371,
+   262,    98,   374,    99,    87,    94,    94,   351,    52,   650,
+   650,   355,     0,     1,    93,    99,    98,   279,   362,    63,
+   686,   413,    94,   689,   351,   369,   398,   371,   355,   650,
+   374,    99,   424,   671,    99,    98,     1,   299,   430,   301,
+   374,     1,    93,   251,     1,   686,   686,   374,   689,   689,
+    63,    95,    98,     1,   398,   655,    44,    67,     1,   736,
+   421,    94,   524,   703,    97,   686,   747,    55,   689,   129,
+    86,   398,   712,    61,   336,     1,     1,   421,     1,    67,
+   757,   343,    95,   427,     1,   347,     1,    75,     1,   755,
+   730,    95,    52,   355,   566,     0,     1,   763,   736,   765,
+   362,   767,   162,    63,     0,     1,   498,   369,     1,   371,
+     1,    99,   584,    96,   755,   755,   508,   100,   510,   757,
+     1,    86,   763,   763,   765,   765,   767,   767,   520,     1,
+     1,     1,    96,     1,   755,    95,   398,   536,     1,    44,
+   601,     1,   763,    86,   765,    93,   767,   355,    44,   541,
+    55,   543,    45,    46,   362,     1,    61,     1,    72,    55,
+    95,   369,    67,   371,   636,    61,   374,    93,    93,    93,
+    93,    67,    86,    86,   646,    92,    93,   549,    93,   551,
+    97,    98,    99,   100,    53,     1,   578,     0,     1,   251,
+   398,     1,   536,    86,    99,    88,   129,    44,    91,    92,
+    93,   100,    93,    99,    97,   549,    50,   551,   101,   536,
+   571,    92,   673,    94,    61,    86,    97,    98,    99,   100,
+    67,    93,   549,    93,   551,    93,   101,   571,   572,   162,
+    93,    44,    65,    93,    50,     0,     1,     1,    82,    72,
+    86,    85,    86,   587,   616,     1,   618,    95,    61,    97,
+     1,    83,     1,   625,    67,    92,   655,     0,     1,   521,
+    97,    98,   524,   724,    95,    96,    82,   177,    99,    85,
+    86,     0,     1,   665,   107,     1,    86,     1,   650,    44,
+     1,   625,     1,     5,     6,     7,    99,   549,    44,   551,
+   122,   123,   414,   355,    50,   417,    61,     1,   625,   671,
+   362,    44,    67,   101,    94,     1,   650,   369,   101,   371,
+   100,   655,   374,     1,   686,    44,    97,   689,    61,   711,
+     1,   655,    86,   715,    67,   717,    82,     1,   655,    85,
+    86,   703,    61,   251,    99,    86,   398,    86,    67,    11,
+   712,   549,   686,   551,    65,   689,    67,    12,    67,   721,
+     1,    72,   262,    72,     5,    76,    99,    76,   730,   703,
+    86,   623,    86,   625,   736,    86,    13,    86,   712,    65,
+    99,    67,     1,   745,    97,   747,    72,   721,   101,     1,
+    76,     1,    86,   755,     1,   757,   730,    14,   650,     1,
+    86,   763,    99,   765,   721,   767,   251,    64,    86,    97,
+    60,   745,    69,   747,    64,    86,    45,    46,   616,    96,
+   618,   755,    86,    99,   100,    45,    46,   625,   745,   763,
+   747,   765,    95,   767,   686,    93,     1,   689,     9,    97,
+    11,    93,    44,   101,    44,    86,    93,   355,    50,    47,
+    50,   703,   650,    99,   362,    47,   686,    99,     1,   689,
+   712,   369,    33,   371,    62,    99,   374,    86,    97,   721,
+    62,   128,   101,   671,    86,   129,    86,    97,   730,    86,
+    82,   101,    82,    85,    86,    85,    86,     1,   686,   251,
+   398,   689,    94,   745,    94,   747,    99,   549,    96,   551,
+    99,    44,    93,   755,    96,   703,   160,    50,   162,    94,
+   355,   763,    97,   765,   712,   767,   101,   362,   315,   316,
+   317,   318,    94,   721,   369,   755,   371,   251,   100,   374,
+    91,    92,   730,   763,    93,   765,    50,   767,   736,    82,
+    44,    88,    85,    86,    94,    59,    50,   745,    94,   747,
+   100,    65,   100,   398,   100,    94,    66,   755,    72,   757,
+    10,   100,    76,    67,   616,   763,   618,   765,    82,   767,
+    99,    85,    86,   625,    93,    89,    90,    94,    82,    93,
+    30,    85,    86,   100,    34,   416,     1,   418,   102,   103,
+   104,   105,    97,   355,   108,   109,    91,    92,   650,    86,
+   362,    51,    45,    46,    91,    92,    93,   369,    94,   371,
+    97,     1,   374,    93,   101,   251,    44,     3,     4,   671,
+    16,    17,    50,    99,    18,    19,    20,    21,    94,    44,
+    94,   355,    91,    92,   686,    50,   398,   689,   362,    97,
+    98,   549,     1,   551,    59,   369,    61,   371,    94,    44,
+   374,   703,    67,   100,    82,    50,    56,    85,    86,    56,
+   712,    45,    46,    57,    59,    99,    61,    82,    94,   721,
+    85,    86,    67,    94,   398,    91,    92,    93,   730,    45,
+    46,    97,   310,   311,   736,   101,     1,    82,   320,   321,
+    85,    86,    94,   745,     1,   747,     8,     9,    10,    56,
+    95,    96,    94,   755,   549,   757,   551,     0,   616,    94,
+   618,   763,    99,   765,    44,   767,    99,   625,    94,   355,
+    50,     0,    91,    92,    93,   319,   362,    11,    97,    59,
+    54,    61,   101,   369,    44,   371,   139,    67,   374,    86,
+    50,    75,   650,   162,    91,    92,    93,   160,   393,    59,
+    97,    61,    82,   252,   101,    85,    86,    67,    97,    98,
+    99,    52,   398,   671,   282,    95,    96,   398,    44,    94,
+   374,   616,    82,   618,    50,    85,    86,   719,   686,   719,
+   625,   689,   602,    59,   374,    61,    96,   549,   602,   551,
+   322,    67,    86,   323,   325,   703,   324,    91,    92,    93,
+   312,   313,   314,    97,   712,   650,    82,   101,   587,    85,
+    86,   326,    -1,   721,   330,    -1,    -1,    -1,    -1,    -1,
+    96,    -1,   730,    -1,    -1,   549,   671,   551,   736,    -1,
+    91,    92,    93,    -1,    -1,    -1,    97,   745,    -1,   747,
+   101,   686,    -1,    -1,   689,    -1,    -1,   755,    -1,   757,
+    -1,    -1,    -1,    -1,   616,   763,   618,   765,   703,   767,
+    -1,    -1,    -1,   625,    -1,    -1,    -1,   712,    -1,    -1,
+    -1,    -1,    -1,    -1,    -1,    -1,   721,    -1,    -1,    -1,
+    -1,    -1,    -1,    -1,    -1,   730,    -1,    -1,   650,    -1,
+    -1,   736,   616,    -1,   618,    -1,    -1,    -1,    -1,    50,
+   745,   625,   747,    -1,    -1,    -1,    -1,    -1,    59,   671,
+   755,    -1,   757,   549,    65,   551,    -1,    -1,   763,    -1,
+   765,    72,   767,    -1,   686,    76,   650,   689,    -1,    -1,
+    -1,    82,    -1,    -1,    85,    86,    -1,    -1,    89,    90,
+    -1,   703,    93,    -1,    -1,    -1,    -1,   671,    -1,    -1,
+   712,   102,   103,   104,   105,    -1,    -1,   108,   109,   721,
+    -1,    -1,   686,    -1,    -1,   689,    -1,    -1,   730,    -1,
+    -1,    -1,    -1,    -1,   736,    -1,    -1,    -1,    -1,   703,
+   616,    -1,   618,   745,    -1,   747,    -1,    -1,   712,   625,
+    -1,    -1,    -1,   755,    -1,   757,    -1,   721,    -1,    -1,
+    -1,   763,    -1,   765,    -1,   767,   730,    -1,    -1,    -1,
+    -1,    -1,   736,    -1,   650,    -1,   190,   191,   192,   193,
+    -1,   745,    -1,   747,   198,   199,    -1,    -1,    -1,    -1,
+    -1,   755,    -1,   757,    -1,   671,    -1,    -1,    -1,   763,
+    -1,   765,    -1,   767,    -1,    -1,   190,   191,   192,   193,
+   686,    -1,    -1,   689,   198,   199,    -1,    -1,    -1,    -1,
+    -1,    -1,    -1,    -1,    -1,    -1,    -1,   703,    -1,    -1,
+    -1,    -1,    -1,    -1,    -1,    -1,   712,    -1,    -1,    -1,
+    -1,    -1,    -1,    -1,    -1,   721,    -1,    -1,    -1,    -1,
+    -1,    -1,    -1,    -1,   730,    -1,    -1,    -1,    -1,    -1,
+   736,    -1,    -1,    -1,    -1,    -1,    -1,    -1,    -1,   745,
+    -1,   747,    -1,    -1,    -1,    -1,    -1,    -1,    -1,   755,
+    -1,   757,    -1,    -1,    -1,    -1,    -1,   763,    -1,   765,
+    -1,   767,    -1,   307,   308,   309,   310,   311,   312,   313,
+   314,   315,   316,   317,   318,    -1,   320,   321,   322,   323,
+   324,   325,   326,    -1,     1,    -1,    -1,    -1,    -1,    -1,
+    -1,    -1,    -1,   307,   308,   309,   310,   311,   312,   313,
+   314,   315,   316,   317,   318,    -1,   320,   321,   322,   323,
+   324,   325,   326,    -1,    -1,    -1,    -1,    -1,    -1,    -1,
+    -1,    -1,    -1,    -1,    -1,    -1,    -1,    44,    45,    46,
+    47,    48,    49,    50,    51,    -1,    -1,    54,    -1,    -1,
+    -1,    58,    59,    -1,    -1,    62,    -1,    -1,    65,    66,
+    67,    68,    -1,    70,    71,    72,    73,    -1,    -1,    76,
+    -1,    -1,    -1,    -1,    -1,    82,    -1,    -1,    85,    86,
+    -1,     1,    -1,    -1,    -1,    -1,    93,   421,    95,    96,
+    -1,    -1,    99,   427,    -1,   102,   103,   104,   105,    -1,
+    -1,   108,   109,    -1,    -1,    -1,    -1,    -1,    -1,    -1,
+    -1,    -1,    -1,    -1,    -1,    -1,    -1,   421,    -1,    -1,
+    -1,    -1,    -1,   427,    44,    45,    46,    47,    48,    49,
+    50,    51,    -1,    -1,    54,    -1,    -1,    -1,    58,    59,
+    -1,    -1,    62,    -1,    -1,    65,    66,    67,    68,    -1,
+    70,    71,    72,    73,    -1,     1,    76,     3,     4,    -1,
+    -1,    -1,    82,    -1,    -1,    85,    86,    -1,    -1,    -1,
+    -1,    -1,    -1,    93,    -1,    95,    96,    -1,    -1,    99,
+    -1,    -1,   102,   103,   104,   105,    -1,    -1,   108,   109,
+    -1,    -1,    -1,    -1,    -1,    -1,    -1,    -1,    -1,    45,
+    46,    -1,    -1,    -1,    50,    -1,    -1,    -1,    -1,    -1,
+    -1,    -1,    -1,    59,    -1,    -1,    -1,    -1,    -1,    65,
+    -1,    -1,    -1,    -1,    -1,    -1,    72,     1,    -1,    -1,
+    76,    -1,    -1,    -1,    -1,    -1,    82,   571,   572,    85,
+    86,    -1,    -1,    89,    90,    -1,    -1,    93,    -1,    95,
+    96,    -1,    -1,   587,    -1,    -1,   102,   103,   104,   105,
+    -1,    -1,   108,   109,    -1,    -1,    -1,   571,   572,    -1,
+    44,    45,    46,    47,    48,    49,    50,    51,    -1,    -1,
+    54,    -1,    -1,   587,    58,    59,    -1,    -1,    62,    -1,
+    -1,    65,    66,    67,    68,     1,    70,    71,    72,    73,
+    -1,    -1,    76,    -1,    -1,    -1,    -1,    -1,    82,    -1,
+    -1,    85,    86,    -1,    -1,    -1,    -1,    -1,    -1,    93,
+    -1,    95,    96,    -1,    -1,    99,    -1,    -1,   102,   103,
+   104,   105,    -1,    -1,   108,   109,    -1,    -1,    44,    45,
+    46,    -1,    48,    49,    50,    51,    -1,    -1,    54,    -1,
+    -1,    -1,    58,    59,    -1,    -1,    -1,    -1,    -1,    65,
+    66,    67,    68,     1,    70,    71,    72,    73,    -1,    -1,
+    76,    -1,    -1,    -1,    -1,    -1,    82,    -1,    -1,    85,
+    86,    -1,    -1,    -1,    -1,    -1,    -1,    93,    -1,    95,
+    96,    -1,    -1,    99,    -1,    -1,   102,   103,   104,   105,
+    -1,    -1,   108,   109,    -1,    -1,    44,    45,    46,    -1,
+    48,    49,    50,    51,    -1,    -1,    54,    -1,    -1,    -1,
+    58,    59,    -1,    -1,    -1,    -1,    -1,    65,    66,    67,
+    68,     1,    70,    71,    72,    73,    -1,    -1,    76,    -1,
+    -1,    -1,    -1,    -1,    82,    -1,    -1,    85,    86,    -1,
+    -1,    -1,    -1,    -1,    -1,    93,    -1,    95,    96,    -1,
+    -1,    99,    -1,    -1,   102,   103,   104,   105,    -1,    -1,
+   108,   109,    -1,    -1,    44,    45,    46,    -1,    48,    49,
+    50,    51,    -1,    -1,    54,    -1,    -1,    -1,    58,    59,
+    -1,    -1,    -1,    -1,    -1,    65,    66,    67,    68,     1,
+    70,    71,    72,    73,    -1,    -1,    76,    -1,    -1,    -1,
+    -1,    -1,    82,    -1,    -1,    85,    86,    -1,    -1,    -1,
+    -1,    -1,    -1,    93,    -1,    95,    96,    -1,    -1,    99,
+    -1,    -1,   102,   103,   104,   105,    -1,    -1,   108,   109,
+    -1,    -1,    44,    45,    46,    -1,    48,    49,    50,    51,
+    -1,    -1,    54,    -1,    -1,    -1,    58,    59,    -1,    -1,
+    -1,    -1,    -1,    65,    66,    67,    68,     1,    70,    71,
+    72,    73,    -1,    -1,    76,    -1,    -1,    -1,    -1,    -1,
+    82,    -1,    -1,    85,    86,    -1,    -1,    -1,    -1,    -1,
+    -1,    93,    -1,    95,    96,    -1,    -1,    99,    -1,    -1,
+   102,   103,   104,   105,    -1,    -1,   108,   109,    -1,    -1,
+    44,    45,    46,    -1,    48,    49,    50,    51,    -1,    -1,
+    54,    -1,    -1,    -1,    58,    59,    -1,    -1,    -1,    -1,
+    -1,    65,    66,    67,    68,     1,    70,    71,    72,    73,
+    -1,    -1,    76,    -1,    -1,    -1,    -1,    -1,    82,    -1,
+    -1,    85,    86,    -1,    -1,    -1,    -1,    -1,    -1,    93,
+    -1,    95,    96,    -1,    -1,    99,    -1,    -1,   102,   103,
+   104,   105,    -1,    -1,   108,   109,    -1,    -1,    44,    45,
+    46,    -1,    48,    49,    50,    51,    -1,    -1,    54,    -1,
+    -1,    -1,    58,    59,    -1,    -1,    -1,    -1,    -1,    65,
+    66,    67,    68,     1,    70,    71,    72,    73,    -1,    -1,
+    76,    -1,    -1,    -1,    -1,    -1,    82,    -1,    -1,    85,
+    86,    -1,    -1,    -1,    -1,    -1,    -1,    93,    -1,    95,
+    -1,    -1,    -1,    99,    -1,    -1,   102,   103,   104,   105,
+    -1,    -1,   108,   109,    -1,    -1,    44,    45,    46,    -1,
+    48,    49,    50,    51,    -1,    -1,    54,    -1,    -1,    -1,
+    58,    59,    -1,    -1,    -1,    -1,    -1,    65,    66,     1,
+    68,    -1,    70,    71,    72,    73,    -1,    -1,    76,    -1,
+    -1,    -1,    -1,    -1,    82,    -1,    -1,    85,    86,    -1,
+    -1,    -1,    -1,    -1,    -1,    93,    -1,    95,    -1,    -1,
+    -1,    99,    -1,    -1,   102,   103,   104,   105,    -1,    -1,
+   108,   109,    44,    45,    46,    -1,    48,    49,    50,    51,
+    -1,    -1,    54,    -1,    -1,    -1,    58,    59,    -1,    -1,
+    -1,    -1,    -1,    65,    66,    -1,    68,    -1,    70,    71,
+    72,    73,    -1,     1,    76,     3,     4,    -1,    -1,    -1,
+    82,    -1,    -1,    85,    86,    -1,    -1,    -1,    -1,    -1,
+    -1,    93,    -1,    95,    -1,    -1,    -1,    99,    -1,    -1,
+   102,   103,   104,   105,    -1,    -1,   108,   109,    -1,    -1,
+    -1,    -1,    -1,    -1,    -1,    -1,    -1,    45,    46,    -1,
+    -1,    -1,    50,    -1,     1,    -1,     3,     4,    -1,    -1,
+    -1,    59,    -1,    -1,    -1,    -1,    -1,    65,    -1,    -1,
+    -1,    -1,    -1,    -1,    72,    -1,    -1,    -1,    76,    -1,
+    -1,    -1,    -1,    -1,    82,    -1,    -1,    85,    86,    -1,
+    -1,    89,    90,    -1,    -1,    93,    -1,    95,    45,    46,
+    -1,    -1,    -1,    50,   102,   103,   104,   105,    -1,    -1,
+   108,   109,    59,    -1,    -1,    -1,    -1,    -1,    65,     1,
+    -1,     3,     4,    -1,    -1,    72,    -1,    -1,    -1,    76,
+    -1,    -1,    -1,    -1,    -1,    82,    -1,    -1,    85,    86,
+    -1,    88,    89,    90,    -1,    -1,    93,    -1,    -1,    -1,
+    -1,    -1,    -1,    -1,    -1,   102,   103,   104,   105,    -1,
+    -1,   108,   109,    45,    46,    -1,    -1,    -1,    50,    -1,
+     1,    -1,     3,     4,    -1,    -1,    -1,    59,    -1,    -1,
+    -1,    -1,    -1,    65,    -1,    -1,    -1,    -1,    -1,    -1,
+    72,    -1,    -1,    -1,    76,    -1,    -1,    -1,    -1,    -1,
+    82,    -1,    -1,    85,    86,    -1,    -1,    89,    90,    -1,
+    -1,    93,    -1,    -1,    45,    46,    -1,    99,    -1,    50,
+   102,   103,   104,   105,    -1,    -1,   108,   109,    59,    -1,
+    -1,    -1,    -1,    -1,    65,     1,    -1,     3,     4,    -1,
+    -1,    72,    -1,    -1,    -1,    76,    -1,    -1,    -1,    -1,
+    -1,    82,    -1,    -1,    85,    86,    -1,    -1,    89,    90,
+    -1,    -1,    93,    -1,    -1,    -1,    -1,    98,    -1,    -1,
+    -1,   102,   103,   104,   105,    -1,    -1,   108,   109,    45,
+    46,    -1,    -1,    -1,    50,    -1,     1,    -1,     3,     4,
+    -1,    -1,    -1,    59,    -1,    -1,    -1,    -1,    -1,    65,
+    -1,    -1,    -1,    -1,    -1,    -1,    72,    -1,    -1,    -1,
+    76,    -1,    -1,    -1,    -1,    -1,    82,    -1,    -1,    85,
+    86,    -1,    -1,    89,    90,    -1,    -1,    93,    94,    -1,
+    45,    46,    -1,    -1,    -1,    50,   102,   103,   104,   105,
+    -1,    -1,   108,   109,    59,    -1,    -1,    -1,    -1,    -1,
+    65,     1,    -1,     3,     4,    -1,    -1,    72,    -1,    -1,
+    -1,    76,    -1,    -1,    -1,    -1,    -1,    82,    -1,    -1,
+    85,    86,    -1,    -1,    89,    90,    -1,    -1,    93,    -1,
+    -1,    -1,    -1,    98,    -1,    -1,    -1,   102,   103,   104,
+   105,    -1,    -1,   108,   109,    45,    46,    -1,    -1,    -1,
+    50,    -1,     1,    -1,     3,     4,    -1,    -1,    -1,    59,
+    -1,    -1,    -1,    -1,    -1,    65,    -1,    -1,    -1,    -1,
+    -1,    -1,    72,    -1,    -1,    -1,    76,    -1,    -1,    -1,
+    -1,    -1,    82,    -1,    -1,    85,    86,    -1,    -1,    89,
+    90,    -1,    -1,    93,    -1,    -1,    45,    46,    98,    -1,
+    -1,    50,   102,   103,   104,   105,    -1,    -1,   108,   109,
+    59,    -1,    -1,    -1,    -1,    -1,    65,     1,    -1,     3,
+     4,    -1,    -1,    72,    -1,    -1,    -1,    76,    -1,    -1,
+    -1,    -1,    -1,    82,    -1,    -1,    85,    86,    -1,    -1,
+    89,    90,    -1,    -1,    93,    94,    -1,    -1,    -1,    -1,
+    -1,    -1,    -1,   102,   103,   104,   105,    -1,    -1,   108,
+   109,    45,    46,    -1,    -1,    -1,    50,    -1,     1,    -1,
+     3,     4,    -1,    -1,    -1,    59,    -1,    -1,    -1,    -1,
+    -1,    65,    -1,    -1,    -1,    -1,    -1,    -1,    72,    -1,
+    -1,    -1,    76,    -1,    -1,    -1,    -1,    -1,    82,    -1,
+    -1,    85,    86,    -1,    -1,    89,    90,    -1,    -1,    93,
+    -1,    -1,    45,    46,    -1,    99,    -1,    50,   102,   103,
+   104,   105,    -1,    -1,   108,   109,    59,    -1,    -1,    -1,
+    -1,    -1,    65,     1,    -1,     3,     4,    -1,    -1,    72,
+    -1,    -1,    -1,    76,    -1,    -1,    -1,    -1,    -1,    82,
+    -1,    -1,    85,    86,    -1,    -1,    89,    90,    -1,    -1,
+    93,    94,    -1,    -1,    -1,    -1,    -1,    -1,    -1,   102,
+   103,   104,   105,    -1,    -1,   108,   109,    45,    46,    -1,
+    -1,    -1,    50,    -1,     1,    -1,     3,     4,    -1,    -1,
+    -1,    59,    -1,    -1,    -1,    -1,    -1,    65,    -1,    -1,
+    -1,    -1,    -1,    -1,    72,    -1,    -1,    -1,    76,    -1,
+    -1,    -1,    -1,    -1,    82,    -1,    -1,    85,    86,    -1,
+    -1,    89,    90,    -1,    -1,    93,    94,    -1,    45,    46,
+    -1,    -1,    -1,    50,   102,   103,   104,   105,    -1,    -1,
+   108,   109,    59,    -1,    -1,    -1,    -1,    -1,    65,     1,
+    -1,     3,     4,    -1,    -1,    72,    -1,    -1,    -1,    76,
+    -1,    -1,    -1,    -1,    -1,    82,    -1,    -1,    85,    86,
+    -1,    -1,    89,    90,    -1,    -1,    93,    94,    -1,    -1,
+    -1,    -1,    -1,    -1,    -1,   102,   103,   104,   105,    -1,
+    -1,   108,   109,    45,    46,    -1,    -1,    -1,    50,    -1,
+     1,    -1,     3,     4,    -1,    -1,    -1,    59,    -1,    -1,
+    -1,    -1,    -1,    65,    -1,    -1,    -1,    -1,    -1,    -1,
+    72,    -1,    -1,    -1,    76,    -1,    -1,    -1,    -1,    -1,
+    82,    -1,    -1,    85,    86,    -1,    -1,    89,    90,    -1,
+    -1,    93,    -1,    -1,    45,    46,    -1,    99,    -1,    50,
+   102,   103,   104,   105,    -1,    -1,   108,   109,    59,    -1,
+    -1,    -1,    -1,    -1,    65,     1,    -1,     3,     4,    -1,
+    -1,    72,    -1,    -1,    -1,    76,    -1,    -1,    -1,    -1,
+    -1,    82,    -1,    -1,    85,    86,    -1,    -1,    89,    90,
+    -1,    -1,    93,    -1,    -1,    -1,    -1,    -1,    -1,    -1,
+    -1,   102,   103,   104,   105,    -1,    -1,   108,   109,    45,
+    46,    -1,    -1,    -1,    50,    -1,     1,    -1,     3,     4,
+    -1,    -1,    -1,    59,    -1,    -1,    -1,    -1,    -1,    65,
+    -1,    -1,    -1,    -1,    -1,    -1,    72,    -1,    -1,    -1,
+    76,    -1,    -1,    -1,    -1,    -1,    82,    -1,    -1,    85,
+    86,    -1,    -1,    89,    90,    -1,    -1,    93,    -1,    -1,
+    45,    46,    -1,    -1,    -1,    50,   102,   103,   104,   105,
+    -1,    -1,   108,   109,    59,    -1,    -1,    -1,    -1,    -1,
+    65,     1,    -1,     3,     4,    -1,    -1,    72,    -1,    -1,
+    -1,    76,    -1,    -1,    -1,    -1,    -1,    82,    -1,    -1,
+    85,    86,    -1,    -1,    89,    90,    -1,    -1,    93,    -1,
+    -1,    -1,    -1,    -1,    -1,    -1,    -1,   102,   103,   104,
+   105,    -1,    -1,   108,   109,    45,    46,    -1,    -1,    -1,
+    50,    -1,     1,    -1,     3,     4,    -1,    -1,    -1,    59,
+    -1,    -1,    -1,    -1,    -1,    65,    -1,    -1,    -1,    -1,
+    -1,    -1,    72,    -1,    -1,    -1,    76,    -1,    -1,    -1,
+    -1,    -1,    82,    -1,    -1,    85,    86,    -1,    -1,    89,
+    90,    -1,    -1,    93,    -1,    -1,    45,    46,    -1,    -1,
+    -1,    50,   102,   103,   104,   105,    -1,    -1,   108,   109,
+    59,    -1,    -1,    -1,    -1,    -1,    65,     1,    -1,     3,
+     4,    -1,    -1,    72,    -1,    -1,    -1,    76,    -1,    -1,
+    -1,    -1,    -1,    82,    -1,    -1,    85,    86,    -1,    -1,
+    89,    90,    -1,    -1,    93,    -1,    -1,    -1,    -1,    -1,
+    -1,    -1,    -1,   102,   103,   104,   105,    -1,    -1,   108,
+   109,    45,    46,    -1,    -1,    -1,    50,    -1,     1,    -1,
+     3,     4,    -1,    -1,    -1,    59,    -1,    -1,    -1,    -1,
+    -1,    65,    -1,    -1,    -1,    -1,    -1,    -1,    72,    -1,
+    -1,    -1,    76,    -1,    -1,    -1,    -1,    -1,    82,    -1,
+    -1,    85,    86,    -1,    -1,    89,    90,    -1,    -1,    93,
+    -1,    -1,    45,    46,    -1,    -1,    -1,    50,   102,   103,
+   104,   105,    -1,    -1,   108,   109,    59,    -1,    -1,    -1,
+    -1,    -1,    65,     1,    -1,     3,     4,    -1,    -1,    72,
+    -1,    -1,    -1,    76,    -1,    -1,    -1,    -1,    -1,    82,
+    -1,    -1,    85,    86,    -1,    -1,    89,    90,    -1,    -1,
+    93,    -1,    -1,    -1,    -1,    -1,    -1,    -1,    -1,   102,
+   103,   104,   105,    -1,    -1,   108,   109,    45,    46,    -1,
+    -1,    -1,    50,    -1,     1,    -1,     3,     4,    -1,    -1,
+    -1,    59,    -1,    -1,    -1,    -1,    -1,    65,    -1,    -1,
+    -1,    -1,    -1,    -1,    72,    -1,    -1,    -1,    76,    -1,
+    -1,    -1,    -1,    -1,    82,    -1,    -1,    85,    86,    -1,
+    -1,    89,    90,    -1,    -1,    93,    -1,    -1,    45,    46,
+    -1,    -1,    -1,    50,   102,   103,   104,   105,    -1,    -1,
+   108,   109,    59,    -1,    -1,    -1,    -1,    -1,    65,     1,
+    -1,     3,     4,    -1,    -1,    72,    -1,    -1,    -1,    76,
+    -1,    -1,    -1,    -1,    -1,    82,    -1,    -1,    85,    86,
+    -1,    -1,    89,    90,    -1,    -1,    93,    -1,    -1,    -1,
+    -1,    -1,    -1,    -1,    -1,   102,   103,   104,   105,    -1,
+    -1,   108,   109,    45,    46,    -1,    -1,    -1,    50,    -1,
+     1,    -1,     3,     4,    -1,    -1,    -1,    59,    -1,    -1,
+    -1,    -1,    -1,    65,    -1,    -1,    -1,    -1,    -1,    -1,
+    72,    -1,    -1,    -1,    76,    -1,    -1,    -1,    -1,    -1,
+    82,    -1,    -1,    85,    86,    -1,    -1,    89,    90,    -1,
+    -1,    93,    -1,    -1,    45,    46,    -1,    -1,    -1,    50,
+   102,   103,   104,   105,    -1,    -1,   108,   109,    59,    -1,
+    -1,    -1,    -1,    -1,    65,     1,    -1,     3,     4,    -1,
+    -1,    72,    -1,    -1,    -1,    76,    -1,    -1,    -1,    -1,
+    -1,    82,    -1,    -1,    85,    86,    -1,    -1,    89,    90,
+    -1,    -1,    93,    -1,    -1,    -1,    -1,    -1,    -1,    -1,
+    -1,   102,   103,   104,   105,    -1,    -1,   108,   109,    45,
+    46,    -1,    -1,    -1,    50,    -1,     1,    -1,     3,     4,
+    -1,    -1,    -1,    59,    -1,    -1,    -1,    -1,    -1,    65,
+    -1,    -1,    -1,    -1,    -1,    -1,    72,    -1,    -1,    -1,
+    76,    -1,    -1,    -1,    -1,    -1,    82,    -1,    -1,    85,
+    86,    -1,    -1,    89,    90,    -1,    -1,    93,    -1,    -1,
+    45,    46,    -1,    -1,    -1,    50,   102,   103,   104,   105,
+    -1,    -1,   108,   109,    59,    -1,    -1,    -1,    -1,    -1,
+    65,     1,    -1,     3,     4,    -1,    -1,    72,    -1,    -1,
+    -1,    76,    -1,    -1,    -1,    -1,    -1,    82,    -1,    -1,
+    85,    86,    -1,    -1,    89,    90,    -1,    -1,    93,    -1,
+    -1,    -1,    -1,    -1,    -1,    -1,    -1,   102,   103,   104,
+   105,    -1,    -1,   108,   109,    45,    46,    -1,    -1,    -1,
+    50,    -1,     1,    -1,     3,     4,    -1,    -1,    -1,    59,
+    -1,    -1,    -1,    -1,    -1,    65,    -1,    -1,    -1,    -1,
+    -1,    -1,    72,    -1,    -1,    -1,    76,    -1,    -1,    -1,
+    -1,    -1,    82,    -1,    -1,    85,    86,    -1,    -1,    89,
+    90,    -1,    -1,    93,    -1,    -1,    45,    46,    -1,    -1,
+    -1,    50,   102,   103,   104,   105,    -1,    -1,   108,   109,
+    59,    -1,    -1,    -1,    -1,    -1,    65,     1,    -1,     3,
+     4,    -1,    -1,    72,    -1,    -1,    -1,    76,    -1,    -1,
+    -1,    -1,    -1,    82,    -1,    -1,    85,    86,    -1,    -1,
+    89,    90,    -1,    -1,    93,    -1,    -1,    -1,    -1,    -1,
+    -1,    -1,    -1,   102,   103,   104,   105,    -1,    -1,   108,
+   109,    45,    46,    -1,    -1,    -1,    50,    -1,     1,    -1,
+     3,     4,    -1,    -1,    -1,    59,    -1,    -1,    -1,    -1,
+    -1,    65,    -1,    -1,    -1,    -1,    -1,    -1,    72,    -1,
+    -1,    -1,    76,    -1,    -1,    -1,    -1,    -1,    82,    -1,
+    -1,    85,    86,    -1,    -1,    89,    90,    -1,    -1,    93,
+    -1,    -1,    45,    46,    -1,    -1,    -1,    50,   102,   103,
+   104,   105,    -1,    -1,   108,   109,    59,    -1,    -1,    -1,
+    -1,    -1,    65,     1,    -1,     3,     4,    -1,    -1,    72,
+    -1,    -1,    -1,    76,    -1,    -1,    -1,    -1,    -1,    82,
+    -1,    -1,    85,    86,    -1,    -1,    89,    90,    -1,    -1,
+    93,    -1,    -1,    -1,    -1,    -1,    -1,    -1,    -1,   102,
+   103,   104,   105,    -1,    -1,   108,   109,    45,    46,    -1,
+    -1,    -1,    50,    -1,     1,    -1,     3,     4,    -1,    -1,
+    -1,    59,    -1,    -1,    -1,    -1,    -1,    65,    -1,    -1,
+    -1,    -1,    -1,    -1,    72,    -1,    -1,    -1,    76,    -1,
+    -1,    -1,    -1,    -1,    82,    -1,    -1,    85,    86,    -1,
+    -1,    89,    90,    -1,    -1,    93,    -1,    -1,    45,    46,
+    -1,    -1,    -1,    50,   102,   103,   104,   105,    -1,    -1,
+   108,   109,    59,    -1,    -1,    -1,    -1,    -1,    65,     1,
+    -1,     3,     4,    -1,    -1,    72,    -1,    -1,    -1,    76,
+    -1,    -1,    -1,    -1,    -1,    82,    -1,    -1,    85,    86,
+    -1,    -1,    89,    90,    -1,    -1,    93,    -1,    -1,    -1,
+    -1,    -1,    -1,    -1,    -1,   102,   103,   104,   105,    -1,
+    -1,   108,   109,    45,    46,    -1,    -1,    -1,    50,    -1,
+     1,    -1,     3,     4,    -1,    -1,    -1,    59,    -1,    -1,
+    -1,    -1,    -1,    65,    -1,    -1,    -1,    -1,    -1,    -1,
+    72,    -1,    -1,    -1,    76,    -1,    -1,    -1,    -1,    -1,
+    82,    -1,    -1,    85,    86,    -1,    -1,    89,    90,    -1,
+    -1,    93,    -1,    -1,    45,    46,    -1,    -1,    -1,    50,
+   102,   103,   104,   105,    -1,    -1,   108,   109,    59,    -1,
+    -1,    -1,    -1,    -1,    65,     1,    -1,     3,     4,    -1,
+    -1,    72,    -1,    -1,    -1,    76,    -1,    -1,    -1,    -1,
+    -1,    82,    -1,    -1,    85,    86,    -1,    -1,    89,    90,
+    -1,    -1,    93,    -1,    -1,    -1,    -1,    -1,    -1,    -1,
+    -1,   102,   103,   104,   105,    -1,    -1,   108,   109,    45,
+    46,    -1,    -1,    -1,    50,    -1,     1,    -1,     3,     4,
+    -1,    -1,    -1,    59,    -1,    -1,    -1,    -1,    -1,    65,
+    -1,    -1,    -1,    -1,    -1,    -1,    72,    -1,    -1,    -1,
+    76,    -1,    -1,    -1,    -1,    -1,    82,    -1,    -1,    85,
+    86,    -1,    -1,    89,    90,    -1,    -1,    93,    -1,    -1,
+    45,    46,    -1,    -1,    -1,    50,   102,   103,   104,   105,
+    -1,    -1,   108,   109,    59,    -1,    -1,    -1,    -1,    -1,
+    65,     1,    -1,     3,     4,    -1,    -1,    72,    -1,    -1,
+    -1,    76,    -1,    -1,    -1,    -1,    -1,    82,    -1,    -1,
+    85,    86,    -1,    -1,    89,    90,    -1,    -1,    93,    -1,
+    -1,    -1,    -1,    -1,    -1,    -1,    -1,   102,   103,   104,
+   105,    -1,    -1,   108,   109,    45,    46,    -1,    -1,    -1,
+    50,    -1,     1,    -1,     3,     4,    -1,    -1,    -1,    59,
+    -1,    -1,    -1,    -1,    -1,    65,    -1,    -1,    -1,    -1,
+    -1,    -1,    72,    -1,    -1,    -1,    76,    -1,    -1,    -1,
+    -1,    -1,    82,    -1,    -1,    85,    86,    -1,    -1,    89,
+    90,    -1,    -1,    93,    -1,    -1,    45,    46,    -1,    -1,
+    -1,    50,   102,   103,   104,   105,    -1,    -1,   108,   109,
+    59,    -1,    -1,    -1,    -1,    -1,    65,     1,    -1,     3,
+     4,    -1,    -1,    72,    -1,    -1,    -1,    76,    -1,    -1,
+    -1,    -1,    -1,    82,    -1,    -1,    85,    86,    -1,    -1,
+    89,    90,    -1,    -1,    93,    -1,    -1,    -1,    -1,    -1,
+    -1,    -1,    -1,   102,   103,   104,   105,    -1,    -1,   108,
+   109,    45,    46,    -1,    -1,    -1,    50,    -1,     1,    -1,
+     3,     4,    -1,    -1,    -1,    59,    -1,    -1,    -1,    -1,
+    -1,    65,    -1,    -1,    -1,    -1,    -1,    -1,    72,    -1,
+    -1,    -1,    76,    -1,    -1,    -1,    -1,    -1,    82,    -1,
+    -1,    85,    86,    -1,    -1,    89,    90,    -1,    -1,    93,
+    -1,    -1,    45,    46,    -1,    -1,    -1,    50,   102,   103,
+   104,   105,    -1,    -1,   108,   109,    59,    -1,    -1,    -1,
+    -1,    -1,    65,     1,    -1,     3,     4,    -1,    -1,    72,
+    -1,    -1,    -1,    76,    -1,    -1,    -1,    -1,    -1,    82,
+    -1,    -1,    85,    86,    -1,    -1,    89,    90,    -1,    -1,
+    93,    -1,    -1,    -1,    -1,    -1,    -1,    -1,    -1,   102,
+   103,   104,   105,    -1,    -1,   108,   109,    45,    46,    -1,
+    -1,    -1,    50,    -1,     1,    -1,     3,     4,    -1,    -1,
+    -1,    59,    -1,    -1,    -1,    -1,    -1,    65,    -1,    -1,
+    -1,    -1,    -1,    -1,    72,    -1,    -1,    -1,    76,    -1,
+    -1,    -1,    -1,    -1,    82,    -1,    -1,    85,    86,    -1,
+    -1,    89,    90,    -1,    -1,    93,    -1,    -1,    45,    46,
+    -1,    -1,    -1,    50,   102,   103,   104,   105,    -1,    -1,
+   108,   109,    59,    -1,    -1,    -1,    -1,    -1,    65,     1,
+    -1,     3,     4,    -1,    -1,    72,    -1,    -1,    -1,    76,
+    -1,    -1,    -1,    -1,    -1,    82,    -1,    -1,    85,    86,
+    -1,    -1,    89,    90,    -1,    -1,    93,    -1,    -1,    -1,
+    -1,    -1,    -1,    -1,    -1,   102,   103,   104,   105,    -1,
+    -1,   108,   109,    45,    46,    -1,    -1,    -1,    50,    -1,
+     1,    -1,     3,     4,    -1,    -1,    -1,    59,    -1,    -1,
+    -1,    -1,    -1,    65,    -1,    -1,    -1,    -1,    -1,    -1,
+    72,    -1,    -1,    -1,    76,    -1,    -1,    -1,    -1,    -1,
+    82,    -1,    -1,    85,    86,    -1,    -1,    89,    90,    -1,
+    -1,    93,    -1,    -1,    45,    46,    -1,    -1,    -1,    50,
+   102,   103,   104,   105,    -1,    -1,   108,   109,    59,    -1,
+    -1,    -1,    -1,    -1,    65,     1,    -1,     3,     4,    -1,
+    -1,    72,    -1,    -1,    -1,    76,    -1,    -1,    -1,    -1,
+    -1,    82,    -1,    -1,    85,    86,    -1,    -1,    89,    90,
+    -1,    -1,    93,    -1,    -1,    -1,    -1,    -1,    -1,    -1,
+    -1,   102,   103,   104,   105,    -1,    -1,   108,   109,    45,
+    46,    -1,    -1,    -1,    50,    -1,     1,    -1,     3,     4,
+    -1,    -1,    -1,    59,    -1,    -1,    -1,    -1,    -1,    65,
+    -1,    -1,    -1,    -1,    -1,    -1,    72,    -1,    -1,    -1,
+    76,    -1,    -1,    -1,    -1,    -1,    82,    -1,    -1,    85,
+    86,    -1,    -1,    89,    90,    -1,    -1,    93,    -1,    -1,
+    45,    46,    -1,    -1,    -1,    50,   102,   103,   104,   105,
+    -1,    -1,   108,   109,    59,    -1,    -1,    -1,    -1,    -1,
+    65,     1,    -1,     3,     4,    -1,    -1,    72,    -1,    -1,
+    -1,    76,    -1,    -1,    -1,    -1,    -1,    82,    -1,    -1,
+    85,    86,    -1,    -1,    89,    90,    -1,    -1,    93,    -1,
+    -1,    -1,    -1,    -1,    -1,    -1,    -1,   102,   103,   104,
+   105,    -1,    -1,   108,   109,    45,    46,    -1,    -1,    -1,
+    50,    -1,     1,    -1,     3,     4,    -1,    -1,    -1,    59,
+    -1,    -1,    -1,    -1,    -1,    65,    -1,    -1,    -1,    -1,
+    -1,    -1,    72,    -1,    -1,    -1,    76,    -1,    -1,    -1,
+    -1,    -1,    82,    -1,    -1,    85,    86,    -1,    -1,    89,
+    90,    -1,    -1,    93,    -1,    -1,    45,    46,    -1,    -1,
+    -1,    50,   102,   103,   104,   105,    -1,    -1,   108,   109,
+    59,    -1,    -1,    -1,    -1,    -1,    65,     1,    -1,     3,
+     4,    -1,    -1,    72,    -1,    -1,    -1,    76,    -1,    -1,
+    -1,    -1,    -1,    82,    -1,    -1,    85,    86,    -1,    -1,
+    89,    90,    -1,    -1,    93,    -1,    -1,    -1,    -1,    -1,
+    -1,    -1,    -1,   102,   103,   104,   105,    -1,    -1,   108,
+   109,    45,    46,    -1,    -1,    -1,    50,    -1,     1,    -1,
+     3,     4,    -1,    -1,    -1,    59,    -1,    -1,    -1,    -1,
+    -1,    65,    -1,    -1,    -1,    -1,    -1,    -1,    72,    -1,
+    -1,    -1,    76,    -1,    -1,    -1,    -1,    -1,    82,    -1,
+    -1,    85,    86,    -1,    -1,    89,    90,    -1,    -1,    93,
+    -1,    -1,    45,    46,    -1,    -1,    -1,    50,   102,   103,
+   104,   105,    -1,    -1,   108,   109,    59,    -1,    -1,    -1,
+    -1,    -1,    65,     1,    -1,     3,     4,    -1,    -1,    72,
+    -1,    -1,    -1,    76,    -1,    -1,    -1,    -1,    -1,    82,
+    -1,    -1,    85,    86,    -1,    -1,    89,    90,    -1,    -1,
+    93,    -1,    -1,    -1,    -1,    -1,    -1,    -1,    -1,   102,
+   103,   104,   105,    -1,    -1,   108,   109,    45,    46,    -1,
+    -1,    -1,    50,    -1,     1,    -1,     3,     4,    -1,    -1,
+    -1,    59,    -1,    -1,    -1,    -1,    -1,    65,    -1,    -1,
+    -1,    -1,    -1,    -1,    72,    -1,    -1,    -1,    76,    -1,
+    -1,    -1,    -1,    -1,    82,    -1,    -1,    85,    86,    -1,
+    -1,    89,    90,    -1,    -1,    93,    -1,    -1,    45,    46,
+    -1,    -1,    -1,    50,   102,   103,   104,   105,    -1,    -1,
+   108,   109,    59,    -1,    -1,    -1,    -1,    -1,    65,    -1,
+     1,    -1,    -1,    -1,    -1,    72,    -1,    -1,    -1,    76,
+    -1,    -1,    -1,    -1,    -1,    82,    -1,    -1,    85,    86,
+    -1,    -1,    89,    90,    -1,     1,    93,    -1,    -1,    -1,
+    -1,    -1,    -1,    -1,    -1,   102,   103,   104,   105,    -1,
+    -1,   108,   109,    44,    45,    46,    -1,    -1,    -1,    50,
+    -1,    -1,    -1,    -1,    -1,    -1,    -1,    -1,    59,    -1,
+    -1,    -1,    -1,    -1,    65,    -1,    -1,     1,    -1,    45,
+    46,    72,    -1,    -1,    50,    76,    -1,    -1,    -1,    -1,
+    -1,    82,    -1,    59,    85,    86,    -1,    -1,    -1,    65,
+    -1,    -1,    93,    -1,     3,     4,    72,    -1,    99,    -1,
+    76,   102,   103,   104,   105,    -1,    82,   108,   109,    85,
+    86,    45,    46,    -1,    -1,    -1,    50,    93,    94,    -1,
+    -1,    -1,    -1,    -1,    -1,    59,   102,   103,   104,   105,
+    -1,    65,   108,   109,     1,    -1,    45,    46,    72,    -1,
+    -1,    50,    76,    -1,    -1,    -1,    -1,    -1,    82,    -1,
+    59,    85,    86,    -1,    -1,    -1,    65,    -1,    -1,    93,
+    94,     3,     4,    72,    -1,    -1,    -1,    76,   102,   103,
+   104,   105,    -1,    82,   108,   109,    85,    86,    45,    46,
+    89,    90,    -1,    50,    93,    -1,    95,    96,    -1,    -1,
+    -1,   100,    59,   102,   103,   104,   105,    -1,    65,   108,
+   109,    -1,    -1,    45,    46,    72,    -1,    -1,    50,    76,
+    -1,    -1,     3,     4,    -1,    82,    -1,    59,    85,    86,
+    -1,    -1,    -1,    65,    -1,    -1,    93,    -1,    -1,    -1,
+    72,    -1,    -1,    -1,    76,   102,   103,   104,   105,    -1,
+    82,   108,   109,    85,    86,    -1,    -1,    89,    90,    -1,
+    -1,    93,    94,    -1,    45,    46,    -1,    -1,    -1,    50,
+   102,   103,   104,   105,    -1,    -1,   108,   109,    59,    -1,
+    -1,    -1,    -1,    -1,    65,    -1,    -1,     3,     4,    -1,
+    -1,    72,    -1,    -1,    -1,    76,    -1,    -1,    -1,    -1,
+    -1,    82,    -1,    -1,    85,    86,    -1,    -1,    89,    90,
+    -1,    -1,    93,    94,    -1,    -1,    -1,    -1,    -1,    -1,
+    -1,   102,   103,   104,   105,    -1,    -1,   108,   109,    45,
+    46,    -1,    -1,    -1,    50,    -1,    -1,    -1,     3,     4,
+    -1,    -1,    -1,    59,    -1,    -1,    -1,    -1,    -1,    65,
+    -1,    -1,    -1,    -1,    -1,    -1,    72,    -1,    -1,    -1,
+    76,    -1,    -1,    -1,    -1,    -1,    82,    -1,    -1,    85,
+    86,    -1,    -1,    89,    90,    -1,    -1,    93,    94,    -1,
+    45,    46,    -1,    -1,    -1,    50,   102,   103,   104,   105,
+    -1,    -1,   108,   109,    59,    -1,    -1,    -1,    -1,    -1,
+    65,    -1,    -1,     3,     4,    -1,    -1,    72,    -1,    -1,
+    -1,    76,    -1,    -1,    -1,    -1,    -1,    82,    -1,    -1,
+    85,    86,    -1,    -1,    89,    90,    -1,    -1,    93,    94,
+    -1,    -1,    -1,    -1,    -1,    -1,    -1,   102,   103,   104,
+   105,    -1,    -1,   108,   109,    45,    46,    -1,    -1,    -1,
+    50,    -1,    -1,    -1,    -1,    -1,    -1,    -1,    -1,    59,
+    -1,    -1,    -1,    -1,    -1,    65,    -1,    -1,    -1,    -1,
+    -1,    -1,    72,    -1,    -1,    -1,    76,    -1,    -1,    -1,
+    -1,    -1,    82,    -1,    -1,    85,    86,    -1,    -1,    89,
+    90,    -1,    -1,    93,    -1,    44,    45,    46,    -1,    -1,
+    -1,    50,   102,   103,   104,   105,    -1,    -1,   108,   109,
+    59,    -1,    -1,    -1,    -1,    -1,    65,    -1,    -1,    -1,
+    -1,    -1,    -1,    72,    -1,    -1,    -1,    76,    -1,    -1,
+    -1,    -1,    -1,    82,    -1,    -1,    85,    86,    -1,    -1,
+    -1,    -1,    -1,    -1,    93,    -1,    -1,    -1,    -1,    -1,
+    -1,    -1,    -1,   102,   103,   104,   105,    -1,    -1,   108,
+   109
+};
+#define YYPURE 1
+
+/* -*-C-*-  Note some compilers choke on comments on `#line' lines.  */
+#line 3 "/usr/cygnus/gnupro-98r1/share/bison.simple"
+
+/* Skeleton output parser for bison,
+   Copyright (C) 1984, 1989, 1990 Free Software Foundation, Inc.
+
+   This program is free software; you can redistribute it and/or modify
+   it under the terms of the GNU General Public License as published by
+   the Free Software Foundation; either version 2, or (at your option)
+   any later version.
+
+   This program is distributed in the hope that it will be useful,
+   but WITHOUT ANY WARRANTY; without even the implied warranty of
+   MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+   GNU General Public License for more details.
+
+   You should have received a copy of the GNU General Public License
+   along with this program; if not, write to the Free Software
+   Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA.  */
+
+/* As a special exception, when this file is copied by Bison into a
+   Bison output file, you may use that output file without restriction.
+   This special exception was added by the Free Software Foundation
+   in version 1.24 of Bison.  */
+
+#ifndef alloca
+#ifdef __GNUC__
+#define alloca __builtin_alloca
+#else /* not GNU C.  */
+#if (!defined (__STDC__) && defined (sparc)) || defined (__sparc__) || defined (__sparc) || defined (__sgi)
+#include <alloca.h>
+#else /* not sparc */
+#if defined (MSDOS) && !defined (__TURBOC__)
+#include <malloc.h>
+#else /* not MSDOS, or __TURBOC__ */
+#if defined(_AIX)
+#include <malloc.h>
+ #pragma alloca
+#else /* not MSDOS, __TURBOC__, or _AIX */
+#ifdef __hpux
+#ifdef __cplusplus
+extern "C" {
+void *alloca (unsigned int);
+};
+#else /* not __cplusplus */
+void *alloca ();
+#endif /* not __cplusplus */
+#endif /* __hpux */
+#endif /* not _AIX */
+#endif /* not MSDOS, or __TURBOC__ */
+#endif /* not sparc.  */
+#endif /* not GNU C.  */
+#endif /* alloca not defined.  */
+
+/* This is the parser code that is written into each bison parser
+  when the %semantic_parser declaration is not specified in the grammar.
+  It was written by Richard Stallman by simplifying the hairy parser
+  used when %semantic_parser is specified.  */
+
+/* Note: there must be only one dollar sign in this file.
+   It is replaced by the list of actions, each action
+   as one case of the switch.  */
+
+#define yyerrok                (yyerrstatus = 0)
+#define yyclearin      (yychar = YYEMPTY)
+#define YYEMPTY                -2
+#define YYEOF          0
+#define YYACCEPT       return(0)
+#define YYABORT        return(1)
+#define YYERROR                goto yyerrlab1
+/* Like YYERROR except do call yyerror.
+   This remains here temporarily to ease the
+   transition to the new meaning of YYERROR, for GCC.
+   Once GCC version 2 has supplanted version 1, this can go.  */
+#define YYFAIL         goto yyerrlab
+#define YYRECOVERING()  (!!yyerrstatus)
+#define YYBACKUP(token, value) \
+do                                                             \
+  if (yychar == YYEMPTY && yylen == 1)                         \
+    { yychar = (token), yylval = (value);                      \
+      yychar1 = YYTRANSLATE (yychar);                          \
+      YYPOPSTACK;                                              \
+      goto yybackup;                                           \
+    }                                                          \
+  else                                                         \
+    { yyerror ("syntax error: cannot back up"); YYERROR; }     \
+while (0)
+
+#define YYTERROR       1
+#define YYERRCODE      256
+
+#ifndef YYPURE
+#define YYLEX          yylex()
+#endif
+
+#ifdef YYPURE
+#ifdef YYLSP_NEEDED
+#ifdef YYLEX_PARAM
+#define YYLEX          yylex(&yylval, &yylloc, YYLEX_PARAM)
+#else
+#define YYLEX          yylex(&yylval, &yylloc)
+#endif
+#else /* not YYLSP_NEEDED */
+#ifdef YYLEX_PARAM
+#define YYLEX          yylex(&yylval, YYLEX_PARAM)
+#else
+#define YYLEX          yylex(&yylval)
+#endif
+#endif /* not YYLSP_NEEDED */
+#endif
+
+/* If nonreentrant, generate the variables here */
+
+#ifndef YYPURE
+
+int    yychar;                 /*  the lookahead symbol                */
+YYSTYPE        yylval;                 /*  the semantic value of the           */
+                               /*  lookahead symbol                    */
+
+#ifdef YYLSP_NEEDED
+YYLTYPE yylloc;                        /*  location data for the lookahead     */
+                               /*  symbol                              */
+#endif
+
+int yynerrs;                   /*  number of parse errors so far       */
+#endif  /* not YYPURE */
+
+#if YYDEBUG != 0
+int yydebug;                   /*  nonzero means print parse trace     */
+/* Since this is uninitialized, it does not stop multiple parsers
+   from coexisting.  */
+#endif
+
+/*  YYINITDEPTH indicates the initial size of the parser's stacks      */
+
+#ifndef        YYINITDEPTH
+#define YYINITDEPTH 200
+#endif
+
+/*  YYMAXDEPTH is the maximum size the stacks can grow to
+    (effective only if the built-in stack extension method is used).  */
+
+#if YYMAXDEPTH == 0
+#undef YYMAXDEPTH
+#endif
+
+#ifndef YYMAXDEPTH
+#define YYMAXDEPTH 10000
+#endif
+
+/* Prevent warning if -Wstrict-prototypes.  */
+#ifdef __GNUC__
+int yyparse (void);
+#endif
+\f
+#if __GNUC__ > 1               /* GNU C and GNU C++ define this.  */
+#define __yy_memcpy(TO,FROM,COUNT)     __builtin_memcpy(TO,FROM,COUNT)
+#else                          /* not GNU C or C++ */
+#ifndef __cplusplus
+
+/* This is the most reliable way to avoid incompatibilities
+   in available built-in functions on various systems.  */
+static void
+__yy_memcpy (to, from, count)
+     char *to;
+     char *from;
+     int count;
+{
+  register char *f = from;
+  register char *t = to;
+  register int i = count;
+
+  while (i-- > 0)
+    *t++ = *f++;
+}
+
+#else /* __cplusplus */
+
+/* This is the most reliable way to avoid incompatibilities
+   in available built-in functions on various systems.  */
+static void
+__yy_memcpy (char *to, char *from, int count)
+{
+  register char *f = from;
+  register char *t = to;
+  register int i = count;
+
+  while (i-- > 0)
+    *t++ = *f++;
+}
+
+#endif
+#endif
+\f
+#line 196 "/usr/cygnus/gnupro-98r1/share/bison.simple"
+
+/* The user can define YYPARSE_PARAM as the name of an argument to be passed
+   into yyparse.  The argument should have type void *.
+   It should actually point to an object.
+   Grammar actions can access the variable by casting it
+   to the proper pointer type.  */
+
+#ifdef YYPARSE_PARAM
+#ifdef __cplusplus
+#define YYPARSE_PARAM_ARG void *YYPARSE_PARAM
+#define YYPARSE_PARAM_DECL
+#else /* not __cplusplus */
+#define YYPARSE_PARAM_ARG YYPARSE_PARAM
+#define YYPARSE_PARAM_DECL void *YYPARSE_PARAM;
+#endif /* not __cplusplus */
+#else /* not YYPARSE_PARAM */
+#define YYPARSE_PARAM_ARG
+#define YYPARSE_PARAM_DECL
+#endif /* not YYPARSE_PARAM */
+
+int
+yyparse(YYPARSE_PARAM_ARG)
+     YYPARSE_PARAM_DECL
+{
+  register int yystate;
+  register int yyn;
+  register short *yyssp;
+  register YYSTYPE *yyvsp;
+  int yyerrstatus;     /*  number of tokens to shift before error messages enabled */
+  int yychar1 = 0;             /*  lookahead token as an internal (translated) token number */
+
+  short        yyssa[YYINITDEPTH];     /*  the state stack                     */
+  YYSTYPE yyvsa[YYINITDEPTH];  /*  the semantic value stack            */
+
+  short *yyss = yyssa;         /*  refer to the stacks thru separate pointers */
+  YYSTYPE *yyvs = yyvsa;       /*  to allow yyoverflow to reallocate them elsewhere */
+
+#ifdef YYLSP_NEEDED
+  YYLTYPE yylsa[YYINITDEPTH];  /*  the location stack                  */
+  YYLTYPE *yyls = yylsa;
+  YYLTYPE *yylsp;
+
+#define YYPOPSTACK   (yyvsp--, yyssp--, yylsp--)
+#else
+#define YYPOPSTACK   (yyvsp--, yyssp--)
+#endif
+
+  int yystacksize = YYINITDEPTH;
+
+#ifdef YYPURE
+  int yychar;
+  YYSTYPE yylval;
+  int yynerrs;
+#ifdef YYLSP_NEEDED
+  YYLTYPE yylloc;
+#endif
+#endif
+
+  YYSTYPE yyval;               /*  the variable used to return         */
+                               /*  semantic values from the action     */
+                               /*  routines                            */
+
+  int yylen;
+
+#if YYDEBUG != 0
+  if (yydebug)
+    fprintf(stderr, "Starting parse\n");
+#endif
+
+  yystate = 0;
+  yyerrstatus = 0;
+  yynerrs = 0;
+  yychar = YYEMPTY;            /* Cause a token to be read.  */
+
+  /* Initialize stack pointers.
+     Waste one element of value and location stack
+     so that they stay on the same level as the state stack.
+     The wasted elements are never initialized.  */
+
+  yyssp = yyss - 1;
+  yyvsp = yyvs;
+#ifdef YYLSP_NEEDED
+  yylsp = yyls;
+#endif
+
+/* Push a new state, which is found in  yystate  .  */
+/* In all cases, when you get here, the value and location stacks
+   have just been pushed. so pushing a state here evens the stacks.  */
+yynewstate:
+
+  *++yyssp = yystate;
+
+  if (yyssp >= yyss + yystacksize - 1)
+    {
+      /* Give user a chance to reallocate the stack */
+      /* Use copies of these so that the &'s don't force the real ones into memory. */
+      YYSTYPE *yyvs1 = yyvs;
+      short *yyss1 = yyss;
+#ifdef YYLSP_NEEDED
+      YYLTYPE *yyls1 = yyls;
+#endif
+
+      /* Get the current used size of the three stacks, in elements.  */
+      int size = yyssp - yyss + 1;
+
+#ifdef yyoverflow
+      /* Each stack pointer address is followed by the size of
+        the data in use in that stack, in bytes.  */
+#ifdef YYLSP_NEEDED
+      /* This used to be a conditional around just the two extra args,
+        but that might be undefined if yyoverflow is a macro.  */
+      yyoverflow("parser stack overflow",
+                &yyss1, size * sizeof (*yyssp),
+                &yyvs1, size * sizeof (*yyvsp),
+                &yyls1, size * sizeof (*yylsp),
+                &yystacksize);
+#else
+      yyoverflow("parser stack overflow",
+                &yyss1, size * sizeof (*yyssp),
+                &yyvs1, size * sizeof (*yyvsp),
+                &yystacksize);
+#endif
+
+      yyss = yyss1; yyvs = yyvs1;
+#ifdef YYLSP_NEEDED
+      yyls = yyls1;
+#endif
+#else /* no yyoverflow */
+      /* Extend the stack our own way.  */
+      if (yystacksize >= YYMAXDEPTH)
+       {
+         yyerror("parser stack overflow");
+         return 2;
+       }
+      yystacksize *= 2;
+      if (yystacksize > YYMAXDEPTH)
+       yystacksize = YYMAXDEPTH;
+      yyss = (short *) alloca (yystacksize * sizeof (*yyssp));
+      __yy_memcpy ((char *)yyss, (char *)yyss1, size * sizeof (*yyssp));
+      yyvs = (YYSTYPE *) alloca (yystacksize * sizeof (*yyvsp));
+      __yy_memcpy ((char *)yyvs, (char *)yyvs1, size * sizeof (*yyvsp));
+#ifdef YYLSP_NEEDED
+      yyls = (YYLTYPE *) alloca (yystacksize * sizeof (*yylsp));
+      __yy_memcpy ((char *)yyls, (char *)yyls1, size * sizeof (*yylsp));
+#endif
+#endif /* no yyoverflow */
+
+      yyssp = yyss + size - 1;
+      yyvsp = yyvs + size - 1;
+#ifdef YYLSP_NEEDED
+      yylsp = yyls + size - 1;
+#endif
+
+#if YYDEBUG != 0
+      if (yydebug)
+       fprintf(stderr, "Stack size increased to %d\n", yystacksize);
+#endif
+
+      if (yyssp >= yyss + yystacksize - 1)
+       YYABORT;
+    }
+
+#if YYDEBUG != 0
+  if (yydebug)
+    fprintf(stderr, "Entering state %d\n", yystate);
+#endif
+
+  goto yybackup;
+ yybackup:
+
+/* Do appropriate processing given the current state.  */
+/* Read a lookahead token if we need one and don't already have one.  */
+/* yyresume: */
+
+  /* First try to decide what to do without reference to lookahead token.  */
+
+  yyn = yypact[yystate];
+  if (yyn == YYFLAG)
+    goto yydefault;
+
+  /* Not known => get a lookahead token if don't already have one.  */
+
+  /* yychar is either YYEMPTY or YYEOF
+     or a valid token in external form.  */
+
+  if (yychar == YYEMPTY)
+    {
+#if YYDEBUG != 0
+      if (yydebug)
+       fprintf(stderr, "Reading a token: ");
+#endif
+      yychar = YYLEX;
+    }
+
+  /* Convert token to internal form (in yychar1) for indexing tables with */
+
+  if (yychar <= 0)             /* This means end of input. */
+    {
+      yychar1 = 0;
+      yychar = YYEOF;          /* Don't call YYLEX any more */
+
+#if YYDEBUG != 0
+      if (yydebug)
+       fprintf(stderr, "Now at end of input.\n");
+#endif
+    }
+  else
+    {
+      yychar1 = YYTRANSLATE(yychar);
+
+#if YYDEBUG != 0
+      if (yydebug)
+       {
+         fprintf (stderr, "Next token is %d (%s", yychar, yytname[yychar1]);
+         /* Give the individual parser a way to print the precise meaning
+            of a token, for further debugging info.  */
+#ifdef YYPRINT
+         YYPRINT (stderr, yychar, yylval);
+#endif
+         fprintf (stderr, ")\n");
+       }
+#endif
+    }
+
+  yyn += yychar1;
+  if (yyn < 0 || yyn > YYLAST || yycheck[yyn] != yychar1)
+    goto yydefault;
+
+  yyn = yytable[yyn];
+
+  /* yyn is what to do for this token type in this state.
+     Negative => reduce, -yyn is rule number.
+     Positive => shift, yyn is new state.
+       New state is final state => don't bother to shift,
+       just return success.
+     0, or most negative number => error.  */
+
+  if (yyn < 0)
+    {
+      if (yyn == YYFLAG)
+       goto yyerrlab;
+      yyn = -yyn;
+      goto yyreduce;
+    }
+  else if (yyn == 0)
+    goto yyerrlab;
+
+  if (yyn == YYFINAL)
+    YYACCEPT;
+
+  /* Shift the lookahead token.  */
+
+#if YYDEBUG != 0
+  if (yydebug)
+    fprintf(stderr, "Shifting token %d (%s), ", yychar, yytname[yychar1]);
+#endif
+
+  /* Discard the token being shifted unless it is eof.  */
+  if (yychar != YYEOF)
+    yychar = YYEMPTY;
+
+  *++yyvsp = yylval;
+#ifdef YYLSP_NEEDED
+  *++yylsp = yylloc;
+#endif
+
+  /* count tokens shifted since error; after three, turn off error status.  */
+  if (yyerrstatus) yyerrstatus--;
+
+  yystate = yyn;
+  goto yynewstate;
+
+/* Do the default action for the current state.  */
+yydefault:
+
+  yyn = yydefact[yystate];
+  if (yyn == 0)
+    goto yyerrlab;
+
+/* Do a reduction.  yyn is the number of a rule to reduce with.  */
+yyreduce:
+  yylen = yyr2[yyn];
+  if (yylen > 0)
+    yyval = yyvsp[1-yylen]; /* implement default value of the action */
+
+#if YYDEBUG != 0
+  if (yydebug)
+    {
+      int i;
+
+      fprintf (stderr, "Reducing via rule %d (line %d), ",
+              yyn, yyrline[yyn]);
+
+      /* Print the symbols being reduced, and their result.  */
+      for (i = yyprhs[yyn]; yyrhs[i] > 0; i++)
+       fprintf (stderr, "%s ", yytname[yyrhs[i]]);
+      fprintf (stderr, " -> %s\n", yytname[yyr1[yyn]]);
+    }
+#endif
+
+
+  switch (yyn) {
+
+case 1:
+#line 253 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{;
+    break;}
+case 18:
+#line 297 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ 
+                 yyval.node = build_java_array_type (yyvsp[-2].node, -1);
+                 CLASS_LOADED_P (yyval.node) = 1;
+               ;
+    break;}
+case 19:
+#line 302 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = build_unresolved_array_type (yyvsp[-2].node); ;
+    break;}
+case 20:
+#line 304 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = build_unresolved_array_type (yyvsp[-2].node); ;
+    break;}
+case 21:
+#line 306 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{RULE ("']' expected"); RECOVER;;
+    break;}
+case 22:
+#line 308 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{RULE ("']' expected"); RECOVER;;
+    break;}
+case 26:
+#line 323 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = make_qualified_name (yyvsp[-2].node, yyvsp[0].node, yyvsp[-1].operator.location); ;
+    break;}
+case 28:
+#line 332 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyval.node = NULL;;
+    break;}
+case 36:
+#line 344 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 yyval.node = NULL;
+               ;
+    break;}
+case 37:
+#line 348 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 yyval.node = NULL;
+               ;
+    break;}
+case 40:
+#line 360 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ ctxp->package = EXPR_WFL_NODE (yyvsp[-1].node); ;
+    break;}
+case 41:
+#line 362 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Missing name"); RECOVER;;
+    break;}
+case 42:
+#line 364 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("';' expected"); RECOVER;;
+    break;}
+case 45:
+#line 374 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 tree name = EXPR_WFL_NODE (yyvsp[-1].node), node, last_name;
+                 int   i = IDENTIFIER_LENGTH (name)-1;
+                 char *last = &IDENTIFIER_POINTER (name)[i];
+                 while (last != IDENTIFIER_POINTER (name))
+                   {
+                     if (last [0] == '.')
+                       break;
+                     last--;
+                   }
+                 last_name = get_identifier (++last);
+                 if (IS_A_SINGLE_IMPORT_CLASSFILE_NAME_P (last_name))
+                   {
+                     tree err = find_name_in_single_imports (last_name);
+                     if (err && err != name)
+                       parse_error_context
+                         (yyvsp[-1].node, "Ambiguous class: `%s' and `%s'",
+                          IDENTIFIER_POINTER (name), 
+                          IDENTIFIER_POINTER (err));
+                   }
+                 else
+                   {
+                     IS_A_SINGLE_IMPORT_CLASSFILE_NAME_P (last_name) = 1;
+                     node = build_tree_list (yyvsp[-1].node, last_name);
+                     TREE_CHAIN (node) = ctxp->import_list;
+                     ctxp->import_list = node;
+                   }
+               ;
+    break;}
+case 46:
+#line 403 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Missing name"); RECOVER;;
+    break;}
+case 47:
+#line 405 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("';' expected"); RECOVER;;
+    break;}
+case 48:
+#line 410 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 tree name = EXPR_WFL_NODE (yyvsp[-3].node);
+                 tree node = build_tree_list (yyvsp[-3].node, NULL_TREE);
+                 if (!IS_AN_IMPORT_ON_DEMAND_P (name))
+                   {
+                     read_import_dir (yyvsp[-3].node);
+                     IS_AN_IMPORT_ON_DEMAND_P (name) = 1;
+                   }
+                 TREE_CHAIN (node) = ctxp->import_demand_list;
+                 ctxp->import_demand_list = node;
+               ;
+    break;}
+case 49:
+#line 422 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("'*' expected"); RECOVER;;
+    break;}
+case 50:
+#line 424 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("';' expected"); RECOVER;;
+    break;}
+case 51:
+#line 429 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 maybe_generate_clinit ();
+                 yyval.node = yyvsp[0].node;
+               ;
+    break;}
+case 53:
+#line 435 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = NULL; ;
+    break;}
+case 54:
+#line 437 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 YYERROR_NOW;
+                 yyerror ("Class or interface declaration expected");
+               ;
+    break;}
+case 55:
+#line 448 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 yyval.value = (1 << yyvsp[0].value);
+               ;
+    break;}
+case 56:
+#line 452 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 int acc = (1 << yyvsp[0].value);
+                 if (yyval.value & acc)
+                   parse_error_context 
+                     (ctxp->modifier_ctx [yyvsp[0].value], "Modifier `%s' declared twice",
+                      java_accstring_lookup (acc));
+                 else
+                   {
+                     yyval.value |= acc;
+                   }
+               ;
+    break;}
+case 57:
+#line 468 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ create_class (yyvsp[-4].value, yyvsp[-2].node, yyvsp[-1].node, yyvsp[0].node); ;
+    break;}
+case 58:
+#line 470 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ 
+                 yyval.node = yyvsp[0].node;
+               ;
+    break;}
+case 59:
+#line 474 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ create_class (0, yyvsp[-2].node, yyvsp[-1].node, yyvsp[0].node); ;
+    break;}
+case 60:
+#line 476 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{      
+                 yyval.node = yyvsp[0].node;
+               ;
+    break;}
+case 61:
+#line 480 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Missing class name"); RECOVER;;
+    break;}
+case 62:
+#line 482 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Missing class name"); RECOVER;;
+    break;}
+case 63:
+#line 484 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{if (!ctxp->class_err) yyerror ("'{' expected"); DRECOVER(class1);;
+    break;}
+case 64:
+#line 486 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{if (!ctxp->class_err) yyerror ("'{' expected"); RECOVER;;
+    break;}
+case 65:
+#line 490 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = NULL; ;
+    break;}
+case 66:
+#line 492 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = yyvsp[0].node; ;
+    break;}
+case 67:
+#line 494 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("'{' expected"); ctxp->class_err=1;;
+    break;}
+case 68:
+#line 496 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Missing super class name"); ctxp->class_err=1;;
+    break;}
+case 69:
+#line 500 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = NULL_TREE; ;
+    break;}
+case 70:
+#line 502 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = yyvsp[0].node; ;
+    break;}
+case 71:
+#line 504 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 ctxp->class_err=1;
+                 yyerror ("Missing interface name"); 
+               ;
+    break;}
+case 72:
+#line 512 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ 
+                 ctxp->interface_number = 1;
+                 yyval.node = build_tree_list (yyvsp[0].node, NULL_TREE);
+               ;
+    break;}
+case 73:
+#line 517 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ 
+                 ctxp->interface_number++;
+                 yyval.node = chainon (yyvsp[-2].node, build_tree_list (yyvsp[0].node, NULL_TREE));
+               ;
+    break;}
+case 74:
+#line 522 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Missing interface name"); RECOVER;;
+    break;}
+case 75:
+#line 527 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = ctxp->current_parsed_class; ;
+    break;}
+case 76:
+#line 529 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = ctxp->current_parsed_class; ;
+    break;}
+case 87:
+#line 554 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ register_fields (0, yyvsp[-2].node, yyvsp[-1].node); ;
+    break;}
+case 88:
+#line 556 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 int acc_count = 0;
+
+                 check_modifiers 
+                   ("Illegal modifier `%s' for field declaration",
+                    yyvsp[-3].value, FIELD_MODIFIERS);
+                 check_modifiers_consistency (yyvsp[-3].value);
+                 register_fields (yyvsp[-3].value, yyvsp[-2].node, yyvsp[-1].node);
+               ;
+    break;}
+case 90:
+#line 571 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = chainon (yyvsp[-2].node, yyvsp[0].node); ;
+    break;}
+case 91:
+#line 573 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Missing term"); RECOVER;;
+    break;}
+case 92:
+#line 578 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = build_tree_list (yyvsp[0].node, NULL_TREE); ;
+    break;}
+case 93:
+#line 580 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ 
+                 if (java_error_count)
+                   yyvsp[0].node = NULL_TREE;
+                 yyval.node = build_tree_list 
+                   (yyvsp[-2].node, build_assignment (yyvsp[-1].operator.token, yyvsp[-1].operator.location, yyvsp[-2].node, yyvsp[0].node));
+               ;
+    break;}
+case 94:
+#line 587 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 yyerror ("Missing variable initializer");
+                 yyval.node = build_tree_list (yyvsp[-2].node, NULL_TREE);
+                 RECOVER;
+               ;
+    break;}
+case 95:
+#line 593 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 yyerror ("';' expected");
+                 yyval.node = build_tree_list (yyvsp[-3].node, NULL_TREE);
+                 RECOVER;
+               ;
+    break;}
+case 97:
+#line 603 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 yyval.node = NULL;    /* FIXME */
+               ;
+    break;}
+case 98:
+#line 607 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Invalid declaration"); DRECOVER(vdi);;
+    break;}
+case 99:
+#line 609 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("']' expected"); DRECOVER(vdi);;
+    break;}
+case 100:
+#line 611 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Unbalanced ']'"); DRECOVER(vdi);;
+    break;}
+case 102:
+#line 617 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = NULL; ;
+    break;}
+case 103:
+#line 623 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 current_function_decl = yyvsp[0].node;
+                 source_start_java_method (current_function_decl);
+               ;
+    break;}
+case 104:
+#line 628 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ 
+                 BLOCK_EXPR_BODY 
+                   (DECL_FUNCTION_BODY (current_function_decl)) = yyvsp[0].node;
+                 maybe_absorb_scoping_blocks ();
+                 exit_block (); /* Exit function's body. */
+
+                 /* Merge last line of the function with first line,
+                    directly in the function decl. It will be used to
+                    emit correct debug info. */
+                 DECL_SOURCE_LINE_MERGE (current_function_decl,
+                                         ctxp->last_ccb_indent1);
+               ;
+    break;}
+case 105:
+#line 641 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{YYNOT_TWICE yyerror ("'{' expected"); RECOVER;;
+    break;}
+case 106:
+#line 646 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = method_header (0, yyvsp[-2].node, yyvsp[-1].node, NULL); ;
+    break;}
+case 107:
+#line 648 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = method_header (0, void_type_node, yyvsp[-1].node, NULL); ;
+    break;}
+case 108:
+#line 650 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = method_header (yyvsp[-3].value, yyvsp[-2].node, yyvsp[-1].node, NULL); ;
+    break;}
+case 109:
+#line 652 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = method_header (yyvsp[-3].value, void_type_node, yyvsp[-1].node, NULL); ;
+    break;}
+case 110:
+#line 654 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{RECOVER;;
+    break;}
+case 111:
+#line 656 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{RECOVER;;
+    break;}
+case 112:
+#line 658 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Identifier expected"); RECOVER;;
+    break;}
+case 113:
+#line 660 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Identifier expected"); RECOVER;;
+    break;}
+case 114:
+#line 662 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 yyerror ("Invalid method declaration, return type required");
+                 RECOVER;
+               ;
+    break;}
+case 115:
+#line 670 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = method_declarator (yyvsp[-2].node, NULL_TREE); ;
+    break;}
+case 116:
+#line 672 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = method_declarator (yyvsp[-3].node, yyvsp[-1].node); ;
+    break;}
+case 117:
+#line 674 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 /* Issue a warning here: obsolete declaration. FIXME */
+                 yyval.node = NULL;    /* FIXME */
+               ;
+    break;}
+case 118:
+#line 679 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("')' expected"); DRECOVER(method_declarator);;
+    break;}
+case 119:
+#line 681 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("']' expected"); RECOVER;;
+    break;}
+case 120:
+#line 686 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 ctxp->formal_parameter_number = 1;
+               ;
+    break;}
+case 121:
+#line 690 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 ctxp->formal_parameter_number += 1;
+                 yyval.node = chainon (yyvsp[-2].node, yyvsp[0].node);
+               ;
+    break;}
+case 122:
+#line 695 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Missing formal parameter term"); RECOVER;;
+    break;}
+case 123:
+#line 700 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 yyval.node = build_tree_list (yyvsp[0].node, yyvsp[-1].node);
+               ;
+    break;}
+case 124:
+#line 704 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 SOURCE_FRONTEND_DEBUG (("Modifiers: %d", yyvsp[-2].value));
+                 yyval.node = NULL;    /* FIXME */
+               ;
+    break;}
+case 125:
+#line 709 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Missing identifier"); RECOVER;;
+    break;}
+case 126:
+#line 711 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 SOURCE_FRONTEND_DEBUG (("Modifiers: %d", yyvsp[-2].value));
+                 yyerror ("Missing identifier"); RECOVER;
+               ;
+    break;}
+case 129:
+#line 720 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Missing class type term"); RECOVER;;
+    break;}
+case 132:
+#line 727 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Missing class type term"); RECOVER;;
+    break;}
+case 135:
+#line 734 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = NULL_TREE; ;
+    break;}
+case 136:
+#line 740 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 RULE ("STATIC_INITIALIZER");
+               ;
+    break;}
+case 137:
+#line 744 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 RULE ("STATIC_INITIALIZER");
+               ;
+    break;}
+case 138:
+#line 751 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 SOURCE_FRONTEND_DEBUG (("Modifiers: %d", yyvsp[0].value));
+               ;
+    break;}
+case 139:
+#line 763 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 RULE ("CONSTRUCTOR_DECLARATION");
+               ;
+    break;}
+case 140:
+#line 767 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 SOURCE_FRONTEND_DEBUG (("Modifiers: %d", yyvsp[-3].value));
+                 RULE ("CONSTRUCTOR_DECLARATION (modifier)");
+               ;
+    break;}
+case 141:
+#line 773 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 RULE ("CONSTRUCTOR_DECLARATION");
+               ;
+    break;}
+case 142:
+#line 778 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 SOURCE_FRONTEND_DEBUG (("Modifiers: %d", yyvsp[-4].value));
+                 RULE ("CONSTRUCTOR_DECLARATION (modifier)");
+               ;
+    break;}
+case 152:
+#line 806 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{RULE ("explicit_constructor_invocation (X.super)");;
+    break;}
+case 153:
+#line 811 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 tree wfl = build_wfl_node (this_identifier_node, input_filename, 0, 0);
+                 EXPR_WFL_LINECOL (wfl) = yyvsp[0].operator.location;
+                 yyval.node = wfl;
+               ;
+    break;}
+case 154:
+#line 817 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 tree wfl = build_wfl_node (super_identifier_node, input_filename, 0, 0);
+                 EXPR_WFL_LINECOL (wfl) = yyvsp[0].operator.location;
+                 yyval.node = wfl;
+               ;
+    break;}
+case 155:
+#line 828 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ create_interface (0, yyvsp[0].node, NULL_TREE); ;
+    break;}
+case 156:
+#line 830 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 yyval.node = yyvsp[0].node;
+               ;
+    break;}
+case 157:
+#line 834 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ create_interface (yyvsp[-2].value, yyvsp[0].node, NULL_TREE); ;
+    break;}
+case 158:
+#line 836 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 yyval.node = yyvsp[0].node;
+               ;
+    break;}
+case 159:
+#line 840 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ create_interface (0, yyvsp[-1].node, yyvsp[0].node); ;
+    break;}
+case 160:
+#line 842 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 yyval.node = yyvsp[0].node;
+               ;
+    break;}
+case 161:
+#line 846 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ create_interface (yyvsp[-3].value, yyvsp[-1].node, yyvsp[0].node); ;
+    break;}
+case 162:
+#line 848 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 yyval.node = yyvsp[0].node;
+               ;
+    break;}
+case 163:
+#line 852 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("(here)'{' expected"); RECOVER;;
+    break;}
+case 164:
+#line 854 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("(there)'{' expected"); RECOVER;;
+    break;}
+case 165:
+#line 859 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ 
+                 ctxp->interface_number = 1;
+                 yyval.node = build_tree_list (yyvsp[0].node, NULL_TREE);
+               ;
+    break;}
+case 166:
+#line 864 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ 
+                 ctxp->interface_number++;
+                 yyval.node = chainon (yyvsp[-2].node, build_tree_list (yyvsp[0].node, NULL_TREE));
+               ;
+    break;}
+case 167:
+#line 869 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Invalid interface type"); RECOVER;;
+    break;}
+case 168:
+#line 871 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Missing term"); RECOVER;;
+    break;}
+case 169:
+#line 876 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = NULL_TREE; ;
+    break;}
+case 170:
+#line 878 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = NULL_TREE; ;
+    break;}
+case 178:
+#line 899 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ 
+                 check_abstract_method_header (yyvsp[-1].node);
+                 current_function_decl = NULL_TREE; /* FIXME ? */
+               ;
+    break;}
+case 179:
+#line 904 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("';' expected"); RECOVER;;
+    break;}
+case 180:
+#line 910 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 RULE ("ARRAY_INITIALIZER (empty)");
+               ;
+    break;}
+case 181:
+#line 914 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 RULE ("ARRAY_INITIALIZER (variable)");
+               ;
+    break;}
+case 182:
+#line 918 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 RULE ("ARRAY_INITIALIZER (,)");
+               ;
+    break;}
+case 183:
+#line 922 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 RULE ("ARRAY_INITIALIZER (variable, ,)");
+               ;
+    break;}
+case 186:
+#line 931 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Missing term"); RECOVER;;
+    break;}
+case 187:
+#line 937 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = size_zero_node; ;
+    break;}
+case 188:
+#line 939 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ enter_block (); ;
+    break;}
+case 189:
+#line 942 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ 
+                 maybe_absorb_scoping_blocks ();
+                 yyval.node = exit_block ();
+               ;
+    break;}
+case 193:
+#line 956 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = java_method_add_stmt (current_function_decl, yyvsp[0].node); ;
+    break;}
+case 196:
+#line 966 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ declare_local_variables (0, yyvsp[-1].node, yyvsp[0].node); ;
+    break;}
+case 197:
+#line 968 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ declare_local_variables (yyvsp[-2].value, yyvsp[-1].node, yyvsp[0].node); ;
+    break;}
+case 199:
+#line 974 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ RULE ("STATEMENT (labeled)"); ;
+    break;}
+case 200:
+#line 976 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ RULE ("STATEMENT (if-then)"); ;
+    break;}
+case 201:
+#line 978 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ RULE ("STATEMENT (if-then-else)"); ;
+    break;}
+case 202:
+#line 980 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ RULE ("STATEMENT (while)"); ;
+    break;}
+case 203:
+#line 982 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ 
+                 /* If the for loop is unlabeled, we must return the
+                    block it was defined it. It our last chance to
+                    get a hold on it. */
+                 if (!LOOP_HAS_LABEL_P (yyval.node))
+                   yyval.node = exit_block ();
+               ;
+    break;}
+case 205:
+#line 994 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ RULE ("NSI STATEMENT (labeled)"); ;
+    break;}
+case 206:
+#line 996 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ RULE ("NSI STATEMENT (if-then-else)"); ;
+    break;}
+case 207:
+#line 998 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ RULE ("NSI STATEMENT (while)"); ;
+    break;}
+case 208:
+#line 1000 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ RULE ("NSI STATEMENT (for)"); ;
+    break;}
+case 209:
+#line 1005 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ RULE ("STATEMENT (block)"); ;
+    break;}
+case 210:
+#line 1007 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ RULE ("STATEMENT (empty)"); ;
+    break;}
+case 211:
+#line 1009 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ RULE ("STATEMENT (expression)"); ;
+    break;}
+case 212:
+#line 1011 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ RULE ("STATEMENT (switch)"); ;
+    break;}
+case 213:
+#line 1013 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ RULE ("STATEMENT (do)"); ;
+    break;}
+case 214:
+#line 1015 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ RULE ("STATEMENT (break)"); ;
+    break;}
+case 215:
+#line 1017 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ RULE ("STATEMENT (continue)"); ;
+    break;}
+case 217:
+#line 1020 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ RULE ("STATEMENT (synchronized)"); ;
+    break;}
+case 218:
+#line 1022 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ RULE ("STATEMENT (throw)"); ;
+    break;}
+case 219:
+#line 1024 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ RULE ("STATEMENT (try)"); ;
+    break;}
+case 220:
+#line 1029 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = size_zero_node; ;
+    break;}
+case 221:
+#line 1034 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 yyval.node = build_labeled_block (EXPR_WFL_LINECOL (yyvsp[-1].node), 
+                                           EXPR_WFL_NODE (yyvsp[-1].node), yyvsp[-1].node);
+                 pushlevel (2);
+                 push_labeled_block (yyval.node);
+                 PUSH_LABELED_BLOCK (yyval.node);
+               ;
+    break;}
+case 222:
+#line 1045 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ 
+                 yyval.node = complete_labeled_statement (yyvsp[-1].node, yyvsp[0].node);
+                 pop_labeled_block ();
+                 POP_LABELED_BLOCK ();
+               ;
+    break;}
+case 223:
+#line 1051 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("':' expected"); RECOVER;;
+    break;}
+case 224:
+#line 1056 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ 
+                 yyval.node = complete_labeled_statement (yyvsp[-1].node, yyvsp[0].node);
+                 pop_labeled_block ();
+                 POP_LABELED_BLOCK ();
+               ;
+    break;}
+case 225:
+#line 1067 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 /* We have a statement. Generate a WFL around it so
+                    we can debug it */
+                 yyval.node = build_expr_wfl (yyvsp[-1].node, input_filename, lineno, 0);
+                 /* We know we have a statement, so set the debug
+                     info to be eventually generate here. */
+                 yyval.node = JAVA_MAYBE_GENERATE_DEBUG_INFO (yyval.node);
+               ;
+    break;}
+case 226:
+#line 1076 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 if (ctxp->prevent_ese != lineno)
+                   yyerror ("Invalid expression statement");
+                 DRECOVER (expr_stmt);
+               ;
+    break;}
+case 227:
+#line 1082 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 if (ctxp->prevent_ese != lineno)
+                   yyerror ("Invalid expression statement");
+                 DRECOVER (expr_stmt);
+               ;
+    break;}
+case 228:
+#line 1088 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 if (ctxp->prevent_ese != lineno)
+                   yyerror ("Invalid expression statement");
+                 DRECOVER (expr_stmt);
+               ;
+    break;}
+case 229:
+#line 1094 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("')' expected"); RECOVER;;
+    break;}
+case 230:
+#line 1096 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("';' expected"); RECOVER;;
+    break;}
+case 231:
+#line 1098 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("')' expected"); RECOVER;;
+    break;}
+case 232:
+#line 1100 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("';' expected"); RECOVER;;
+    break;}
+case 233:
+#line 1102 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("'(' expected"); RECOVER;;
+    break;}
+case 234:
+#line 1104 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("')' expected"); RECOVER;;
+    break;}
+case 235:
+#line 1106 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("')' expected"); RECOVER;;
+    break;}
+case 236:
+#line 1108 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("';' expected"); RECOVER;;
+    break;}
+case 237:
+#line 1110 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("';' expected"); RECOVER;;
+    break;}
+case 239:
+#line 1116 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 RULE ("++INCREMENT");
+               ;
+    break;}
+case 240:
+#line 1120 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 RULE ("--DECREMENT");
+               ;
+    break;}
+case 241:
+#line 1124 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 RULE ("INCREMENT++");
+               ;
+    break;}
+case 242:
+#line 1128 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 RULE ("DECREMENT--");
+               ;
+    break;}
+case 244:
+#line 1133 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 RULE ("INSTANCE CREATION");
+               ;
+    break;}
+case 245:
+#line 1140 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = build_if_else_statement (yyvsp[-3].operator.location, yyvsp[-2].node, yyvsp[0].node, NULL_TREE); ;
+    break;}
+case 246:
+#line 1142 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("'(' expected"); RECOVER;;
+    break;}
+case 247:
+#line 1144 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Missing term"); RECOVER;;
+    break;}
+case 248:
+#line 1146 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("')' expected"); RECOVER;;
+    break;}
+case 249:
+#line 1151 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = build_if_else_statement (yyvsp[-5].operator.location, yyvsp[-4].node, yyvsp[-2].node, yyvsp[0].node); ;
+    break;}
+case 250:
+#line 1156 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = build_if_else_statement (yyvsp[-5].operator.location, yyvsp[-4].node, yyvsp[-2].node, yyvsp[0].node); ;
+    break;}
+case 252:
+#line 1162 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("'(' expected"); RECOVER;;
+    break;}
+case 253:
+#line 1164 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Missing term or ')'"); DRECOVER(switch_statement);;
+    break;}
+case 254:
+#line 1166 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("'{' expected"); RECOVER;;
+    break;}
+case 266:
+#line 1195 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Missing or invalid constant expression"); RECOVER;;
+    break;}
+case 267:
+#line 1197 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("':' expected"); RECOVER;;
+    break;}
+case 268:
+#line 1199 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("':' expected"); RECOVER;;
+    break;}
+case 269:
+#line 1204 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ 
+                 tree body = build_loop_body (yyvsp[-2].operator.location, yyvsp[-1].node, 0);
+                 yyval.node = build_new_loop (body);
+               ;
+    break;}
+case 270:
+#line 1212 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = complete_loop_body (0, NULL_TREE, yyvsp[0].node, 0); ;
+    break;}
+case 271:
+#line 1214 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{YYERROR_NOW; yyerror ("'(' expected"); RECOVER;;
+    break;}
+case 272:
+#line 1216 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Missing term and ')' expected"); RECOVER;;
+    break;}
+case 273:
+#line 1218 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("')' expected"); RECOVER;;
+    break;}
+case 274:
+#line 1223 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = complete_loop_body (0, NULL_TREE, yyvsp[0].node, 0); ;
+    break;}
+case 275:
+#line 1228 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ 
+                 tree body = build_loop_body (0, NULL_TREE, 1);
+                 yyval.node = build_new_loop (body);
+               ;
+    break;}
+case 276:
+#line 1237 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = complete_loop_body (yyvsp[-3].operator.location, yyvsp[-2].node, yyvsp[-5].node, 1); ;
+    break;}
+case 277:
+#line 1242 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = complete_for_loop (EXPR_WFL_LINECOL (yyvsp[-4].node), yyvsp[-4].node, yyvsp[-2].node, yyvsp[0].node);;
+    break;}
+case 278:
+#line 1244 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ 
+                 yyval.node = complete_for_loop (0, NULL_TREE, yyvsp[-2].node, yyvsp[0].node);
+                 /* We have not condition, so we get rid of the EXIT_EXPR */
+                 LOOP_EXPR_BODY_CONDITION_EXPR (LOOP_EXPR_BODY (yyval.node), 0) = 
+                   size_zero_node;
+               ;
+    break;}
+case 279:
+#line 1251 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Invalid control expression"); RECOVER;;
+    break;}
+case 280:
+#line 1253 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Invalid update expression"); RECOVER;;
+    break;}
+case 281:
+#line 1255 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Invalid update expression"); RECOVER;;
+    break;}
+case 282:
+#line 1260 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = complete_for_loop (EXPR_WFL_LINECOL (yyvsp[-4].node), yyvsp[-4].node, yyvsp[-2].node, yyvsp[0].node);;
+    break;}
+case 283:
+#line 1262 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ 
+                 yyval.node = complete_for_loop (0, NULL_TREE, yyvsp[-2].node, yyvsp[0].node);
+                 /* We have not condition, so we get rid of the EXIT_EXPR */
+                 LOOP_EXPR_BODY_CONDITION_EXPR (LOOP_EXPR_BODY (yyval.node), 0) = 
+                   size_zero_node;
+               ;
+    break;}
+case 284:
+#line 1272 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ 
+                 /* This scope defined for local variable that may be
+                     defined within the scope of the for loop */
+                 enter_block (); 
+               ;
+    break;}
+case 285:
+#line 1278 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("'(' expected"); DRECOVER(for_1);;
+    break;}
+case 286:
+#line 1280 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Invalid init statement"); RECOVER;;
+    break;}
+case 287:
+#line 1285 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ 
+                 /* We now declare the loop body. The loop is
+                     declared as a for loop. */
+                 tree body = build_loop_body (0, NULL_TREE, 0);
+                 yyval.node =  build_new_loop (body);
+                 IS_FOR_LOOP_P (yyval.node) = 1;
+                 /* The loop is added to the current block the for
+                     statement is defined within */
+                 java_method_add_stmt (current_function_decl, yyval.node);
+               ;
+    break;}
+case 288:
+#line 1297 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = size_zero_node; ;
+    break;}
+case 289:
+#line 1299 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ 
+                 /* Init statement recorded within the previously
+                     defined block scope */
+                 yyval.node = java_method_add_stmt (current_function_decl, yyvsp[0].node);
+               ;
+    break;}
+case 290:
+#line 1305 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ 
+                 /* Local variable are recorded within the previously
+                    defined block scope */
+                 yyval.node = NULL_TREE;
+               ;
+    break;}
+case 291:
+#line 1311 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("';' expected"); DRECOVER(for_init_1);;
+    break;}
+case 292:
+#line 1315 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyval.node = size_zero_node;;
+    break;}
+case 293:
+#line 1317 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = build_debugable_stmt (BUILD_LOCATION (), yyvsp[0].node); ;
+    break;}
+case 294:
+#line 1322 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = add_stmt_to_compound (NULL_TREE, NULL_TREE, yyvsp[0].node); ;
+    break;}
+case 295:
+#line 1324 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = add_stmt_to_compound (yyvsp[-2].node, NULL_TREE, yyvsp[0].node); ;
+    break;}
+case 296:
+#line 1326 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Missing term"); RECOVER;;
+    break;}
+case 297:
+#line 1331 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = build_bc_statement (yyvsp[-1].operator.location, 1, NULL_TREE); ;
+    break;}
+case 298:
+#line 1333 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = build_bc_statement (yyvsp[-2].operator.location, 1, yyvsp[-1].node); ;
+    break;}
+case 299:
+#line 1335 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Missing term"); RECOVER;;
+    break;}
+case 300:
+#line 1337 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("';' expected"); RECOVER;;
+    break;}
+case 301:
+#line 1342 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = build_bc_statement (yyvsp[-1].operator.location, 0, NULL_TREE); ;
+    break;}
+case 302:
+#line 1344 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = build_bc_statement (yyvsp[-2].operator.location, 0, yyvsp[-1].node); ;
+    break;}
+case 303:
+#line 1346 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Missing term"); RECOVER;;
+    break;}
+case 304:
+#line 1348 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("';' expected"); RECOVER;;
+    break;}
+case 305:
+#line 1353 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = build_return (yyvsp[-1].operator.location, NULL_TREE); ;
+    break;}
+case 306:
+#line 1355 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = build_return (yyvsp[-2].operator.location, yyvsp[-1].node); ;
+    break;}
+case 307:
+#line 1357 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Missing term"); RECOVER;;
+    break;}
+case 308:
+#line 1359 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("';' expected"); RECOVER;;
+    break;}
+case 310:
+#line 1365 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Missing term"); RECOVER;;
+    break;}
+case 311:
+#line 1367 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("';' expected"); RECOVER;;
+    break;}
+case 313:
+#line 1373 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("'{' expected"); RECOVER;;
+    break;}
+case 314:
+#line 1375 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("'(' expected"); RECOVER;;
+    break;}
+case 315:
+#line 1377 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Missing term"); RECOVER;;
+    break;}
+case 316:
+#line 1379 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Missing term"); RECOVER;;
+    break;}
+case 317:
+#line 1384 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 SOURCE_FRONTEND_DEBUG (("Modifiers: %d", yyvsp[0].value));
+               ;
+    break;}
+case 321:
+#line 1394 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("'{' expected"); DRECOVER (try_statement);;
+    break;}
+case 325:
+#line 1405 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("'(' expected"); RECOVER;;
+    break;}
+case 326:
+#line 1407 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Missing term"); DRECOVER (1);;
+    break;}
+case 327:
+#line 1409 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Missing term"); DRECOVER (2);;
+    break;}
+case 329:
+#line 1415 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("'{' expected"); RECOVER; ;
+    break;}
+case 333:
+#line 1427 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = build_this (yyvsp[0].operator.location); ;
+    break;}
+case 334:
+#line 1429 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyval.node = yyvsp[-1].node;;
+    break;}
+case 343:
+#line 1445 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("')' expected"); RECOVER;;
+    break;}
+case 344:
+#line 1447 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("'class' or 'this' expected" ); RECOVER;;
+    break;}
+case 345:
+#line 1449 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("'class' expected" ); RECOVER;;
+    break;}
+case 346:
+#line 1451 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("'class' expected" ); RECOVER;;
+    break;}
+case 347:
+#line 1456 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 yyval.node = build_method_invocation (yyvsp[-3].node, yyvsp[-1].node);
+                 TREE_SET_CODE (yyval.node, JAVA_NEW_CLASS_EXPR);
+               ;
+    break;}
+case 348:
+#line 1461 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 yyval.node = build_method_invocation (yyvsp[-2].node, NULL_TREE);
+                 TREE_SET_CODE (yyval.node, JAVA_NEW_CLASS_EXPR);
+               ;
+    break;}
+case 349:
+#line 1469 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyval.node = yyvsp[-4].node;;
+    break;}
+case 350:
+#line 1471 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyval.node = yyvsp[-3].node;;
+    break;}
+case 355:
+#line 1480 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("'(' expected"); DRECOVER(new_1);;
+    break;}
+case 356:
+#line 1482 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("'(' expected"); RECOVER;;
+    break;}
+case 357:
+#line 1484 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("')' or term expected"); RECOVER;;
+    break;}
+case 358:
+#line 1486 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("')' expected"); RECOVER;;
+    break;}
+case 359:
+#line 1488 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{YYERROR_NOW; yyerror ("Identifier expected"); RECOVER;;
+    break;}
+case 360:
+#line 1490 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("'(' expected"); RECOVER;;
+    break;}
+case 363:
+#line 1500 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ 
+                 yyval.node = tree_cons (NULL_TREE, yyvsp[0].node, NULL_TREE);
+                 ctxp->formal_parameter_number = 1; 
+               ;
+    break;}
+case 364:
+#line 1505 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 ctxp->formal_parameter_number += 1;
+                 yyval.node = tree_cons (NULL_TREE, yyvsp[0].node, yyvsp[-2].node);
+               ;
+    break;}
+case 365:
+#line 1510 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Missing term"); RECOVER;;
+    break;}
+case 366:
+#line 1515 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = build_newarray_node (yyvsp[-1].node, yyvsp[0].node, 0); ;
+    break;}
+case 367:
+#line 1517 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = build_newarray_node (yyvsp[-1].node, yyvsp[0].node, 0); ;
+    break;}
+case 368:
+#line 1519 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = build_newarray_node (yyvsp[-2].node, yyvsp[-1].node, ctxp->osb_number); ;
+    break;}
+case 369:
+#line 1521 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = build_newarray_node (yyvsp[-2].node, yyvsp[-1].node, ctxp->osb_number); ;
+    break;}
+case 370:
+#line 1525 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyval.node = yyvsp[-2].node;;
+    break;}
+case 371:
+#line 1527 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyval.node = yyvsp[-2].node;;
+    break;}
+case 372:
+#line 1529 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("'[' expected"); DRECOVER ("]");;
+    break;}
+case 373:
+#line 1531 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("']' expected"); RECOVER;;
+    break;}
+case 374:
+#line 1536 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = build_tree_list (NULL_TREE, yyvsp[0].node); ;
+    break;}
+case 375:
+#line 1538 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = tree_cons (NULL_TREE, yyvsp[0].node, yyval.node); ;
+    break;}
+case 376:
+#line 1543 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ 
+                 EXPR_WFL_LINECOL (yyvsp[-1].node) = yyvsp[-2].operator.location;
+                 yyval.node = yyvsp[-1].node;
+               ;
+    break;}
+case 377:
+#line 1548 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("']' expected"); RECOVER;;
+    break;}
+case 378:
+#line 1550 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 yyerror ("Missing term");
+                 yyerror ("']' expected");
+                 RECOVER;
+               ;
+    break;}
+case 379:
+#line 1559 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ ctxp->osb_number = 1; ;
+    break;}
+case 380:
+#line 1561 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ ctxp->osb_number++; ;
+    break;}
+case 381:
+#line 1563 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyerror ("']' expected"); RECOVER;;
+    break;}
+case 382:
+#line 1568 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = make_qualified_primary (yyvsp[-2].node, yyvsp[0].node, yyvsp[-1].operator.location); ;
+    break;}
+case 383:
+#line 1570 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 tree super_wfl = 
+                   build_wfl_node (super_identifier_node, input_filename, 0, 0);
+                 EXPR_WFL_LINECOL (super_wfl) = yyvsp[-2].operator.location;
+                 yyval.node = make_qualified_name (super_wfl, yyvsp[0].node, yyvsp[-1].operator.location);
+               ;
+    break;}
+case 384:
+#line 1577 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Field expected"); DRECOVER (super_field_acces);;
+    break;}
+case 385:
+#line 1582 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = build_method_invocation (yyvsp[-2].node, NULL_TREE); ;
+    break;}
+case 386:
+#line 1584 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = build_method_invocation (yyvsp[-3].node, yyvsp[-1].node); ;
+    break;}
+case 387:
+#line 1586 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ 
+                 tree invok = build_method_invocation (yyvsp[-2].node, NULL_TREE);
+                 yyval.node = make_qualified_primary (yyvsp[-4].node, invok, yyvsp[-3].operator.location);
+               ;
+    break;}
+case 388:
+#line 1591 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ 
+                 tree invok = build_method_invocation (yyvsp[-3].node, yyvsp[-1].node);
+                 yyval.node = make_qualified_primary (yyvsp[-5].node, invok, yyvsp[-4].operator.location);
+               ;
+    break;}
+case 389:
+#line 1596 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 tree invok;
+                 tree wfl = build_wfl_node (super_identifier_node, input_filename, 0, 0);
+                 EXPR_WFL_LINECOL (wfl) = yyvsp[-4].operator.location;
+                 invok = build_method_invocation (yyvsp[-2].node, NULL_TREE);
+                 yyval.node = make_qualified_primary (wfl, invok, yyvsp[-3].operator.location);
+               ;
+    break;}
+case 390:
+#line 1604 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 tree invok;
+                 tree wfl = build_wfl_node (super_identifier_node, input_filename, 0, 0);
+                 EXPR_WFL_LINECOL (wfl) = yyvsp[-5].operator.location;
+                 invok = build_method_invocation (yyvsp[-3].node, yyvsp[-1].node);
+                 yyval.node = make_qualified_primary (wfl, invok, yyvsp[-4].operator.location);
+               ;
+    break;}
+case 391:
+#line 1616 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyerror ("'(' expected"); DRECOVER (method_invocation); ;
+    break;}
+case 392:
+#line 1618 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyerror ("'(' expected"); DRECOVER (method_invocation); ;
+    break;}
+case 393:
+#line 1623 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = build_array_ref (yyvsp[-2].operator.location, yyvsp[-3].node, yyvsp[-1].node); ;
+    break;}
+case 394:
+#line 1625 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = build_array_ref (yyvsp[-2].operator.location, yyvsp[-3].node, yyvsp[-1].node); ;
+    break;}
+case 395:
+#line 1627 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 yyerror ("Missing term and ']' expected");
+                 DRECOVER(array_access);
+               ;
+    break;}
+case 396:
+#line 1632 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 yyerror ("']' expected");
+                 DRECOVER(array_access);
+               ;
+    break;}
+case 397:
+#line 1637 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 yyerror ("Missing term and ']' expected");
+                 DRECOVER(array_access);
+               ;
+    break;}
+case 398:
+#line 1642 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 yyerror ("']' expected");
+                 DRECOVER(array_access);
+               ;
+    break;}
+case 403:
+#line 1657 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = build_incdec (yyvsp[0].operator.token, yyvsp[0].operator.location, yyvsp[-1].node, 1); ;
+    break;}
+case 404:
+#line 1662 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = build_incdec (yyvsp[0].operator.token, yyvsp[0].operator.location, yyvsp[-1].node, 1); ;
+    break;}
+case 407:
+#line 1669 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyval.node = build_unaryop (yyvsp[-1].operator.token, yyvsp[-1].operator.location, yyvsp[0].node); ;
+    break;}
+case 408:
+#line 1671 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyval.node = build_unaryop (yyvsp[-1].operator.token, yyvsp[-1].operator.location, yyvsp[0].node); ;
+    break;}
+case 410:
+#line 1674 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Missing term"); RECOVER;
+    break;}
+case 411:
+#line 1676 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Missing term"); RECOVER;
+    break;}
+case 412:
+#line 1681 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyval.node = build_incdec (yyvsp[-1].operator.token, yyvsp[-1].operator.location, yyvsp[0].node, 0); ;
+    break;}
+case 413:
+#line 1683 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Missing term"); RECOVER;
+    break;}
+case 414:
+#line 1688 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyval.node = build_incdec (yyvsp[-1].operator.token, yyvsp[-1].operator.location, yyvsp[0].node, 0); ;
+    break;}
+case 415:
+#line 1690 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Missing term"); RECOVER;
+    break;}
+case 417:
+#line 1696 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyval.node = build_unaryop (yyvsp[-1].operator.token, yyvsp[-1].operator.location, yyvsp[0].node); ;
+    break;}
+case 418:
+#line 1698 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyval.node = build_unaryop (yyvsp[-1].operator.token, yyvsp[-1].operator.location, yyvsp[0].node); ;
+    break;}
+case 420:
+#line 1701 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Missing term"); RECOVER;
+    break;}
+case 421:
+#line 1703 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Missing term"); RECOVER;
+    break;}
+case 422:
+#line 1708 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ 
+                 tree type = yyvsp[-3].node;
+                 while (ctxp->osb_number--)
+                   type = build_java_array_type (type, -1);
+                 yyval.node = build_cast (yyvsp[-4].operator.location, type, yyvsp[0].node); 
+               ;
+    break;}
+case 423:
+#line 1715 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = build_cast (yyvsp[-3].operator.location, yyvsp[-2].node, yyvsp[0].node); ;
+    break;}
+case 424:
+#line 1717 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = build_cast (yyvsp[-3].operator.location, yyvsp[-2].node, yyvsp[0].node); ;
+    break;}
+case 425:
+#line 1719 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ 
+                 char *ptr;
+                 while (ctxp->osb_number--)
+                   obstack_1grow (&temporary_obstack, '[');
+                 obstack_grow0 (&temporary_obstack, 
+                                IDENTIFIER_POINTER (EXPR_WFL_NODE (yyvsp[-3].node)),
+                                IDENTIFIER_LENGTH (EXPR_WFL_NODE (yyvsp[-3].node)));
+                 ptr = obstack_finish (&temporary_obstack);
+                 EXPR_WFL_NODE (yyvsp[-3].node) = get_identifier (ptr);
+                 yyval.node = build_cast (yyvsp[-4].operator.location, yyvsp[-3].node, yyvsp[0].node);
+               ;
+    break;}
+case 426:
+#line 1731 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("']' expected, invalid type expression");;
+    break;}
+case 427:
+#line 1733 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 if (ctxp->prevent_ese != lineno)
+                   yyerror ("Invalid type expression"); RECOVER;
+                 RECOVER;
+               ;
+    break;}
+case 428:
+#line 1739 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Missing term"); RECOVER;;
+    break;}
+case 429:
+#line 1741 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Missing term"); RECOVER;;
+    break;}
+case 430:
+#line 1743 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Missing term"); RECOVER;;
+    break;}
+case 432:
+#line 1749 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ 
+                 yyval.node = build_binop (BINOP_LOOKUP (yyvsp[-1].operator.token), 
+                                   yyvsp[-1].operator.location, yyvsp[-2].node, yyvsp[0].node);
+               ;
+    break;}
+case 433:
+#line 1754 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 yyval.node = build_binop (BINOP_LOOKUP (yyvsp[-1].operator.token), yyvsp[-1].operator.location,
+                                   yyvsp[-2].node, yyvsp[0].node); 
+               ;
+    break;}
+case 434:
+#line 1759 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 yyval.node = build_binop (BINOP_LOOKUP (yyvsp[-1].operator.token), yyvsp[-1].operator.location,
+                                   yyvsp[-2].node, yyvsp[0].node); 
+               ;
+    break;}
+case 435:
+#line 1764 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Missing term"); RECOVER;;
+    break;}
+case 436:
+#line 1766 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Missing term"); RECOVER;;
+    break;}
+case 437:
+#line 1768 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Missing term"); RECOVER;;
+    break;}
+case 439:
+#line 1774 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 yyval.node = build_binop (BINOP_LOOKUP (yyvsp[-1].operator.token), yyvsp[-1].operator.location,
+                                   yyvsp[-2].node, yyvsp[0].node); 
+               ;
+    break;}
+case 440:
+#line 1779 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 yyval.node = build_binop (BINOP_LOOKUP (yyvsp[-1].operator.token), yyvsp[-1].operator.location,
+                                   yyvsp[-2].node, yyvsp[0].node); 
+               ;
+    break;}
+case 441:
+#line 1784 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Missing term"); RECOVER;;
+    break;}
+case 442:
+#line 1786 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Missing term"); RECOVER;;
+    break;}
+case 444:
+#line 1792 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 yyval.node = build_binop (BINOP_LOOKUP (yyvsp[-1].operator.token), yyvsp[-1].operator.location,
+                                   yyvsp[-2].node, yyvsp[0].node); 
+               ;
+    break;}
+case 445:
+#line 1797 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 yyval.node = build_binop (BINOP_LOOKUP (yyvsp[-1].operator.token), yyvsp[-1].operator.location,
+                                   yyvsp[-2].node, yyvsp[0].node); 
+               ;
+    break;}
+case 446:
+#line 1802 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 yyval.node = build_binop (BINOP_LOOKUP (yyvsp[-1].operator.token), yyvsp[-1].operator.location,
+                                   yyvsp[-2].node, yyvsp[0].node); 
+               ;
+    break;}
+case 447:
+#line 1807 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Missing term"); RECOVER;;
+    break;}
+case 448:
+#line 1809 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Missing term"); RECOVER;;
+    break;}
+case 449:
+#line 1811 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Missing term"); RECOVER;;
+    break;}
+case 451:
+#line 1817 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 yyval.node = build_binop (BINOP_LOOKUP (yyvsp[-1].operator.token), yyvsp[-1].operator.location,
+                                   yyvsp[-2].node, yyvsp[0].node); 
+               ;
+    break;}
+case 452:
+#line 1822 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 yyval.node = build_binop (BINOP_LOOKUP (yyvsp[-1].operator.token), yyvsp[-1].operator.location,
+                                   yyvsp[-2].node, yyvsp[0].node); 
+               ;
+    break;}
+case 453:
+#line 1827 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 yyval.node = build_binop (BINOP_LOOKUP (yyvsp[-1].operator.token), yyvsp[-1].operator.location,
+                                   yyvsp[-2].node, yyvsp[0].node); 
+               ;
+    break;}
+case 454:
+#line 1832 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 yyval.node = build_binop (BINOP_LOOKUP (yyvsp[-1].operator.token), yyvsp[-1].operator.location,
+                                   yyvsp[-2].node, yyvsp[0].node); 
+               ;
+    break;}
+case 456:
+#line 1838 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Missing term"); RECOVER;;
+    break;}
+case 457:
+#line 1840 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Missing term"); RECOVER;;
+    break;}
+case 458:
+#line 1842 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Missing term"); RECOVER;;
+    break;}
+case 459:
+#line 1844 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Missing term"); RECOVER;;
+    break;}
+case 460:
+#line 1846 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Invalid reference type"); RECOVER;;
+    break;}
+case 462:
+#line 1852 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 yyval.node = build_binop (BINOP_LOOKUP (yyvsp[-1].operator.token), yyvsp[-1].operator.location,
+                                   yyvsp[-2].node, yyvsp[0].node); 
+               ;
+    break;}
+case 463:
+#line 1857 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 yyval.node = build_binop (BINOP_LOOKUP (yyvsp[-1].operator.token), yyvsp[-1].operator.location,
+                                   yyvsp[-2].node, yyvsp[0].node); 
+               ;
+    break;}
+case 464:
+#line 1862 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Missing term"); RECOVER;;
+    break;}
+case 465:
+#line 1864 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Missing term"); RECOVER;;
+    break;}
+case 467:
+#line 1870 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 yyval.node = build_binop (BINOP_LOOKUP (yyvsp[-1].operator.token), yyvsp[-1].operator.location,
+                                   yyvsp[-2].node, yyvsp[0].node); 
+               ;
+    break;}
+case 468:
+#line 1875 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Missing term"); RECOVER;;
+    break;}
+case 470:
+#line 1881 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 yyval.node = build_binop (BINOP_LOOKUP (yyvsp[-1].operator.token), yyvsp[-1].operator.location,
+                                   yyvsp[-2].node, yyvsp[0].node); 
+               ;
+    break;}
+case 471:
+#line 1886 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Missing term"); RECOVER;;
+    break;}
+case 473:
+#line 1892 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 yyval.node = build_binop (BINOP_LOOKUP (yyvsp[-1].operator.token), yyvsp[-1].operator.location,
+                                   yyvsp[-2].node, yyvsp[0].node); 
+               ;
+    break;}
+case 474:
+#line 1897 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Missing term"); RECOVER;;
+    break;}
+case 476:
+#line 1903 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 yyval.node = build_binop (BINOP_LOOKUP (yyvsp[-1].operator.token), yyvsp[-1].operator.location,
+                                   yyvsp[-2].node, yyvsp[0].node); 
+               ;
+    break;}
+case 477:
+#line 1908 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Missing term"); RECOVER;;
+    break;}
+case 479:
+#line 1914 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 yyval.node = build_binop (BINOP_LOOKUP (yyvsp[-1].operator.token), yyvsp[-1].operator.location,
+                                   yyvsp[-2].node, yyvsp[0].node); 
+               ;
+    break;}
+case 480:
+#line 1919 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Missing term"); RECOVER;;
+    break;}
+case 483:
+#line 1926 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 YYERROR_NOW;
+                 yyerror ("Missing term");
+                 DRECOVER (1);
+               ;
+    break;}
+case 484:
+#line 1932 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Missing term"); DRECOVER (2);;
+    break;}
+case 485:
+#line 1934 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{yyerror ("Missing term"); DRECOVER (3);;
+    break;}
+case 488:
+#line 1944 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{ yyval.node = build_assignment (yyvsp[-1].operator.token, yyvsp[-1].operator.location, yyvsp[-2].node, yyvsp[0].node); ;
+    break;}
+case 489:
+#line 1946 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+{
+                 if (ctxp->prevent_ese != lineno)
+                   yyerror ("Missing term");
+                 DRECOVER (assign);
+               ;
+    break;}
+}
+   /* the action file gets copied in in place of this dollarsign */
+#line 498 "/usr/cygnus/gnupro-98r1/share/bison.simple"
+\f
+  yyvsp -= yylen;
+  yyssp -= yylen;
+#ifdef YYLSP_NEEDED
+  yylsp -= yylen;
+#endif
+
+#if YYDEBUG != 0
+  if (yydebug)
+    {
+      short *ssp1 = yyss - 1;
+      fprintf (stderr, "state stack now");
+      while (ssp1 != yyssp)
+       fprintf (stderr, " %d", *++ssp1);
+      fprintf (stderr, "\n");
+    }
+#endif
+
+  *++yyvsp = yyval;
+
+#ifdef YYLSP_NEEDED
+  yylsp++;
+  if (yylen == 0)
+    {
+      yylsp->first_line = yylloc.first_line;
+      yylsp->first_column = yylloc.first_column;
+      yylsp->last_line = (yylsp-1)->last_line;
+      yylsp->last_column = (yylsp-1)->last_column;
+      yylsp->text = 0;
+    }
+  else
+    {
+      yylsp->last_line = (yylsp+yylen-1)->last_line;
+      yylsp->last_column = (yylsp+yylen-1)->last_column;
+    }
+#endif
+
+  /* Now "shift" the result of the reduction.
+     Determine what state that goes to,
+     based on the state we popped back to
+     and the rule number reduced by.  */
+
+  yyn = yyr1[yyn];
+
+  yystate = yypgoto[yyn - YYNTBASE] + *yyssp;
+  if (yystate >= 0 && yystate <= YYLAST && yycheck[yystate] == *yyssp)
+    yystate = yytable[yystate];
+  else
+    yystate = yydefgoto[yyn - YYNTBASE];
+
+  goto yynewstate;
+
+yyerrlab:   /* here on detecting error */
+
+  if (! yyerrstatus)
+    /* If not already recovering from an error, report this error.  */
+    {
+      ++yynerrs;
+
+#ifdef YYERROR_VERBOSE
+      yyn = yypact[yystate];
+
+      if (yyn > YYFLAG && yyn < YYLAST)
+       {
+         int size = 0;
+         char *msg;
+         int x, count;
+
+         count = 0;
+         /* Start X at -yyn if nec to avoid negative indexes in yycheck.  */
+         for (x = (yyn < 0 ? -yyn : 0);
+              x < (sizeof(yytname) / sizeof(char *)); x++)
+           if (yycheck[x + yyn] == x)
+             size += strlen(yytname[x]) + 15, count++;
+         msg = (char *) malloc(size + 15);
+         if (msg != 0)
+           {
+             strcpy(msg, "parse error");
+
+             if (count < 5)
+               {
+                 count = 0;
+                 for (x = (yyn < 0 ? -yyn : 0);
+                      x < (sizeof(yytname) / sizeof(char *)); x++)
+                   if (yycheck[x + yyn] == x)
+                     {
+                       strcat(msg, count == 0 ? ", expecting `" : " or `");
+                       strcat(msg, yytname[x]);
+                       strcat(msg, "'");
+                       count++;
+                     }
+               }
+             yyerror(msg);
+             free(msg);
+           }
+         else
+           yyerror ("parse error; also virtual memory exceeded");
+       }
+      else
+#endif /* YYERROR_VERBOSE */
+       yyerror("parse error");
+    }
+
+  goto yyerrlab1;
+yyerrlab1:   /* here on error raised explicitly by an action */
+
+  if (yyerrstatus == 3)
+    {
+      /* if just tried and failed to reuse lookahead token after an error, discard it.  */
+
+      /* return failure if at end of input */
+      if (yychar == YYEOF)
+       YYABORT;
+
+#if YYDEBUG != 0
+      if (yydebug)
+       fprintf(stderr, "Discarding token %d (%s).\n", yychar, yytname[yychar1]);
+#endif
+
+      yychar = YYEMPTY;
+    }
+
+  /* Else will try to reuse lookahead token
+     after shifting the error token.  */
+
+  yyerrstatus = 3;             /* Each real token shifted decrements this */
+
+  goto yyerrhandle;
+
+yyerrdefault:  /* current state does not do anything special for the error token. */
+
+#if 0
+  /* This is wrong; only states that explicitly want error tokens
+     should shift them.  */
+  yyn = yydefact[yystate];  /* If its default is to accept any token, ok.  Otherwise pop it.*/
+  if (yyn) goto yydefault;
+#endif
+
+yyerrpop:   /* pop the current state because it cannot handle the error token */
+
+  if (yyssp == yyss) YYABORT;
+  yyvsp--;
+  yystate = *--yyssp;
+#ifdef YYLSP_NEEDED
+  yylsp--;
+#endif
+
+#if YYDEBUG != 0
+  if (yydebug)
+    {
+      short *ssp1 = yyss - 1;
+      fprintf (stderr, "Error: state stack now");
+      while (ssp1 != yyssp)
+       fprintf (stderr, " %d", *++ssp1);
+      fprintf (stderr, "\n");
+    }
+#endif
+
+yyerrhandle:
+
+  yyn = yypact[yystate];
+  if (yyn == YYFLAG)
+    goto yyerrdefault;
+
+  yyn += YYTERROR;
+  if (yyn < 0 || yyn > YYLAST || yycheck[yyn] != YYTERROR)
+    goto yyerrdefault;
+
+  yyn = yytable[yyn];
+  if (yyn < 0)
+    {
+      if (yyn == YYFLAG)
+       goto yyerrpop;
+      yyn = -yyn;
+      goto yyreduce;
+    }
+  else if (yyn == 0)
+    goto yyerrpop;
+
+  if (yyn == YYFINAL)
+    YYACCEPT;
+
+#if YYDEBUG != 0
+  if (yydebug)
+    fprintf(stderr, "Shifting error token, ");
+#endif
+
+  *++yyvsp = yylval;
+#ifdef YYLSP_NEEDED
+  *++yylsp = yylloc;
+#endif
+
+  yystate = yyn;
+  goto yynewstate;
+}
+#line 1972 "/nfs/hoser/beer/java/egcs/gcc/java/parse.y"
+
+\f
+
+#include "lex.c"
+
+/* Flag for the error report routine to issue the error the first time
+   it's called (overriding the default behavior which is to drop the
+   first invocation and honor the second one, taking advantage of a
+   richer context.  */
+static int force_error = 0;
+
+/* Create a new parser context and make it the current one. */
+
+void
+java_push_parser_context ()
+{
+  struct parser_ctxt *new = 
+    (struct parser_ctxt *)malloc(sizeof (struct parser_ctxt));
+
+  bzero (new, sizeof (struct parser_ctxt));
+  new->next = ctxp;
+  ctxp = new;
+  if (ctxp->next)
+    ctxp->incomplete_class = ctxp->next->incomplete_class;
+}  
+
+void
+java_parser_context_save_global ()
+{
+  ctxp->finput = finput;
+  ctxp->lineno = lineno;
+  ctxp->current_class = current_class;
+  ctxp->filename = input_filename;
+  ctxp->current_function_decl = current_function_decl;
+}
+
+void
+java_parser_context_restore_global ()
+{
+  finput = ctxp->finput;
+  lineno = ctxp->lineno;
+  current_class = ctxp->current_class;
+  input_filename = ctxp->filename;
+  current_function_decl = ctxp->current_function_decl;
+}
+
+void 
+java_pop_parser_context ()
+{
+  tree current;
+  struct parser_ctxt *toFree = ctxp;
+  struct parser_ctxt *next = ctxp->next;
+
+  if (next)
+    {
+      next->incomplete_class = ctxp->incomplete_class;
+      lineno = ctxp->lineno;
+      finput = ctxp->finput;
+      current_class = ctxp->current_class;
+    }
+
+  /* Set the single import class file flag to 0 for the current list
+     of imported things */
+  for (current = ctxp->import_list; current; current = TREE_CHAIN (current))
+    IS_A_SINGLE_IMPORT_CLASSFILE_NAME_P (TREE_PURPOSE (current)) = 0;
+
+  /* And restore those of the previous context */
+  if (ctxp = next)
+    for (current = ctxp->import_list; current; current = TREE_CHAIN (current))
+      IS_A_SINGLE_IMPORT_CLASSFILE_NAME_P (TREE_PURPOSE (current)) = 1;
+
+  free (toFree);
+}
+
+static int do_warning = 0;
+
+void
+yyerror (msg)
+     char *msg;
+{
+  static java_lc elc;
+  static int  prev_lineno;
+  static char *prev_msg;
+
+  int i, save_lineno;
+  char *remainder, *code_from_source;
+  extern struct obstack temporary_obstack;
+  
+  if (!force_error && prev_lineno == lineno)
+    return;
+
+  /* Save current error location but report latter, when the context is
+     richer.  */
+  if (ctxp->java_error_flag == 0)
+    {
+      ctxp->java_error_flag = 1;
+      elc = ctxp->elc;
+      /* Do something to use the previous line if we're reaching the
+        end of the file... */
+#ifdef VERBOSE_SKELETON
+      printf ("* Error detected (%s)\n", (msg ? msg : "(null)"));
+#endif
+      return;
+    }
+
+  /* Ignore duplicate message on the same line. BTW, this is dubious. FIXME */
+  if (!force_error && msg == prev_msg && prev_lineno == elc.line)
+    return;
+
+  ctxp->java_error_flag = 0;
+  if (do_warning)
+    java_warning_count++;
+  else
+    java_error_count++;
+  
+  if (elc.col == 0 && msg[1] == ';')
+    {
+      elc.col  = ctxp->p_line->char_col-1;
+      elc.line = ctxp->p_line->lineno;
+    }
+
+  save_lineno = lineno;
+  prev_lineno = lineno = elc.line;
+  prev_msg = msg;
+
+  code_from_source = java_get_line_col (ctxp->filename, elc.line, elc.col);
+  obstack_grow0 (&temporary_obstack, 
+                code_from_source, strlen (code_from_source));
+  remainder = obstack_finish (&temporary_obstack);
+  if (do_warning)
+    warning ("%s.\n%s", msg, remainder);
+  else
+    error ("%s.\n%s", msg, remainder);
+
+  /* This allow us to cheaply avoid an extra 'Invalid expression
+     statement' error report when errors have been already reported on
+     the same line. This occurs when we report an error but don't have
+     a synchronization point other than ';', which
+     expression_statement is the only one to take care of.  */
+  ctxp->prevent_ese = lineno = save_lineno;
+}
+
+static void
+parse_error (msg)
+     char *msg;
+{
+  java_error (NULL);
+  java_error (msg);
+}
+
+/* Issue an error message at a current source line CL */
+
+static void
+parse_error_context VPROTO ((tree cl, char *msg, ...))
+{
+#ifndef __STDC__
+  tree cl;
+  char *msg;
+#endif
+  char buffer [4096];
+  va_list ap;
+
+  VA_START (ap, msg);
+#ifndef __STDC__
+  cl = va_arg (ap, tree);
+  msg = va_arg (ap, char *);
+#endif
+  vsprintf (buffer, msg, ap);
+
+  force_error = 1;
+  ctxp->elc.line = EXPR_WFL_LINENO (cl);
+  ctxp->elc.col  = (EXPR_WFL_COLNO (cl) == 0xfff ? -1 : EXPR_WFL_COLNO (cl));
+
+  parse_error (buffer);
+  force_error = 0;
+}
+
+/* Issue a warning at a current source line CL */
+
+static void
+parse_warning_context VPROTO ((tree cl, char *msg, ...))
+{
+#ifndef __STDC__
+  tree cl;
+  char *msg;
+#endif
+  char buffer [4096];
+  va_list ap;
+
+  VA_START (ap, msg);
+#ifndef __STDC__
+  cl = va_arg (ap, tree);
+  msg = va_arg (ap, char *);
+#endif
+  vsprintf (buffer, msg, ap);
+
+  force_error = do_warning = 1;
+  ctxp->elc.line = EXPR_WFL_LINENO (cl);
+  ctxp->elc.col  = (EXPR_WFL_COLNO (cl) == 0xfff ? -1 : EXPR_WFL_COLNO (cl));
+
+  parse_error (buffer);
+  do_warning = force_error = 0;
+}
+
+void
+java_report_errors ()
+{
+  if (java_error_count)
+    fprintf (stderr, "%d error%s", 
+            java_error_count, (java_error_count == 1 ? "" : "s"));
+  if (java_warning_count)
+    fprintf (stderr, "%s%d warning%s", (java_error_count ? ", " : ""),
+            java_warning_count, (java_warning_count == 1 ? "" : "s"));
+  if (java_error_count || java_warning_count)
+    putc ('\n', stderr);
+}
+
+static char *
+java_accstring_lookup (flags)
+     int flags;
+{
+  static char buffer [80];
+#define COPY_RETURN(S) {strcpy (buffer, S); return buffer;}
+
+  /* Access modifier looked-up first for easier report on forbidden
+     access. */
+  if (flags & ACC_PUBLIC) COPY_RETURN ("public");
+  if (flags & ACC_PRIVATE) COPY_RETURN ("private");
+  if (flags & ACC_PROTECTED) COPY_RETURN ("protected");
+  if (flags & ACC_STATIC) COPY_RETURN ("static");
+  if (flags & ACC_FINAL) COPY_RETURN ("final");
+  if (flags & ACC_SYNCHRONIZED) COPY_RETURN ("synchronized");
+  if (flags & ACC_VOLATILE) COPY_RETURN ("volatile");
+  if (flags & ACC_TRANSIENT) COPY_RETURN ("transient");
+  if (flags & ACC_NATIVE) COPY_RETURN ("native");
+  if (flags & ACC_INTERFACE) COPY_RETURN ("interface");
+  if (flags & ACC_ABSTRACT) COPY_RETURN ("abstract");
+
+  buffer [0] = '\0';
+  return buffer;
+#undef COPY_RETURN
+}
+
+static void
+redefinition_error (context, id, decl, cl)
+     char *context;
+     tree id, decl, cl;
+{
+  parse_error_context (cl, "%s `%s' already defined in %s:%d", 
+                      context, IDENTIFIER_POINTER (id), 
+                      DECL_SOURCE_FILE (decl), DECL_SOURCE_LINE (decl));
+  /* Here we should point out where its redefined. It's a unicode. FIXME */
+}
+
+/* Build something that the type identifier resolver will identify as
+   being an array to an unresolved type. TYPE_WFL is a WFL on a
+   identifier. */
+
+static tree
+build_unresolved_array_type (type_or_wfl)
+     tree type_or_wfl;
+{
+  char *ptr;
+
+  /* TYPE_OR_WFL might be an array on a primitive type. In this case,
+     just create a array type */
+  if (TREE_CODE (type_or_wfl) == RECORD_TYPE)
+    {
+      tree type = build_java_array_type (type_or_wfl, -1);
+      CLASS_LOADED_P (type) = CLASS_LOADED_P (type_or_wfl);
+      return type;
+    }
+
+  obstack_1grow (&temporary_obstack, '[');
+  obstack_grow0 (&temporary_obstack,
+                IDENTIFIER_POINTER (EXPR_WFL_NODE (type_or_wfl)),
+                IDENTIFIER_LENGTH (EXPR_WFL_NODE (type_or_wfl)));
+  ptr = obstack_finish (&temporary_obstack);
+  return build_expr_wfl (get_identifier (ptr),
+                        EXPR_WFL_FILENAME (type_or_wfl),
+                        EXPR_WFL_LINENO (type_or_wfl),
+                        EXPR_WFL_COLNO (type_or_wfl));
+}
+
+/* Check modifiers. If one doesn't fit, retrieve it in its declaration line
+  and point it out.  */
+
+static void
+check_modifiers (message, value, mask)
+     char *message;
+     int value;
+     int mask;
+{
+  /* Should point out the one that don't fit. ASCII/unicode,
+     going backward. FIXME */
+  if (value & ~mask)
+    {
+      int i, remainder = value & ~mask;
+      for (i = 0; i <= 10; i++)
+        if ((1 << i) & remainder)
+         parse_error_context (ctxp->modifier_ctx [i], message, 
+                              java_accstring_lookup (1 << i));
+    }
+}
+
+static void
+parser_add_interface (class_decl, interface_decl, wfl)
+     tree class_decl, interface_decl, wfl;
+{
+  if (maybe_add_interface (TREE_TYPE (class_decl), TREE_TYPE (interface_decl)))
+    parse_error_context (wfl, "Interface `%s' repeated",
+                        IDENTIFIER_POINTER (DECL_NAME (interface_decl)));
+}
+
+/* Bulk of common class/interface checks. Return 1 if an error was
+   encountered. TAG is 0 for a class, 1 for an interface.  */
+
+static int
+check_class_interface_creation (is_interface, flags, raw_name, qualified_name, decl, cl)
+     int is_interface, flags;
+     tree raw_name, qualified_name, decl, cl;
+{
+  tree node;
+
+  if (!quiet_flag)
+    fprintf (stderr, " %s %s", (is_interface ? "interface" : "class"), 
+            IDENTIFIER_POINTER (qualified_name));
+
+  /* Scope of an interface/class type name:
+       - Can't be imported by a single type import
+       - Can't already exists in the package */
+  if (IS_A_SINGLE_IMPORT_CLASSFILE_NAME_P (raw_name)
+      && (node = find_name_in_single_imports (raw_name)))
+    {
+      parse_error_context 
+       (cl, "%s name `%s' clashes with imported type `%s'",
+        (is_interface ? "Interface" : "Class"),
+        IDENTIFIER_POINTER (raw_name), IDENTIFIER_POINTER (node));
+      return 1;
+    }
+  if (decl && CLASS_COMPLETE_P (decl))
+    {
+      redefinition_error ((is_interface ? "Interface" : "Class"), 
+                         qualified_name, decl, cl);
+      return 1;
+    }
+
+  /* If public, file name should match class/interface name */
+  if (flags & ACC_PUBLIC)
+    {
+      char *f;
+
+      /* Contains OS dependent assumption on path separator. FIXME */
+      for (f = &input_filename [strlen (input_filename)]; 
+          f != input_filename && f[0] != '/'; f--);
+      if (f[0] == '/')
+       f++;
+      if (strncmp (IDENTIFIER_POINTER (raw_name), 
+                  f , IDENTIFIER_LENGTH (raw_name)) ||
+         f [IDENTIFIER_LENGTH (raw_name)] != '.')
+       parse_error_context (cl, "Public %s `%s' must be defined in a file "
+                            "called `%s.java'", 
+                            (is_interface ? "interface" : "class"),
+                            IDENTIFIER_POINTER (qualified_name),
+                            IDENTIFIER_POINTER (raw_name));
+    }
+
+  check_modifiers ((is_interface ? 
+                   "Illegal modifier `%s' for interface declaration" :
+                   "Illegal modifier `%s' for class declaration"), flags,
+                  (is_interface ? INTERFACE_MODIFIERS : CLASS_MODIFIERS));
+  return 0;
+}
+
+/* If DECL is NULL, create and push a new DECL, record the current
+   line CL and do other maintenance things.  */
+
+static tree
+maybe_create_class_interface_decl (decl, qualified_name, cl)
+     tree decl, qualified_name, cl;
+{
+  if (decl)
+    DECL_ARTIFICIAL (decl) = 1; /* FIXME */
+  else
+    decl = push_class (make_class (), qualified_name);
+  
+  /* Take care of the file and line business */
+  DECL_SOURCE_FILE (decl) = EXPR_WFL_FILENAME (cl);
+  DECL_SOURCE_LINE (decl) = EXPR_WFL_LINENO (cl);
+  CLASS_FROM_SOURCE_P (TREE_TYPE (decl)) = 1;
+
+  ctxp->current_parsed_class = decl;
+  
+  /* Link the declaration to the already seen ones */
+  TREE_CHAIN (decl) = ctxp->class_list;
+  ctxp->class_list = decl;
+  /* Install a new dependency list element */
+  create_jdep_list (ctxp);
+
+  SOURCE_FRONTEND_DEBUG (("Defining class/interface %s", 
+                         IDENTIFIER_POINTER (qualified_name)));
+  return decl;
+}
+
+static void
+add_superinterfaces (decl, interface_list)
+     tree decl, interface_list;
+{
+  tree node;
+  /* Superinterface(s): if present and defined, parser_check_super_interface ()
+     takes care of ensuring that:
+       - This is an accessible interface type,
+       - Circularity detection.
+   parser_add_interface is then called. If present but not defined,
+   the check operation is delayed until the super interface gets
+   defined.  */
+  for (node = interface_list; node; node = TREE_CHAIN (node))
+    {
+      tree current = TREE_PURPOSE (node), interface_decl;
+      if ((interface_decl = IDENTIFIER_CLASS_VALUE (EXPR_WFL_NODE (current))))
+       {
+         if (!parser_check_super_interface (interface_decl, decl, current))
+           parser_add_interface (decl, interface_decl, current);
+       }
+      else
+       register_incomplete_type (JDEP_INTERFACE,
+                                 current, decl, NULL_TREE);
+    }
+}
+
+/* Create an interface in pass1 and return its decl. Return the
+   interface's decl in pass 2.  */
+
+static tree
+create_interface (flags, id, super)
+     int flags;
+     tree id, super;
+{
+  int chk;
+  tree raw_name = EXPR_WFL_NODE (id);
+  tree q_name = parser_qualified_classname (id);
+  tree decl = IDENTIFIER_CLASS_VALUE (q_name);
+
+  EXPR_WFL_NODE (id) = q_name; /* Keep source location, even if refined. */
+
+  /* Basic checks: scope, redefinition, modifiers */ 
+  if (check_class_interface_creation (1, flags, raw_name, q_name, decl, id))
+    return NULL_TREE;
+
+  /* Interface modifiers check
+       - public/abstract allowed (already done at that point)
+       - abstract is obsolete (comes first, it's a warning, or should be)
+       - Can't use twice the same (checked in the modifier rule) */
+  if (flags & ACC_ABSTRACT)
+    parse_warning_context 
+      (MODIFIER_WFL (ABSTRACT_TK),
+       "Obsolete use of `abstract' modifier. Interface `%s' is implicitely "
+       "abstract", IDENTIFIER_POINTER (raw_name));
+  if (flags & ACC_PUBLIC && flags & ACC_ABSTRACT)
+    parse_error_context 
+      (MODIFIER_WFL (ABSTRACT_TK),
+       "Can't specify both `public' and `abstract' modifiers in the "
+       "definition of interface `%s'", IDENTIFIER_POINTER (raw_name));
+
+  /* Create a new decl if DECL is NULL, otherwise fix it */
+  decl = maybe_create_class_interface_decl (decl, q_name, id);
+
+  /* Set super info and mark the class a complete */
+  set_super_info (ACC_ABSTRACT | ACC_INTERFACE | flags, TREE_TYPE (decl), 
+                 object_type_node, ctxp->interface_number);
+  ctxp->interface_number = 0;
+  CLASS_COMPLETE_P (decl) = 1;
+  add_superinterfaces (decl, super);
+
+  return decl;
+}
+
+/* Create an class in pass1 and return its decl. Return class
+   interface's decl in pass 2.  */
+
+static tree
+create_class (flags, id, super, interfaces)
+     int flags;
+     tree id, super, interfaces;
+{
+  int chk;
+  tree raw_name = EXPR_WFL_NODE (id);
+  tree class_id, decl;
+  tree super_decl = NULL, super_decl_type;
+
+  class_id = parser_qualified_classname (id);
+  decl = IDENTIFIER_CLASS_VALUE (class_id);
+  EXPR_WFL_NODE (id) = class_id;
+
+  /* Basic check: scope, redefinition, modifiers */
+  if (check_class_interface_creation (0, flags, raw_name, class_id, decl, id))
+    return NULL_TREE;
+
+  /* Class modifier check: 
+       - Allowed modifier (already done at that point)
+       - abstract AND final forbidden 
+       - Public classes defined in the correct file */
+  if ((flags & ACC_ABSTRACT) && (flags & ACC_FINAL))
+    parse_error_context (id, "Class `%s' can't be declared both abstract "
+                        "and final", IDENTIFIER_POINTER (raw_name));
+
+  /* Create a new decl if DECL is NULL, otherwise fix it */
+  decl = maybe_create_class_interface_decl (decl, class_id, id);
+
+  /* If SUPER exists, use it, otherwise use Object */
+  if (super)
+    {
+      /* Can't extend java.lang.Object */
+      if (TREE_TYPE (IDENTIFIER_CLASS_VALUE (class_id)) == object_type_node)
+       {
+         parse_error_context (id, "Can't extend `java.lang.Object'");
+         return NULL_TREE;
+       }
+
+      /* The class is known and exists if there is a decl. Otherwise,
+         postpone the operation and do it later. */
+      super_decl = IDENTIFIER_CLASS_VALUE (EXPR_WFL_NODE (super));
+      if (super_decl)
+       {
+         parser_check_super (super_decl, decl, id);
+         super_decl_type = TREE_TYPE (super_decl);
+       }
+      else
+       super_decl_type = 
+         register_incomplete_type (JDEP_SUPER, super, decl, NULL_TREE);
+    }
+  else if (TREE_TYPE (decl) != object_type_node)
+    super_decl_type = object_type_node;
+  /* We're defining java.lang.Object */
+  else
+    super_decl_type = NULL_TREE;
+
+  /* Set super info and mark the class a complete */
+  set_super_info (flags, TREE_TYPE (decl), super_decl_type, 
+                 ctxp->interface_number);
+  ctxp->interface_number = 0;
+  CLASS_COMPLETE_P (decl) = 1;
+  add_superinterfaces (decl, interfaces);
+
+  return decl;
+}
+
+/* Can't use lookup_field () since we don't want to load the class and
+   can't set the CLASS_LOADED_P flag */
+
+static tree
+find_field (class, name)
+     tree class;
+     tree name;
+{
+  tree decl;
+  for (decl = TYPE_FIELDS (class); decl; decl = TREE_CHAIN (decl))
+    {
+      if (DECL_NAME (decl) == name)
+       return decl;
+    }
+  return NULL_TREE;
+}
+
+/* Wrap around lookup_field that doesn't potentially upset the value
+   of CLASS */
+
+static tree
+lookup_field_wrapper (class, name)
+     tree class, name;
+{
+  tree type = class;
+  return lookup_field (&type, name);
+}
+
+/* Find duplicate field within the same class declarations and report
+   the error */
+
+static int
+duplicate_declaration_error (class, new_field_name, new_type, cl)
+     tree class, new_field_name, new_type, cl;
+{
+  /* This might be modified to work with method decl as well */
+  tree decl = find_field (TREE_TYPE (ctxp->current_parsed_class), 
+                         new_field_name);
+  if (decl)
+    {
+      char *t1 = strdup ((char *)lang_printable_name (new_type, 1));
+      char *t2 = 
+       strdup ((TREE_CODE (TREE_TYPE (decl)) == TREE_LIST ?
+                IDENTIFIER_POINTER (TYPE_NAME 
+                                    (TREE_PURPOSE (TREE_TYPE (decl)))) :
+                (char *)lang_printable_name (TREE_TYPE (decl), 1)));
+      parse_error_context 
+       (cl , "Duplicate variable declaration: `%s %s' was `%s %s' (%s:%d)", 
+        t1, IDENTIFIER_POINTER (new_field_name),
+        t2, IDENTIFIER_POINTER (DECL_NAME (decl)),
+        DECL_SOURCE_FILE (decl), DECL_SOURCE_LINE (decl));
+      free (t1);
+      free (t2);
+      return 0;
+    }
+  return 1;
+}
+
+/* Field registration routine. If TYPE doesn't exist, field
+   declarations are linked to the undefined TYPE dependency list, to
+   be later resolved in java_complete_class () */
+
+static void
+register_fields (flags, type, variable_list)
+     int flags;
+     tree type, variable_list;
+{
+  tree current, type_decl, returned_type;
+  tree class_type = TREE_TYPE (ctxp->current_parsed_class);
+  int saved_lineno = lineno;
+  int must_chain = 0;
+  tree wfl = NULL_TREE;
+
+  /* If we're adding fields to interfaces, those fields are public,
+     static, final */
+  if (CLASS_INTERFACE (TYPE_NAME (class_type)))
+    {
+      OBSOLETE_MODIFIER_WARNING (MODIFIER_WFL (PUBLIC_TK),
+                                flags, ACC_PUBLIC, 
+                                "%s", "interface field(s)");
+      OBSOLETE_MODIFIER_WARNING (MODIFIER_WFL (STATIC_TK),
+                                flags, ACC_STATIC, 
+                                "%s", "interface field(s)");
+      OBSOLETE_MODIFIER_WARNING (MODIFIER_WFL (FINAL_TK),
+                                flags, ACC_FINAL, "%s", "interface field(s)");
+      check_modifiers ("Illegal interface member modifier `%s'", flags,
+                      INTERFACE_FIELD_MODIFIERS);
+      flags |= (ACC_PUBLIC | ACC_STATIC | ACC_FINAL);
+    }
+
+  if (unresolved_type_p (type, &returned_type))
+    {
+      if (returned_type)
+       type = returned_type;
+      else
+       {
+         wfl = type;
+         type = obtain_incomplete_type (type);
+         must_chain = 1;
+       }
+    }
+
+  for (current = variable_list; current; current = TREE_CHAIN (current))
+    {
+      tree cl = TREE_PURPOSE (current);
+      tree init = TREE_VALUE (current);
+      tree current_name = EXPR_WFL_NODE (cl);
+
+      if (duplicate_declaration_error (class_type, current_name, type, cl))
+        {
+         tree field_decl;
+         lineno = EXPR_WFL_LINENO (cl);
+         field_decl = add_field (class_type, current_name, type, flags);
+
+         /* Check if we must chain. */
+         if (must_chain)
+           register_incomplete_type (JDEP_FIELD, wfl, field_decl, type);
+         
+         /* Default value of a static field is 0 and it is considered
+             initialized. */
+         if (flags & ACC_STATIC)
+           INITIALIZED_P (field_decl) = 1;
+
+         /* If we have an initialization value tied to the field */
+         if (init)
+           {
+             /* The field is declared static */
+             if (flags & ACC_STATIC)
+               {
+                 /* FIXME */
+                 if (flags & ACC_FINAL)
+                   ;           
+                 /* Otherwise, the field should be initialized in
+                    <clinit>. This field is remembered so we can
+                    generate <clinit> later. */
+                 else
+                   {
+                     INITIALIZED_P (field_decl) = 1;
+                     TREE_CHAIN (init) = ctxp->static_initialized;
+                     ctxp->static_initialized = init;
+                   }
+               }
+             /* A non-static field declared with an immediate
+                initialization is to be initialized in <init>, if
+                any.  This field is remembered to be processed at the
+                time of the generation of <init>. */
+             else
+               {
+                 TREE_CHAIN (init) = ctxp->non_static_initialized;
+                 ctxp->non_static_initialized = init;
+               }
+           }
+       }
+    }
+  lineno = saved_lineno;
+}
+
+/* Check whether it is necessary to generate a <clinit> for the class
+   we just parsed. */
+
+static void
+maybe_generate_clinit ()
+{
+  int saved_lineno;
+  tree meth, mdecl, c;
+  tree cclass, class_wfl;
+
+  if (!ctxp->static_initialized || java_error_count)
+    return;
+
+  cclass = TREE_TYPE (ctxp->current_parsed_class);
+  class_wfl = build_expr_wfl (DECL_NAME (TYPE_NAME (cclass)),
+                             input_filename, 0, 0);
+  
+  saved_lineno = lineno;
+  lineno = 0;
+  meth = make_node (FUNCTION_TYPE);
+  TREE_TYPE (meth) = void_type_node;
+  TYPE_ARG_TYPES (meth) = NULL_TREE;
+  mdecl = add_method (cclass, ACC_STATIC, clinit_identifier_node,
+                    build_java_signature (meth));
+  lineno = saved_lineno;
+
+  DECL_SOURCE_LINE (mdecl) = 1;
+  DECL_SOURCE_LINE_MERGE (mdecl, 1);
+  source_start_java_method (mdecl);
+  enter_block ();
+
+  /* Keep initialization in order to enforce 8.5 */
+  ctxp->static_initialized = nreverse (ctxp->static_initialized);
+
+  /* We process the list of assignment we produced as the result of
+     the declaration of initialized static field and add them as
+     statement to the <clinit> method. */
+  for (c = ctxp->static_initialized; c; c = TREE_CHAIN (c))
+    {
+      /* We build the assignment expression that will initialize the
+        field to its value. There are strict rules on static
+        initializers (8.5). FIXME */
+      java_method_add_stmt (mdecl, c);
+    }
+
+  BLOCK_EXPR_BODY (DECL_FUNCTION_BODY (mdecl)) = exit_block ();
+  exit_block ();
+  ctxp->static_initialized = NULL_TREE;
+}
+
+/* Shared accros method_declarator and method_header to remember the
+   patch stage that was reached during the declaration of the method.
+   A method DECL is built differently is there is no patch
+   (JDEP_NO_PATCH) or a patch (JDEP_METHOD or JDEP_METHOD_RETURN)
+   pending on the currently defined method.  */
+
+static int patch_stage;
+
+/* Check the method declaration and add the method to its current
+   class.  If the argument list is known to contain incomplete types,
+   the method is partially added and the registration will be resume
+   once the method arguments resolved  */
+
+static tree
+method_header (flags, type, mdecl, throws)
+     int flags;
+     tree type, mdecl, throws;
+{
+  tree meth = TREE_VALUE (mdecl);
+  tree id = TREE_PURPOSE (mdecl);
+  tree this_class = TREE_TYPE (ctxp->current_parsed_class);
+  tree handle_class = CLASS_TO_HANDLE_TYPE (this_class);
+  tree meth_name, returned_type;
+  int saved_lineno;
+  
+  check_modifiers_consistency (flags);
+  
+  /* There are some forbidden modifiers for an abstract method and its
+     class must be abstract as well.  */
+  if (flags & ACC_ABSTRACT)
+    {
+      ABSTRACT_CHECK (flags, ACC_PRIVATE, id, "Private");
+      ABSTRACT_CHECK (flags, ACC_STATIC, id, "Static");
+      ABSTRACT_CHECK (flags, ACC_FINAL, id, "Final");
+      ABSTRACT_CHECK (flags, ACC_NATIVE, id, "Native");
+      ABSTRACT_CHECK (flags, ACC_SYNCHRONIZED,id, "Synchronized");
+      if (!CLASS_ABSTRACT (TYPE_NAME (this_class)))
+       parse_error_context 
+         (id, "Class `%s' must be declared abstract to define abstract "
+          "method `%s'", 
+          IDENTIFIER_POINTER (DECL_NAME (ctxp->current_parsed_class)),
+          IDENTIFIER_POINTER (EXPR_WFL_NODE (id)));
+    }
+
+
+  /* Method declared within the scope of an interface are implicitly
+     abstract and public. Conflicts with other erroneously provided
+     modifiers are check right after. */
+
+  if (CLASS_INTERFACE (TYPE_NAME (this_class)))
+    {
+      /* If FLAGS isn't set because of a modifier, turn the
+        corresponding modifier WFL to NULL so we issue a warning on
+        the obsolete use of the modifier */
+      if (!(flags & ACC_PUBLIC))
+        MODIFIER_WFL (PUBLIC_TK) = NULL;
+      if (!(flags & ACC_ABSTRACT))
+        MODIFIER_WFL (ABSTRACT_TK) = NULL;
+      flags |= ACC_PUBLIC;
+      flags |= ACC_ABSTRACT;
+    }
+
+  /* Modifiers context reset moved up, so abstract method declaration
+     modifiers can be later checked.  */
+
+  meth_name = EXPR_WFL_NODE (id);
+
+  if (unresolved_type_p (type, &returned_type))
+    {
+      if (returned_type)
+       TREE_TYPE (meth) = returned_type;
+      else 
+       {
+         patch_stage = JDEP_METHOD_RETURN;
+         TREE_TYPE (meth) = 
+           register_incomplete_type (patch_stage, type, id, NULL_TREE);
+       }
+    }
+  else
+    TREE_TYPE (meth) = type;
+
+  saved_lineno = lineno;
+  /* When defining an abstract or interface method, the curly
+     bracket at level 1 doesn't exist because there is no function
+     body */
+  lineno = (ctxp->first_ccb_indent1 ? ctxp->first_ccb_indent1 : 
+           EXPR_WFL_LINENO (id));
+
+  if (patch_stage)             /* includes ret type and/or all args */
+    {
+      jdep *jdep;
+      meth = add_method_1 (this_class, flags, meth_name, meth);
+      /* Patch for the return type */
+      if (patch_stage == JDEP_METHOD_RETURN)
+       {
+         jdep = CLASSD_LAST (ctxp->classd_list);
+         JDEP_GET_PATCH (jdep) = &TREE_TYPE (TREE_TYPE (meth));
+       }
+      /* This is the stop JDEP. METH allows the function's signature
+        to be computed. */
+      register_incomplete_type (JDEP_METHOD_END, NULL_TREE, meth, NULL_TREE);
+    }
+  else
+    {
+      tree signature = build_java_signature (meth);
+      tree arg, orig_arg;
+      /* Save original argument list, including argument's names */
+      orig_arg = TYPE_ARG_TYPES (meth);
+      /* Add the method to its class */
+      meth = add_method (this_class, flags, meth_name, signature);
+      /* Fix the method argument list so we have the argument name
+         information */
+      arg = TYPE_ARG_TYPES (TREE_TYPE (meth));
+      if (TREE_CODE (TREE_TYPE (meth)) == METHOD_TYPE)
+       {
+         TREE_PURPOSE (arg) = this_identifier_node;
+         arg = TREE_CHAIN (arg);
+       }
+      while (orig_arg)
+        {
+         TREE_PURPOSE (arg) = TREE_PURPOSE (orig_arg);
+         orig_arg = TREE_CHAIN (orig_arg);
+         arg = TREE_CHAIN (arg);
+       }
+    }
+  DECL_MAX_LOCALS (meth) = ctxp->formal_parameter_number+1;
+  lineno = saved_lineno;
+  /* We set the DECL_NAME to ID so we can track the location where
+     the function was declared. This allow us to report
+     redefinition error accurately. When method are verified,
+     DECL_NAME is reinstalled properly (using the content of the
+     WFL node ID) (see check_method_redefinition). We don't do that
+     when Object is being defined. */
+  if (TREE_TYPE (ctxp->current_parsed_class) != object_type_node)
+    DECL_NAME (meth) = id;
+  return meth;
+}
+
+/* Check modifiers that can be declared but exclusively */
+
+static void
+check_modifiers_consistency (flags)
+     int flags;
+{
+  int acc_count = 0;
+  tree cl = NULL_TREE;
+
+  THIS_MODIFIER_ONLY (flags, ACC_PUBLIC, 0, acc_count, cl);
+  THIS_MODIFIER_ONLY (flags, ACC_PRIVATE, 1, acc_count, cl);
+  THIS_MODIFIER_ONLY (flags, ACC_PROTECTED, 2, acc_count, cl);
+  if (acc_count > 1)
+    parse_error_context
+      (cl, "Inconsistent member declaration. At most one of `public', "
+       "`private', or `protected' may be specified");
+}
+
+/* Check the methode header METH for abstract specifics features */
+
+static void
+check_abstract_method_header (meth)
+     tree meth;
+{
+  int flags = get_access_flags_from_decl (meth);
+  /* DECL_NAME might still be a WFL node */
+  tree name = (TREE_CODE (DECL_NAME (meth)) == EXPR_WITH_FILE_LOCATION ?
+              EXPR_WFL_NODE (DECL_NAME (meth)) : DECL_NAME (meth));
+
+  OBSOLETE_MODIFIER_WARNING (MODIFIER_WFL (ABSTRACT_TK), flags,
+                            ACC_ABSTRACT, "abstract method `%s'",
+                            IDENTIFIER_POINTER (name));
+  OBSOLETE_MODIFIER_WARNING (MODIFIER_WFL (PUBLIC_TK), flags, 
+                            ACC_PUBLIC, "abstract method `%s'",
+                            IDENTIFIER_POINTER (name));
+
+  check_modifiers ("Illegal modifier `%s' for interface method",
+                 flags, INTERFACE_METHOD_MODIFIERS);
+}
+
+/* Create a FUNCTION_TYPE node and start augmenting it with the
+   declared function arguments. Arguments type that can't be resolved
+   are left as they are, but the returned node is marked as containing
+   incomplete types.  */
+
+static tree
+method_declarator (id, list)
+     tree id, list;
+{
+  tree arg_types = NULL_TREE, current, node;
+  tree meth = make_node (FUNCTION_TYPE);
+  jdep *jdep;
+  int incomplete = 0;
+
+  patch_stage = JDEP_NO_PATCH;
+  
+  for (current = list; current; current = TREE_CHAIN (current))
+    {
+      tree wfl_name = TREE_PURPOSE (current);
+      tree type = TREE_VALUE (current);
+      tree name = EXPR_WFL_NODE (wfl_name);
+      tree patchable_type = NULL_TREE, already;
+      tree arg_node, returned_type;
+
+      /* Check redefinition */
+      for (already = arg_types; already; already = TREE_CHAIN (already))
+       if (TREE_PURPOSE (already) == name)
+         {
+           parse_error_context 
+             (wfl_name, "Variable `%s' is used more than once in the "
+              "argument list of method `%s'", IDENTIFIER_POINTER (name),
+              IDENTIFIER_POINTER (EXPR_WFL_NODE (id)));
+           break;
+         }
+
+      /* If we've an incomplete argument type, we know there is a location
+        to patch when the type get resolved, later.  */
+      jdep = NULL;
+      if (unresolved_type_p (type, &returned_type))
+       {
+         if (returned_type)
+           type = returned_type;
+         else
+           {
+             patch_stage = JDEP_METHOD;
+             type = register_incomplete_type (patch_stage, type, 
+                                              wfl_name, NULL_TREE);
+             jdep = CLASSD_LAST (ctxp->classd_list);
+             JDEP_MISC (jdep) = id;
+           }
+       }
+      /* The argument node: a name and a (possibly) incomplete type */
+      arg_node = build_tree_list (name, type);
+      if (jdep)
+       JDEP_GET_PATCH (jdep) = &TREE_VALUE (arg_node);
+      TREE_CHAIN (arg_node) = arg_types;
+      arg_types = arg_node;
+    }
+  TYPE_ARG_TYPES (meth) = nreverse (arg_types);
+  node = build_tree_list (id, meth);
+  return node;
+}
+
+static int
+unresolved_type_p (wfl, returned)
+     tree wfl;
+     tree *returned;
+     
+{
+  if (TREE_CODE (wfl) == EXPR_WITH_FILE_LOCATION)
+    {
+      tree decl = IDENTIFIER_CLASS_VALUE (EXPR_WFL_NODE (wfl));
+      if (returned)
+       *returned = (decl ? TREE_TYPE (decl) : NULL_TREE);
+      return 1;
+    }
+  if (returned)
+    *returned = wfl;
+  return 0;
+}
+
+/* From NAME, build a qualified identifier node using the
+   qualification from the current package definition. */
+
+static tree
+parser_qualified_classname (name)
+     tree name;
+{
+  if (ctxp->package)
+    return merge_qualified_name (ctxp->package, EXPR_WFL_NODE (name));
+  else 
+    return EXPR_WFL_NODE (name);
+}
+
+/* Called once the type a interface extends is resolved. Returns 0 if
+   everything is OK.  */
+
+static int
+parser_check_super_interface (super_decl, this_decl, this_wfl)
+     tree super_decl, this_decl, this_wfl;
+{
+  tree super_type = TREE_TYPE (super_decl);
+
+  /* Has to be an interface */
+  if (!CLASS_INTERFACE (TYPE_NAME (TREE_TYPE (super_decl))))
+    {
+      parse_error_context 
+       (this_wfl, "Can't use %s `%s' to implement/extend %s `%s'",
+        (TYPE_ARRAY_P (super_type) ? "array" : "class"),
+        IDENTIFIER_POINTER (DECL_NAME (super_decl)),
+        (CLASS_INTERFACE (TYPE_NAME (TREE_TYPE (this_decl))) ? 
+         "interface" : "class"),
+        IDENTIFIER_POINTER (DECL_NAME (this_decl)));
+      return 1;
+    }
+
+  /* Check scope: same package OK, other package: OK if public */
+  if (check_pkg_class_access (DECL_NAME (super_decl), lookup_cl (this_decl)))
+    return 1;
+
+  SOURCE_FRONTEND_DEBUG (("Completing interface %s with %s",
+                         IDENTIFIER_POINTER (DECL_NAME (this_decl)),
+                         IDENTIFIER_POINTER (DECL_NAME (super_decl))));
+  return 0;
+}
+
+/* Makes sure that SUPER_DECL is suitable to extend THIS_DECL. Returns
+   0 if everthing is OK.  */
+
+static int
+parser_check_super (super_decl, this_decl, wfl)
+     tree super_decl, this_decl, wfl;
+{
+  tree this_type  = TREE_TYPE (this_decl);
+  tree super_type = TREE_TYPE (super_decl);
+
+  /* SUPER should be a CLASS (neither an array nor an interface) */
+  if (TYPE_ARRAY_P (super_type) || CLASS_INTERFACE (TYPE_NAME (super_type)))
+    {
+      parse_error_context 
+       (wfl, "Class `%s' can't subclass %s `%s'",
+        IDENTIFIER_POINTER (DECL_NAME (this_decl)),
+        (CLASS_INTERFACE (TYPE_NAME (super_type)) ? "interface" : "array"),
+        IDENTIFIER_POINTER (DECL_NAME (super_decl)));
+      return 1;
+    }
+
+  if (CLASS_FINAL (TYPE_NAME (super_type)))
+    {
+      parse_error_context (wfl, "Can't subclass final classes: %s",
+                          IDENTIFIER_POINTER (DECL_NAME (super_decl)));
+      return 1;
+    }
+
+  /* Check scope: same package OK, other package: OK if public */
+  if (check_pkg_class_access (DECL_NAME (super_decl), wfl))
+    return 1;
+  
+  SOURCE_FRONTEND_DEBUG (("Completing class %s with %s",
+                         IDENTIFIER_POINTER (DECL_NAME (this_decl)),
+                         IDENTIFIER_POINTER (DECL_NAME (super_decl))));
+  return 0;
+}
+
+/* Create a new dependency list and link it (in a LIFO manner) to the
+   CTXP list of type dependency list.  */
+
+static void
+create_jdep_list (ctxp)
+     struct parser_ctxt *ctxp;
+{
+  jdeplist *new = malloc (sizeof (jdeplist));  
+  
+  if (!new)
+    fatal ("Can't alloc jdeplist - create_jdep_list");
+    
+  new->first = new->last = NULL;
+  new->next = ctxp->classd_list;
+  ctxp->classd_list = new;
+}
+
+static jdeplist *
+reverse_jdep_list (ctxp)
+     struct parser_ctxt *ctxp;
+{
+  register jdeplist *prev = NULL, *current, *next;
+  for (current = ctxp->classd_list; current; current = next)
+    {
+      next = current->next;
+      current->next = prev;
+      prev = current;
+    }
+  return prev;
+}
+
+/* Create a fake pointer based on the ID stored in the WFL */
+
+static tree
+obtain_incomplete_type (wfl)
+     tree wfl;
+{
+  tree ptr;
+  tree name = EXPR_WFL_NODE (wfl);
+
+  for (ptr = ctxp->incomplete_class; ptr; ptr = TREE_CHAIN (ptr))
+    if (TYPE_NAME (TREE_PURPOSE (ptr)) == name)
+      break;
+
+  if (!ptr)
+    {
+      tree core;
+      push_obstacks (&permanent_obstack, &permanent_obstack);
+      BUILD_PTR_FROM_NAME (core, name);
+      ptr = build_tree_list (core, NULL_TREE);
+      pop_obstacks ();
+      TREE_CHAIN (ptr) = ctxp->incomplete_class;
+      ctxp->incomplete_class = ptr;
+    }
+
+  return ptr;
+}
+
+/* Register a incomplete type whose name is WFL. Reuse PTR if PTR is
+   non NULL instead of computing a new fake type based on WFL. The new
+   dependency is inserted in the current type dependency list, in FIFO
+   manner.  */
+
+static tree
+register_incomplete_type (kind, wfl, decl, ptr)
+     int kind;
+     tree wfl, decl, ptr;
+{
+  jdep *new = malloc (sizeof (jdep));
+
+  if (!new)
+    fatal ("Can't allocate new jdep - register_incomplete_type");
+  if (!ptr && kind != JDEP_METHOD_END) /* JDEP_METHOD_END is a mere marker */
+    ptr = obtain_incomplete_type (wfl);
+
+  JDEP_KIND (new) = kind;
+  JDEP_DECL (new) = decl;
+  JDEP_SOLV (new) = ptr;
+  JDEP_WFL (new) = wfl;
+  JDEP_CHAIN (new) = NULL;
+  JDEP_MISC (new) = NULL_TREE;
+  JDEP_GET_PATCH (new) = (tree *)NULL;
+
+  JDEP_INSERT (ctxp->classd_list, new);
+
+  return ptr;
+}
+
+void
+java_check_circular_reference ()
+{
+  tree current;
+  for (current = ctxp->class_list; current; current = TREE_CHAIN (current))
+    {
+      tree type = TREE_TYPE (current);
+      if (CLASS_INTERFACE (TYPE_NAME (type)))
+       {
+         /* Check all interfaces this class extends */
+         tree basetype_vec = TYPE_BINFO_BASETYPES (type);
+         int n, i;
+
+         if (!basetype_vec)
+           return;
+         n = TREE_VEC_LENGTH (basetype_vec);
+         for (i = 0; i < n; i++)
+           {
+             tree vec_elt = TREE_VEC_ELT (basetype_vec, i);
+             if (vec_elt && BINFO_TYPE (vec_elt) != object_type_node 
+                 && interface_of_p (type, BINFO_TYPE (vec_elt)))
+               parse_error_context (lookup_cl (current),
+                                    "Cyclic interface inheritance");
+           }
+       }
+      else
+       if (inherits_from_p (CLASSTYPE_SUPER (type), type))
+         parse_error_context (lookup_cl (current), 
+                              "Cyclic class inheritance");
+    }
+}
+
+void
+safe_layout_class (class)
+     tree class;
+{
+  tree save_current_class = current_class;
+  char *save_input_filename = input_filename;
+  int save_lineno = lineno;
+  
+  push_obstacks (&permanent_obstack, &permanent_obstack);
+  layout_class (class);
+  pop_obstacks ();
+  
+  current_class = save_current_class;
+  input_filename = save_input_filename;
+  lineno = save_lineno;
+  CLASS_LOADED_P (class) = 1;
+}
+
+static tree
+jdep_resolve_class (dep)
+     jdep *dep;
+{
+  tree decl;
+
+  if (!JDEP_RESOLVED_P (dep))
+    {
+      decl = 
+       resolve_class (JDEP_TO_RESOLVE (dep), JDEP_DECL (dep), JDEP_WFL (dep));
+      JDEP_RESOLVED (dep, decl);
+    }
+  else
+    decl = JDEP_RESOLVED_DECL (dep);
+
+  if (!decl)
+    {
+      complete_class_report_errors (dep);
+      return NULL_TREE;
+    }
+  return decl;
+}
+
+/* Complete unsatisfied class declaration and their dependencies */
+
+void
+java_complete_class ()
+{
+  tree current;
+  tree cclass;
+  jdeplist *cclassd;
+  int error_found;
+
+  push_obstacks (&permanent_obstack, &permanent_obstack);
+
+  /* Process imports and reverse the import on demand list */
+  process_imports ();
+  if (ctxp->import_demand_list)
+    ctxp->import_demand_list = nreverse (ctxp->import_demand_list);
+
+  /* Rever things so we have the right order */
+  ctxp->class_list = nreverse (ctxp->class_list);
+  ctxp->classd_list = reverse_jdep_list (ctxp);
+    
+  for (cclassd = ctxp->classd_list, cclass = ctxp->class_list; 
+       cclass && cclassd; 
+       cclass = TREE_CHAIN (cclass), cclassd = CLASSD_CHAIN (cclassd))
+    {
+      jdep *dep;
+      for (dep = CLASSD_FIRST (cclassd); dep; dep = JDEP_CHAIN (dep))
+       {
+         tree decl;
+
+         if (!(decl = jdep_resolve_class (dep)))
+           continue;
+
+         /* Now it's time to patch */
+         switch (JDEP_KIND (dep))
+           {
+           case JDEP_SUPER:
+             /* Simply patch super */
+             if (parser_check_super (decl, JDEP_DECL (dep), JDEP_WFL (dep)))
+               continue;
+             BINFO_TYPE (TREE_VEC_ELT (BINFO_BASETYPES (TYPE_BINFO 
+               (TREE_TYPE (JDEP_DECL (dep)))), 0)) = TREE_TYPE (decl);
+             break;
+
+           case JDEP_FIELD:
+             {
+               /* We do part of the job done in add_field */
+               tree field_decl = JDEP_DECL (dep);
+               tree field_type = TREE_TYPE (decl);
+               push_obstacks (&permanent_obstack, &permanent_obstack);
+#if ! JAVA_PROMOTE_TO_INT
+               if (TREE_CODE (field_type) == RECORD_TYPE)
+#endif
+                 field_type = promote_type (field_type);
+               pop_obstacks ();
+               TREE_TYPE (field_decl) = field_type;
+               SOURCE_FRONTEND_DEBUG 
+                 (("Completed field/var decl `%s' with `%s'",
+                   IDENTIFIER_POINTER (DECL_NAME (field_decl)),
+                   IDENTIFIER_POINTER (DECL_NAME (decl))));
+               break;
+             }
+           case JDEP_METHOD:   /* We start patching a method */
+           case JDEP_METHOD_RETURN:
+             error_found = 0;
+             while (1)
+               {
+                 if (decl)
+                   {
+                     tree type = promote_type (TREE_TYPE(decl));
+                     JDEP_APPLY_PATCH (dep, type);
+                     SOURCE_FRONTEND_DEBUG 
+                       (((JDEP_KIND (dep) == JDEP_METHOD_RETURN ?
+                          "Completing fct `%s' with ret type `%s'":
+                          "Completing arg `%s' with type `%s'"),
+                         IDENTIFIER_POINTER (EXPR_WFL_NODE 
+                                             (JDEP_DECL_WFL (dep))),
+                         IDENTIFIER_POINTER (DECL_NAME (decl))));
+                   }
+                 else
+                   error_found = 1;
+                 dep = JDEP_CHAIN (dep);
+                 if (JDEP_KIND (dep) == JDEP_METHOD_END)
+                   break;
+                 else
+                   decl = jdep_resolve_class (dep);
+               }
+             if (!error_found)
+               {
+                 tree mdecl = JDEP_DECL (dep), signature;
+                 push_obstacks (&permanent_obstack, &permanent_obstack);
+                 /* Recompute and reset the signature */
+                 signature = build_java_signature (TREE_TYPE (mdecl));
+                 set_java_signature (TREE_TYPE (mdecl), signature);
+                 pop_obstacks ();
+               }
+             else
+               continue;
+             break;
+
+           case JDEP_INTERFACE:
+             if (parser_check_super_interface (decl, JDEP_DECL (dep),
+                                               JDEP_WFL (dep)))
+               continue;
+             parser_add_interface (JDEP_DECL (dep), decl, JDEP_WFL (dep));
+             break;
+
+           case JDEP_VARIABLE:
+             JDEP_APPLY_PATCH (dep, promote_type (TREE_TYPE (decl)));
+             SOURCE_FRONTEND_DEBUG 
+               (("Completing variable `%s' with type `%s'",
+                 (TREE_CODE (JDEP_DECL_WFL (dep)) == EXPR_WITH_FILE_LOCATION ?
+                  IDENTIFIER_POINTER (EXPR_WFL_NODE (JDEP_DECL_WFL (dep))) :
+                  IDENTIFIER_POINTER (DECL_NAME (JDEP_DECL_WFL (dep)))),
+                 IDENTIFIER_POINTER (DECL_NAME (decl))));
+             break;
+
+           case JDEP_TYPE:
+             JDEP_APPLY_PATCH (dep, TREE_TYPE (decl));
+             SOURCE_FRONTEND_DEBUG 
+               (("Completing a random type dependency on a '%s' node",
+                 tree_code_name [TREE_CODE (JDEP_DECL (dep))]));
+             break;
+
+           case JDEP_PARM:
+             JDEP_APPLY_PATCH (dep, promote_type (TREE_TYPE (decl)));
+             SOURCE_FRONTEND_DEBUG 
+               (("Completing parameter `%s' with type `%s'",
+                 IDENTIFIER_POINTER (JDEP_MISC (dep)),
+                 IDENTIFIER_POINTER (DECL_NAME (decl))));
+             break;
+
+           default:
+             fatal ("incomplete switch - java_complete_class");
+           }
+       }
+    }
+  pop_obstacks ();
+  return;
+}
+
+/* Resolve class CLASS_TYPE. Handle the case of trying to resolve an
+   array.  */
+
+static tree
+resolve_class (class_type, decl, cl)
+     tree class_type, decl, cl;
+{
+  char *name = IDENTIFIER_POINTER (TYPE_NAME (class_type));
+  char *base = name;
+  tree resolved_type, resolved_type_decl;
+  
+  /* 1- Check to see if we have an array. If true, find what we really
+     want to resolve  */
+  while (name[0] == '[')
+    name++;
+  if (base != name)
+    TYPE_NAME (class_type) = get_identifier (name);
+
+  /* 2- Resolve the bare type */
+  if (!(resolved_type_decl = do_resolve_class (class_type, decl, cl)))
+    return NULL_TREE;
+  resolved_type = TREE_TYPE (resolved_type_decl);
+
+  /* 3- If we have and array, reconstruct the array down to its nesting */
+  if (base != name)
+    {
+      while (base != name)
+       {
+         if (TREE_CODE (resolved_type) == RECORD_TYPE)
+           resolved_type  = promote_type (resolved_type);
+         resolved_type = build_java_array_type (resolved_type, -1);
+         name--;
+       }
+      /* Build a fake decl for this, since this is what is expected to
+         be returned.  */
+      resolved_type_decl =
+       build_decl (TYPE_DECL, TYPE_NAME (resolved_type), resolved_type);
+      /* Figure how those two things are important for error report. FIXME */
+      DECL_SOURCE_LINE (resolved_type_decl) = 0;
+      DECL_SOURCE_FILE (resolved_type_decl) = input_filename;
+    }
+  return resolved_type_decl;
+}
+
+/* Effectively perform the resolution of class CLASS_TYPE. DECL or CL
+   are used to report error messages.  */
+
+static tree
+do_resolve_class (class_type, decl, cl)
+     tree class_type;
+     tree decl;
+     tree cl;
+{
+  tree new_class_decl;
+  tree original_name = NULL_TREE;
+
+  /* Do not try to replace TYPE_NAME (class_type) by a variable, since
+     its is changed by find_in_imports{_on_demand} */
+
+  /* 1- Check for the type in single imports */
+  if (find_in_imports (class_type))
+    return NULL_TREE;
+
+  /* 2- And check for the type in the current compilation unit. If it fails,
+     try with a name qualified with the package name if appropriate. */
+
+  if ((new_class_decl = IDENTIFIER_CLASS_VALUE (TYPE_NAME (class_type))))
+    {
+      if (!CLASS_LOADED_P (TREE_TYPE (new_class_decl)) &&
+         !CLASS_FROM_SOURCE_P (TREE_TYPE (new_class_decl)))
+       load_class (TYPE_NAME (class_type), 0);
+      return IDENTIFIER_CLASS_VALUE (TYPE_NAME (class_type));
+    }
+
+  original_name = TYPE_NAME (class_type);
+  if (!QUALIFIED_P (TYPE_NAME (class_type)) && ctxp->package)
+    TYPE_NAME (class_type) = merge_qualified_name (ctxp->package, 
+                                                  TYPE_NAME (class_type));
+  if ((new_class_decl = IDENTIFIER_CLASS_VALUE (TYPE_NAME (class_type))))
+    {
+      if (!CLASS_LOADED_P (TREE_TYPE (new_class_decl)) &&
+         !CLASS_FROM_SOURCE_P (TREE_TYPE (new_class_decl)))
+       load_class (TYPE_NAME (class_type), 0);
+      return IDENTIFIER_CLASS_VALUE (TYPE_NAME (class_type));
+    }
+  TYPE_NAME (class_type) = original_name;
+
+  /* 3- Check an other compilation unit that bears the name of type */
+  load_class (TYPE_NAME (class_type), 0);
+  if (check_pkg_class_access (TYPE_NAME (class_type), 
+                             (cl ? cl : lookup_cl (decl))))
+    return NULL_TREE;
+
+  if ((new_class_decl = IDENTIFIER_CLASS_VALUE (TYPE_NAME (class_type))))
+    return new_class_decl;
+
+  /* 4- Check the import on demands. Don't allow bar.baz to be
+     imported from foo.* */
+  if (!QUALIFIED_P (TYPE_NAME (class_type)))
+    if (find_in_imports_on_demand (class_type))
+      return NULL_TREE;
+
+  /* 5- Last call for a resolution */
+  return IDENTIFIER_CLASS_VALUE (TYPE_NAME (class_type));
+}
+
+/* Resolve NAME and lay it out (if not done and if not the current
+   parsed class). Return a decl node.  */
+
+static tree
+resolve_and_layout (name, cl)
+     tree name;
+     tree cl;
+{
+  tree decl = resolve_no_layout (name, cl);
+  if (decl && TREE_TYPE (decl) != current_class 
+      && !CLASS_LOADED_P (TREE_TYPE (decl)))
+    safe_layout_class (TREE_TYPE (decl));
+  return decl;
+}
+
+/* Resolve a class, returns its decl but doesn't perform any
+   layout. The current parsing context is saved and restored */
+
+static tree
+resolve_no_layout (name, cl)
+     tree name, cl;
+{
+  tree ptr, decl;
+  BUILD_PTR_FROM_NAME (ptr, name);
+  java_parser_context_save_global ();
+  decl = resolve_class (ptr, NULL_TREE, cl);
+  java_parser_context_restore_global ();
+  
+  return decl;
+}
+
+/* Called to report errors. Skip leader '[' in a complex array type
+   description that failed to be resolved. */
+
+static char *
+purify_type_name (name)
+     char *name;
+{
+  while (*name && *name == '[')
+    name++;
+  return name;
+}
+
+/* The type CURRENT refers to can't be found. We print error messages.  */
+
+static void
+complete_class_report_errors (dep)
+     jdep *dep;
+{
+  switch (JDEP_KIND (dep))
+    {
+    case JDEP_SUPER:
+      parse_error_context  
+       (JDEP_WFL (dep), "Superclass `%s' of class `%s' not found",
+        IDENTIFIER_POINTER (EXPR_WFL_NODE (JDEP_WFL (dep))),
+        IDENTIFIER_POINTER (DECL_NAME (JDEP_DECL (dep))));
+      break;
+    case JDEP_FIELD:
+      parse_error_context
+       (JDEP_WFL (dep), "Type `%s' not found in declaration of field `%s'",
+        IDENTIFIER_POINTER (EXPR_WFL_NODE (JDEP_WFL (dep))),
+        IDENTIFIER_POINTER (DECL_NAME (JDEP_DECL (dep))));
+      break;
+    case JDEP_METHOD:          /* Covers arguments */
+      parse_error_context
+       (JDEP_WFL (dep), "Type `%s' not found in the declaration of the "
+        "argument `%s' of method `%s'",
+        IDENTIFIER_POINTER (EXPR_WFL_NODE (JDEP_WFL (dep))),
+        IDENTIFIER_POINTER (EXPR_WFL_NODE (JDEP_DECL_WFL (dep))),
+        IDENTIFIER_POINTER (EXPR_WFL_NODE (JDEP_MISC (dep))));
+      break;
+    case JDEP_METHOD_RETURN:   /* Covers return type */
+      parse_error_context
+       (JDEP_WFL (dep), "Type `%s' not found in the declaration of the "
+        "return type of method `%s'", 
+        IDENTIFIER_POINTER (EXPR_WFL_NODE (JDEP_WFL (dep))),
+        IDENTIFIER_POINTER (EXPR_WFL_NODE (JDEP_DECL_WFL (dep))));
+      break;
+    case JDEP_INTERFACE:
+      parse_error_context
+       (JDEP_WFL (dep), "Superinterface `%s' of %s `%s' not found",
+        IDENTIFIER_POINTER (EXPR_WFL_NODE (JDEP_WFL (dep))),
+        (CLASS_OR_INTERFACE (JDEP_DECL (dep), "class", "interface")),
+        IDENTIFIER_POINTER (DECL_NAME (JDEP_DECL (dep))));
+      break;
+    case JDEP_VARIABLE:
+      parse_error_context
+       (JDEP_WFL (dep), "Type `%s' not found in the declaration of the "
+        "local variable `%s'", 
+        purify_type_name (IDENTIFIER_POINTER (EXPR_WFL_NODE (JDEP_WFL (dep)))),
+        IDENTIFIER_POINTER (DECL_NAME (JDEP_DECL (dep))));
+      break;
+    }
+}
+
+/* Check uninitialized final.  */
+
+void
+java_check_final ()
+{
+}
+
+static int
+check_method_redefinition (class, method)
+     tree class, method;
+{
+  tree redef, name;
+  tree cl = DECL_NAME (method);
+  tree sig = TYPE_LANG_SPECIFIC (TREE_TYPE (method))->signature;
+  /* decl name of generated <clinit> doesn't need to be fixed and
+     checked */
+  if (DECL_NAME (method) != clinit_identifier_node)
+    {
+      /* NAME is just the plain name when Object is being defined */
+      if (class != object_type_node)
+       name = DECL_NAME (method) = EXPR_WFL_NODE (DECL_NAME (method));
+      else
+       name = DECL_NAME (method);
+    }
+  else 
+    return 0;
+  
+  for (redef = TYPE_METHODS (class); redef; redef = TREE_CHAIN (redef))
+    {
+      struct lang_type *t = TYPE_LANG_SPECIFIC (TREE_TYPE (redef));
+      
+      if (! t || (redef == method))
+       break;
+      if (DECL_NAME (redef) == name && sig == t->signature)
+       {
+         parse_error_context (cl, "Duplicate method declaration");
+         return 1;
+       }
+    }
+  return 0;
+}
+
+/* Check all the methods of CLASS. Methods are first completed then
+   checked according to regular method existance rules.
+   If no constructor were encountered, then build its declaration. */
+
+static void
+java_check_regular_methods (class_decl)
+     tree class_decl;
+{
+  tree method;
+  tree class = CLASS_TO_HANDLE_TYPE (TREE_TYPE (class_decl));
+  tree super_class = CLASSTYPE_SUPER (class);
+  int seen_constructor = 0;
+
+  TYPE_METHODS (class) = nreverse (TYPE_METHODS (class));
+
+  /* Should take interfaces into account. FIXME */
+  for (method = TYPE_METHODS (class); method; method = TREE_CHAIN (method))
+    {
+      tree found, sig;
+      tree method_wfl = DECL_NAME (method);
+      int aflags;
+
+      if (DECL_CONSTRUCTOR_P (method))
+       seen_constructor = 1;
+
+      /* Check for redefinitions */
+      if (check_method_redefinition (class, method))
+       continue;
+
+      sig = build_java_argument_signature (TREE_TYPE (method));
+
+      found = lookup_argument_method (super_class, DECL_NAME (method), sig);
+      if (! found)
+        continue;
+      /* Can't override a method with the same name and different return
+        types. */
+      if (TREE_TYPE (TREE_TYPE (found)) != TREE_TYPE (TREE_TYPE (method)))
+       parse_warning_context 
+         (method_wfl,
+          "Method `%s' redefined with different return type in class `%s'",
+          lang_printable_name (found),
+          IDENTIFIER_POINTER (DECL_NAME (TYPE_NAME (DECL_CONTEXT (found)))));
+
+      /* Can't override final. Can't override static. */
+      if (METHOD_FINAL (found) || METHOD_STATIC (found))
+       {
+         /* Static *can* override static */
+         if (METHOD_STATIC (found) && METHOD_STATIC (method))
+           continue;
+         parse_error_context 
+           (method_wfl,
+            "%s methods can't be overriden. Method `%s' is %s in class `%s'",
+            (METHOD_FINAL (found) ? "Final" : "Static"),
+            lang_printable_name (found),
+            (METHOD_FINAL (found) ? "final" : "static"),
+            IDENTIFIER_POINTER
+              (DECL_NAME (TYPE_NAME (DECL_CONTEXT (found)))));
+         continue;
+       }
+      /* Static method can't override instance method. */
+      if (METHOD_STATIC (method))
+       {
+         parse_error_context 
+           (method_wfl,
+            "Instance methods can't be overriden by a static method. Method "
+            "`%s' is an instance method in class `%s'",
+            lang_printable_name (found),
+            IDENTIFIER_POINTER
+              (DECL_NAME (TYPE_NAME (DECL_CONTEXT (found)))));
+         continue;
+       }
+      /* Overriding/hiding public must be public or
+        overriding/hiding protected must be protected or public */
+      if ((METHOD_PUBLIC (found) && !METHOD_PUBLIC (method)) ||
+         (METHOD_PROTECTED (found) 
+          && !(METHOD_PUBLIC (method) || METHOD_PROTECTED (method))))
+       {
+         parse_error_context 
+           (method_wfl,
+            "Methods can't be overridden to be more private. Method `%s' is "
+            "%s in class `%s'", lang_printable_name (found),
+            (METHOD_PUBLIC (found) ? "public" : "protected"),
+            IDENTIFIER_POINTER 
+              (DECL_NAME (TYPE_NAME (DECL_CONTEXT (found)))));
+         continue;
+       }
+
+      /* If the method has default access in an other package, then
+      issue a warning that the current method doesn't override the one
+      that was found elsewhere */
+      aflags = get_access_flags_from_decl (found);
+      if ((!aflags || (aflags > ACC_PROTECTED))
+         && !class_in_current_package (DECL_CONTEXT (found)))
+       parse_warning_context 
+         (method_wfl, "Method `%s' in class `%s' does not "
+          "override the corresponding method in class `%s', which is "
+          "private to a different package",
+          lang_printable_name (found),
+          IDENTIFIER_POINTER (DECL_NAME (class_decl)),
+          IDENTIFIER_POINTER (DECL_NAME (TYPE_NAME (DECL_CONTEXT (found)))));
+
+      /* Check on (default) package access. FIXME.  */
+      /* Inheriting multiple methods with the same signature. FIXME */
+    }
+  
+  TYPE_METHODS (class) = nreverse (TYPE_METHODS (class));
+
+  if (!seen_constructor)
+    {
+      /* No constructor seen, we craft one, at line 0 */
+      int saved_lineno = lineno;
+      tree meth, decl;
+      lineno = 0;
+      meth = make_node (FUNCTION_TYPE);
+      TREE_TYPE (meth) = void_type_node;
+      TYPE_ARG_TYPES (meth) = NULL_TREE; 
+      decl = add_method (class, 0, init_identifier_node,
+                        build_java_signature (meth));
+      DECL_CONSTRUCTOR_P (decl) = 1;
+      lineno = saved_lineno;
+    }
+}
+
+/* Check abstract method of interface INTERFACE */
+
+static void
+java_check_abstract_methods (interface)
+     tree interface;
+{
+  int i, n;
+  tree method, basetype_vec, found;
+
+  for (method = TYPE_METHODS (interface); method; method = TREE_CHAIN (method))
+    {
+      char *csig;
+      tree name = DECL_NAME (method);
+
+      /* 2- Check for double definition inside the defining interface */
+      if (check_method_redefinition (interface, method))
+       continue;
+
+      /* 3- Overriding is OK as far as we preserve the return type and
+        the thrown exceptions */
+      found = lookup_java_interface_method2 (interface, method);
+      if (found)
+       {
+         parse_error_context 
+           (lookup_cl (method),
+            "Method `%s' previously defined in interface `%s' is "
+            "redefined with different return type in interface `%s'",
+            lang_printable_name (found),
+            IDENTIFIER_POINTER (DECL_NAME (TYPE_NAME (DECL_CONTEXT (found)))),
+            IDENTIFIER_POINTER (DECL_NAME (TYPE_NAME (interface))));
+         continue;
+       }
+    }
+
+  /* 4- Inherited methods can't differ by their returned types */
+  if (!(basetype_vec = TYPE_BINFO_BASETYPES (interface)))
+    return;
+  n = TREE_VEC_LENGTH (basetype_vec);
+  for (i = 0; i < n; i++)
+    {
+      tree sub_interface_method, sub_interface;
+      tree vec_elt = TREE_VEC_ELT (basetype_vec, i);
+      if (!vec_elt)
+       continue;
+      sub_interface = BINFO_TYPE (vec_elt);
+      for (sub_interface_method = TYPE_METHODS (sub_interface); 
+          sub_interface_method;
+          sub_interface_method = TREE_CHAIN (sub_interface_method))
+       {
+         found = lookup_java_interface_method2 (interface, 
+                                                sub_interface_method);
+         if (found && (found != sub_interface_method))
+           parse_error_context 
+             (lookup_cl (sub_interface_method),
+              "Interface `%s' inherits method `%s' from interface `%s'. This "
+              "method is redefined with a different return "
+              "type in interface `%s'",
+              IDENTIFIER_POINTER (DECL_NAME (TYPE_NAME (interface))),
+              lang_printable_name (found),
+              IDENTIFIER_POINTER 
+                (DECL_NAME (TYPE_NAME (DECL_CONTEXT (sub_interface_method)))),
+              IDENTIFIER_POINTER 
+                (DECL_NAME (TYPE_NAME (DECL_CONTEXT (found)))));
+       }
+    }
+}
+
+/* Check the method on all the defined classes. Should be done to the
+   classes declared in the compilation unit only. FIXME  */
+
+void
+java_check_methods ()
+{
+
+  tree current;
+  for (current = ctxp->class_list; current; current = TREE_CHAIN (current))
+    if (CLASS_FROM_SOURCE_P (TREE_TYPE (current)))
+      {
+       tree class = CLASS_TO_HANDLE_TYPE (TREE_TYPE (current));
+
+       if (CLASS_INTERFACE (TYPE_NAME (class)))
+         java_check_abstract_methods (class);
+       else
+         java_check_regular_methods (current);
+      }
+}
+
+/* Lookup methods in interfaces using their name and partial
+   signature. Return a matching method only if their types differ.  */
+
+static tree
+lookup_java_interface_method2 (class, method_decl)
+     tree class, method_decl;
+{
+  int i, n;
+  tree basetype_vec = TYPE_BINFO_BASETYPES (class), to_return;
+
+  if (!basetype_vec)
+    return NULL_TREE;
+
+  n = TREE_VEC_LENGTH (basetype_vec);
+  for (i = 0; i < n; i++)
+    {
+      tree vec_elt = TREE_VEC_ELT (basetype_vec, i), to_return;
+      if ((BINFO_TYPE (vec_elt) != object_type_node)
+         && (to_return = 
+             lookup_java_method2 (BINFO_TYPE (vec_elt), method_decl, 1)))
+       return to_return;
+    }
+  for (i = 0; i < n; i++)
+    {
+      to_return = lookup_java_interface_method2 
+       (BINFO_TYPE (TREE_VEC_ELT (basetype_vec, i)), method_decl);
+      if (to_return)
+       return to_return;
+    }
+
+  return NULL_TREE;
+}
+
+/* Lookup method using their name and partial signature. Return a
+   matching method only if their types differ.  */
+
+static tree
+lookup_java_method2 (clas, method_decl, do_interface)
+     tree clas, method_decl;
+     int do_interface;
+{
+  tree method, method_signature, method_name, method_type;
+  method_signature = build_java_argument_signature (TREE_TYPE (method_decl));
+  method_name = DECL_NAME (method_decl);
+  method_type = TREE_TYPE (TREE_TYPE (method_decl));
+
+  while (clas != NULL_TREE)
+    {
+      for (method = TYPE_METHODS (clas);
+          method != NULL_TREE;  method = TREE_CHAIN (method))
+       {
+         tree method_sig = build_java_argument_signature (TREE_TYPE (method));
+         if (DECL_NAME (method) == method_name 
+             && method_sig == method_signature 
+             && TREE_TYPE (TREE_TYPE (method)) != method_type)
+           {
+             return method;
+           }
+       }
+      clas = (do_interface ? NULL_TREE : CLASSTYPE_SUPER (clas));
+    }
+  return NULL_TREE;
+}
+
+/* Return the line that matches DECL line number. Used during error
+   report */
+
+static tree
+lookup_cl (decl)
+     tree decl;
+{
+  static tree cl = NULL_TREE;
+  
+  if (!decl)
+    return NULL_TREE;
+
+  if (cl == NULL_TREE)
+    cl = build_expr_wfl (NULL_TREE, NULL, 0, 0);
+
+  EXPR_WFL_FILENAME_NODE (cl) = get_identifier (DECL_SOURCE_FILE (decl));
+  EXPR_WFL_SET_LINECOL (cl, DECL_SOURCE_LINE_FIRST (decl), -1);
+
+  return cl;
+}
+
+/* Look for a simple name in the single-type import list */
+
+static tree
+find_name_in_single_imports (name)
+     tree name;
+{
+  tree node;
+
+  for (node = ctxp->import_list; node; node = TREE_CHAIN (node))
+    if (TREE_VALUE (node) == name)
+      return (EXPR_WFL_NODE (TREE_PURPOSE (node)));
+
+  return NULL_TREE;
+}
+
+/* Process all single-type import. */
+
+static int
+process_imports ()
+{
+  tree import;
+  int error_found;
+
+  for (import = ctxp->import_list; import; import = TREE_CHAIN (import))
+    {
+      tree to_be_found = EXPR_WFL_NODE (TREE_PURPOSE (import));
+
+      /* Don't load twice something already defined. */
+      if (IDENTIFIER_CLASS_VALUE (to_be_found))
+       continue;
+      QUALIFIED_P (to_be_found) = 1;
+      load_class (to_be_found, 0);
+      error_found =
+       check_pkg_class_access (to_be_found, TREE_PURPOSE (import));
+      if (!IDENTIFIER_CLASS_VALUE (to_be_found))
+       {
+         parse_error_context (TREE_PURPOSE (import),
+                              "Class or interface `%s' not found in import",
+                              IDENTIFIER_POINTER (to_be_found));
+         return 1;
+       }
+      if (error_found)
+       return 1;
+    }
+  return 0;
+}
+
+/* Possibly find a class imported by a single-type import statement. Return
+   1 if an error occured, 0 otherwise. */
+
+static int
+find_in_imports (class_type)
+     tree class_type;
+{
+  tree import;
+
+  for (import = ctxp->import_list; import; import = TREE_CHAIN (import))
+    if (TREE_VALUE (import) == TYPE_NAME (class_type))
+      {
+       TYPE_NAME (class_type) = EXPR_WFL_NODE (TREE_PURPOSE (import));
+       QUALIFIED_P (TYPE_NAME (class_type)) = 1;
+       return check_pkg_class_access (TYPE_NAME (class_type), 
+                                      TREE_PURPOSE (import));
+      }
+  return 0;
+}
+
+/* Process a import on demand statement (lazy) */
+
+static int
+read_import_entry (jcf, dirp, returned_name)
+     JCF *jcf;
+     DIR *dirp;
+     char **returned_name;
+{
+  if (dirp)
+    {
+      struct dirent *direntp = readdir (dirp);
+      if (!direntp)
+       {
+         *returned_name = NULL;
+         return 0;
+       }
+      else
+       {
+         *returned_name = direntp->d_name;
+         return (strlen (direntp->d_name));
+       }
+    }
+  else
+    {
+      int current_dir_len = strlen (jcf->classname);
+      char *current_entry;
+      int current_entry_len;
+
+      /* Here we read a zip directory as a file directory. The files
+        we're selecting must have the same root than the directory
+        we're examining. */
+
+      ZipDirectory *zipd = (ZipDirectory *)jcf->zipd; 
+
+      while (zipd)
+       {
+         current_entry = ZIPDIR_FILENAME (zipd);
+         current_entry_len = zipd->filename_length;
+         while (current_entry_len && current_entry [current_entry_len] != '/')
+           current_entry_len--;
+         /* If the path of the current file doesn't match the directory we're
+            scanning, that the end of the search */
+         current_entry_len++;
+         if (strncmp (jcf->classname, current_entry, current_dir_len))
+           {
+             *returned_name = NULL;
+             return 0;
+           }
+         /* Ok, we have at least the same path. The position of the last '/'
+            of the current file we're examining should match the size of
+            name of the directory we're browsing, otherwise that an entry
+            belonging to a sub directory, we want to skip it. */
+         if (current_entry_len != current_dir_len)
+           zipd = ZIPDIR_NEXT (zipd);
+         else
+           {
+             jcf->zipd = ZIPDIR_NEXT (zipd); /* Prepare next read */
+             *returned_name = &current_entry [current_entry_len];
+             return (zipd->filename_length - current_entry_len);
+           }
+       }
+    }
+}
+
+/* Read a import directory, gathering potential match for further type
+   references. Indifferently reads a filesystem or a ZIP archive
+   directory.  */
+
+static void
+read_import_dir (wfl)
+     tree wfl;
+{
+  char *name = IDENTIFIER_POINTER (EXPR_WFL_NODE (wfl));
+  int name_len = IDENTIFIER_LENGTH (EXPR_WFL_NODE (wfl)), reclen;
+  DIR *dirp = NULL;
+  tree dirname = ident_subst (name, name_len, "", '.', '/', "");
+  JCF jcfr, *jcf, *saved_jcf = current_jcf;
+  char *founddirname, *d_name;
+  struct ZipFileCache zip_cache;
+
+  jcf = &jcfr;
+  if (!classpath)
+    fix_classpath ();
+  if (!(founddirname = find_class (name, name_len, jcf, 0)))
+    fatal ("Can't import `%s'", name);
+  if (jcf->outofsynch)
+    jcf_out_of_synch (jcf);
+  if (jcf->seen_in_zip)
+    jcf->zipd = ZIPDIR_NEXT ((ZipDirectory *)jcf->zipd);
+
+  else if (founddirname && (dirp = opendir (founddirname)))
+    {
+      readdir (dirp); readdir (dirp);
+    }
+
+  if (!founddirname && !dirp)
+    {
+      static int first = 1;
+      if (first)
+       {
+         char buffer [256];
+         sprintf (buffer, "Can't find default package `%s'. Check "
+                  "the CLASSPATH environment variable and the access to the "
+                  "archives.", name);
+         error (buffer);
+         java_error_count++;
+         first = 0;
+       }
+      else
+       parse_error_context (wfl, "Package `%s' not found in import", name);
+      current_jcf = saved_jcf;
+      return;
+    }
+
+  /* Here we should have a unified way of retrieving an entry, to be
+     indexed. */
+  while ((reclen = read_import_entry (jcf, dirp, &d_name)))
+    {
+      int java_or_class = 0;
+      int len; 
+      if ((reclen > 5) 
+         && !strcmp (&d_name [reclen-5], ".java"))
+       {
+         java_or_class = 1;
+         len = reclen - 5;
+       }
+         
+      if (!java_or_class && (reclen > 6) &&
+         !strcmp (&d_name [reclen-6], ".class"))
+       {
+         java_or_class = 2;
+         len = reclen - 6;
+       }
+
+      if (java_or_class)
+       {
+         char *id_name;
+         tree node, old;
+
+         obstack_grow (&temporary_obstack, name, name_len);
+         obstack_1grow (&temporary_obstack, '/');
+         obstack_grow0 (&temporary_obstack, d_name, len);
+         id_name = obstack_finish (&temporary_obstack);
+
+         node = get_identifier (id_name);
+         IS_A_CLASSFILE_NAME (node) = 1; /* Or soon to be */
+         QUALIFIED_P (node) = 1; /* As soon as we turn / into . */
+       }
+    }
+  if (dirp)
+    closedir (dirp);
+  
+  current_jcf = saved_jcf;
+}
+
+/* Possibly find a type in the import on demands specified
+   types. Returns 1 if an error occured, 0 otherwise. Run throught the
+   entire list, to detected potential double definitions.  */
+                
+static int
+find_in_imports_on_demand (class_type)
+     tree class_type;
+{
+  tree node, import, node_to_use;
+  int seen_once = -1;
+  tree cl;
+
+  for (import = ctxp->import_demand_list; import; import = TREE_CHAIN (import))
+    {
+      char *id_name;
+      tree found;
+      obstack_grow (&temporary_obstack, 
+                   IDENTIFIER_POINTER (EXPR_WFL_NODE (TREE_PURPOSE (import))),
+                   IDENTIFIER_LENGTH (EXPR_WFL_NODE (TREE_PURPOSE (import))));
+      obstack_1grow (&temporary_obstack, '/');
+      obstack_grow0 (&temporary_obstack, 
+                    IDENTIFIER_POINTER (TYPE_NAME (class_type)),
+                    IDENTIFIER_LENGTH (TYPE_NAME (class_type)));
+      id_name = obstack_finish (&temporary_obstack);
+             
+      node = maybe_get_identifier (id_name);
+      if (node && IS_A_CLASSFILE_NAME (node))
+       {
+         if (seen_once < 0)
+           {
+             cl = TREE_PURPOSE (import);
+             seen_once = 1;
+             node_to_use = node;
+           }
+         else
+           {
+             seen_once++;
+             parse_error_context 
+               (import, "Type `%s' also potentially defined in package `%s'",
+                IDENTIFIER_POINTER (TYPE_NAME (class_type)),
+                IDENTIFIER_POINTER (EXPR_WFL_NODE (TREE_PURPOSE (import))));
+           }
+       }
+    }
+
+  if (seen_once == 1)
+    {
+      /* Setup lineno so that it refers to the line of the import (in
+        case we parse a class file and encounter errors */
+      tree decl;
+      int saved_lineno = lineno;
+      lineno = EXPR_WFL_LINENO (cl);
+      TYPE_NAME (class_type) = ident_subst (IDENTIFIER_POINTER (node_to_use),
+                                           IDENTIFIER_LENGTH (node_to_use),
+                                           "", '/', '.', "");
+      QUALIFIED_P (TYPE_NAME (class_type)) = 1;
+      decl = IDENTIFIER_CLASS_VALUE (TYPE_NAME (class_type));
+      /* If there is no DECL set for the class or if the class isn't
+        loaded and not seen in source yet, the load */
+      if (!decl || (!CLASS_LOADED_P (TREE_TYPE (decl))
+                   && !CLASS_FROM_SOURCE_P (TREE_TYPE (decl))))
+       load_class (node_to_use, 0);
+      lineno = saved_lineno;
+      return check_pkg_class_access (TYPE_NAME (class_type), cl);
+    }
+  else
+    return (seen_once < 0 ? 0 : seen_once); /* It's ok not to have found */
+}
+
+/* Check that CLASS_NAME refers to a PUBLIC class. Return 0 if no
+   access violations were found, 1 otherwise.  */
+
+static int
+check_pkg_class_access (class_name, cl)
+     tree class_name;
+     tree cl;
+{
+  tree type;
+  int access;
+
+  if (!QUALIFIED_P (class_name) || !IDENTIFIER_CLASS_VALUE (class_name))
+    return 0;
+
+  if (!(type = TREE_TYPE (IDENTIFIER_CLASS_VALUE (class_name))))
+    return 0;
+
+  if (!CLASS_PUBLIC (TYPE_NAME (type)))
+    {
+      parse_error_context 
+       (cl, "Can't access %s `%s'. Only public classes and interfaces in "
+        "other packages can be accessed",
+        (CLASS_INTERFACE (TYPE_NAME (type)) ? "interface" : "class"),
+        IDENTIFIER_POINTER (class_name));
+      return 1;
+    }
+  return 0;
+}
+
+/* Local variable declaration. */
+
+static void
+declare_local_variables (modifier, type, vlist)
+     int modifier;
+     tree type;
+     tree vlist;
+{
+  tree decl, current, returned_type, type_wfl, init_stmt = NULL_TREE;
+  int must_chain = 0;
+
+  /* Push a new block if statement were seen between the last time we
+     pushed a block and now. Keep a cound of block to close */
+  if (BLOCK_EXPR_BODY (DECL_FUNCTION_BODY (current_function_decl)))
+    {
+      tree body = DECL_FUNCTION_BODY (current_function_decl);
+      tree b = enter_block ();
+      BLOCK_EXPR_ORIGIN(b) = body;
+    }
+
+  if (modifier)
+    {
+      int i;
+      for (i = 0; i <= 10; i++) if (1 << i & modifier) break;
+      parse_error_context 
+       (ctxp->modifier_ctx [i],
+        (modifier == ACC_FINAL ?
+         "Unsupported JDK1.1 `final' locals" :
+         "Only `final' is allowed as a local variables modifier"));
+      return;
+    }
+
+  if (unresolved_type_p (type, &returned_type))
+    {
+      if (returned_type)
+        type = returned_type;
+      else 
+       {
+         type_wfl = type;
+         type = obtain_incomplete_type (type);
+         must_chain = 1;
+       }
+    }
+  
+  for (current = vlist; current; current = TREE_CHAIN (current))
+    {
+      tree wfl  = TREE_PURPOSE (current);
+      tree name = EXPR_WFL_NODE (wfl);
+      tree init = TREE_VALUE (current);
+      tree other = lookup_name_in_blocks (name);
+
+      /* Don't try to use an INIT statement when an error was found */
+      if (init && java_error_count)
+       init = NULL_TREE;
+
+      if (other)
+       parse_error_context 
+         (wfl, "Variable `%s' is already defined in this method and was "
+          "declared `%s %s' in line %d", 
+          IDENTIFIER_POINTER (name), lang_printable_name (TREE_TYPE (other)),
+          IDENTIFIER_POINTER (name), DECL_SOURCE_LINE (other));
+      else
+       {
+         if (!must_chain && TREE_CODE (type) == RECORD_TYPE)
+           type = promote_type (type);
+         /* Never layout this decl. This will be done when its scope
+             will be entered */
+         decl = build_decl_no_layout (VAR_DECL, name, type);
+         BLOCK_CHAIN_DECL (decl);
+
+         /* Add the initialization function to the current function's code */
+         if (init)
+           {
+             tree wfl;
+             MODIFY_EXPR_FROM_INITIALIZATION_P (init) = 1;
+             java_method_add_stmt 
+               (current_function_decl,
+                build_debugable_stmt (EXPR_WFL_LINECOL (init), init));
+           }
+
+         if (must_chain)
+           {
+             jdep *dep;
+             register_incomplete_type (JDEP_VARIABLE, type_wfl, decl, type);
+             dep = CLASSD_LAST (ctxp->classd_list);
+             JDEP_GET_PATCH (dep) = &TREE_TYPE (decl);
+           }
+       }
+    }
+  SOURCE_FRONTEND_DEBUG (("Defined locals"));
+}
+
+/* Called during parsing. Build decls from argument list.  */
+
+static void
+source_start_java_method (fndecl)
+     tree fndecl;
+{
+  tree tem;
+  tree parm_decl;
+  int i;
+
+  extern tree current_binding_level;
+  current_function_decl = fndecl;
+
+  /* New scope for the function */
+  enter_block ();
+  for (tem = TYPE_ARG_TYPES (TREE_TYPE (fndecl)), i = 0;
+       tem != NULL_TREE; tem = TREE_CHAIN (tem), i++)
+    {
+      tree type = TREE_VALUE (tem);
+      tree name = TREE_PURPOSE (tem);
+      
+      /* If type is incomplete. Layout can't take place
+        now. Create an incomplete decl and ask for the decl to be
+        patched later */
+      if (INCOMPLETE_TYPE_P (type))
+       {
+         jdep *jdep;
+         parm_decl = build_decl_no_layout (PARM_DECL, name, type);
+         
+         register_incomplete_type (JDEP_PARM, NULL_TREE, NULL_TREE, type);
+         jdep = CLASSD_LAST (ctxp->classd_list);
+         JDEP_MISC (jdep) = name;
+         JDEP_GET_PATCH (jdep) = &TREE_TYPE (parm_decl);
+       }
+      else
+       parm_decl = build_decl (PARM_DECL, name, type);
+
+      BLOCK_CHAIN_DECL (parm_decl);
+    }
+  tem = BLOCK_EXPR_DECLS (DECL_FUNCTION_BODY (current_function_decl));
+  BLOCK_EXPR_DECLS (DECL_FUNCTION_BODY (current_function_decl)) =
+    nreverse (tem);
+  DECL_ARG_SLOT_COUNT (current_function_decl) = i;
+}
+
+/* Called during expansion. Push decls formerly built from argument
+   list so they're usable during expansion. */
+
+static void
+expand_start_java_method (fndecl)
+     tree fndecl;
+{
+  tree tem, *ptr;
+  tree parm_decl;
+
+  extern tree current_binding_level;
+  current_function_decl = fndecl;
+
+  announce_function (fndecl);
+  pushlevel (1);               /* Push parameters */
+  ptr = &DECL_ARGUMENTS (fndecl);
+  tem  = BLOCK_EXPR_DECLS (DECL_FUNCTION_BODY (current_function_decl));
+  while (tem)
+    {
+      tree next = TREE_CHAIN (tem);
+      DECL_ARG_TYPE (tem) = TREE_TYPE (tem);
+      layout_decl (tem, 0);
+      pushdecl (tem);
+      INITIALIZED_P (tem) = 1; /* Parms are initialized */
+      *ptr = tem;
+      ptr = &TREE_CHAIN (tem);
+      tem = next;
+    }
+  *ptr = NULL_TREE;
+  pushdecl_force_head (DECL_ARGUMENTS (fndecl));
+  lineno = DECL_SOURCE_LINE_FIRST (fndecl);
+  complete_start_java_method (fndecl); 
+}
+
+/* Terminate a function and expand its body.  */
+
+static void
+source_end_java_method ()
+{
+  tree fndecl = current_function_decl;
+
+  java_parser_context_save_global ();
+  lineno = ctxp->last_ccb_indent1;
+
+  /* Generate function's code */
+  if (BLOCK_EXPR_BODY (DECL_FUNCTION_BODY (fndecl))
+      && ! flag_emit_class_files)
+    expand_expr_stmt (BLOCK_EXPR_BODY (DECL_FUNCTION_BODY (fndecl)));
+
+  /* pop out of its parameters */
+  pushdecl_force_head (DECL_ARGUMENTS (fndecl));
+  poplevel (1, 0, 1);
+  BLOCK_SUPERCONTEXT (DECL_INITIAL (fndecl)) = fndecl;
+
+  /* Generate rtl for function exit.  */
+  if (! flag_emit_class_files)
+    {
+      lineno = DECL_SOURCE_LINE_LAST (fndecl);
+      expand_function_end (input_filename, lineno, 0);
+
+      /* Run the optimizers and output assembler code for this function. */
+      rest_of_compilation (fndecl);
+    }
+
+  current_function_decl = NULL_TREE;
+  /*  permanent_allocation (1); */
+  java_parser_context_restore_global ();
+}
+
+/* Record EXPR in the current function block. Complements compound
+   expression second operand if necessary.  */
+
+tree
+java_method_add_stmt (fndecl, expr)
+     tree fndecl, expr;
+{
+  tree body = BLOCK_EXPR_BODY (DECL_FUNCTION_BODY (fndecl));
+  tree node;
+
+  if (java_error_count)
+    return body;
+  if ((node = add_stmt_to_compound (body, NULL_TREE, expr)) == body)
+    return body;
+
+  BLOCK_EXPR_BODY (DECL_FUNCTION_BODY (fndecl)) = node;
+  TREE_SIDE_EFFECTS (node) = 1;
+  return node;
+}
+
+/* Add STMT to EXISTING if possible, otherwise create a new
+   COMPOUND_EXPR and add STMT to it. */
+
+static tree
+add_stmt_to_compound (existing, type, stmt)
+     tree existing, type, stmt;
+{
+  tree node;
+
+  if (existing && (TREE_CODE (existing) == COMPOUND_EXPR)
+      && TREE_OPERAND (existing, 1) == size_zero_node)
+    {
+      TREE_OPERAND (existing, 1) = stmt;
+      TREE_TYPE (existing) = type;
+      return existing;
+    }
+  else if (existing)
+    node = build (COMPOUND_EXPR, type, existing, stmt);
+  else
+    node = build (COMPOUND_EXPR, type, stmt, size_zero_node);
+  
+  return node;
+}
+
+/* Hold THIS for the scope of the current public method decl.  */
+static tree current_this;
+
+/* Layout all class found during parsing */
+
+void
+java_layout_classes ()
+{
+  tree current;
+  for (current = ctxp->class_list; current; current = TREE_CHAIN (current))
+    {
+      current_class = TREE_TYPE (current);
+      TYPE_FIELDS (current_class) = nreverse (TYPE_FIELDS (current_class));
+      if (!TYPE_SIZE (current_class))
+       safe_layout_class (current_class);
+    }
+}
+
+/* Expand all methods in all registered classes.  */
+
+void
+java_complete_expand_methods ()
+{
+  tree current;
+  
+  for (current = ctxp->class_list; current; current = TREE_CHAIN (current))
+    {
+      extern tree current_constant_pool_data_ref;
+      tree class_type = CLASS_TO_HANDLE_TYPE (TREE_TYPE (current));
+      tree decl;
+      int saved_lineno;
+
+      current_class = TREE_TYPE (current);
+
+      /* Initialize a new constant pool */
+      init_outgoing_cpool ();
+
+      /* Don't process function bodies in interfaces */
+      if (!CLASS_INTERFACE (TYPE_NAME (current_class)))
+       for (decl = TYPE_METHODS (class_type); decl; decl = TREE_CHAIN (decl))
+         {
+           current_function_decl = decl;
+           /* Don't generate debug info on line zero when expanding a
+              generated constructor. */
+           if (DECL_CONSTRUCTOR_P (decl) && !DECL_FUNCTION_BODY (decl))
+             {
+               /* If we found errors, it's too dangerous to try to generate
+                  and expand a constructor */
+               if (!java_error_count)
+                 {
+                   restore_line_number_status (1);
+                   java_complete_expand_method (decl);
+                   restore_line_number_status (0);
+                 }
+             }
+           else
+             java_complete_expand_method (decl);
+         }
+
+      /* Make the class data, register it and run the rest of decl
+         compilation on it */
+      if (!java_error_count && ! flag_emit_class_files)
+       {
+         make_class_data (current_class);
+         register_class ();
+         rest_of_decl_compilation (TYPE_NAME (current_class), (char*) 0, 1, 0);
+       }
+    }
+}
+
+/* Complete and expand a method.  */
+
+static void
+java_complete_expand_method (mdecl)
+     tree mdecl;
+{
+  tree node;
+  jdep *current;
+  int no_ac_found = 1;
+
+  /* We generate some code for an empty constructor */
+  if (DECL_CONSTRUCTOR_P (mdecl) && !DECL_FUNCTION_BODY (mdecl))
+    {
+      tree arg_list, func, call;
+      tree method_type = TREE_TYPE (mdecl);
+      tree class_type = CLASS_TO_HANDLE_TYPE (current_class);
+      tree self_type = (CLASSTYPE_SUPER (class_type) ? 
+                       CLASSTYPE_SUPER (class_type) : class_type);
+      tree method_signature = 
+       TYPE_LANG_SPECIFIC (method_type)->signature;
+      tree method = 
+       lookup_java_constructor (CLASS_TO_HANDLE_TYPE (self_type),
+                                method_signature);
+      tree block, compound;
+
+      /* Fixe the begining/ending lines of the method so that with
+         no_line_numbers set to 1 it doesn't generate debug info at
+         line 1 for this artificial constructor. */
+      DECL_SOURCE_LINE (mdecl) = 1;
+      DECL_SOURCE_LINE_MERGE (mdecl, 1);
+      source_start_java_method (mdecl);
+      arg_list = BLOCK_EXPR_DECLS (DECL_FUNCTION_BODY (mdecl));
+      enter_block ();
+      func = build_known_method_ref (method, method_type, self_type,
+                                    method_signature, arg_list);
+      
+      if (! flag_emit_class_files)
+       func = build1 (NOP_EXPR, build_pointer_type (method_type), func);
+      call = build (CALL_EXPR, TREE_TYPE (method_type), func, 
+                   build_tree_list (NULL_TREE, arg_list), NULL_TREE);
+      TREE_SIDE_EFFECTS (call) = 1;
+      call = build_class_init (self_type, call);
+      compound = java_method_add_stmt (mdecl, call);
+      block = exit_block ();
+      BLOCK_EXPR_BODY (DECL_FUNCTION_BODY (mdecl)) = block;
+      /* The function decl, its block and the compound statement
+         within this block are all of void type. */
+      TREE_TYPE (block) = TREE_TYPE (compound) = 
+       TREE_TYPE (DECL_FUNCTION_BODY (mdecl)) = void_type_node;
+      exit_block ();
+      no_ac_found = 0;
+    }
+  
+  if (DECL_FUNCTION_BODY (mdecl))
+    {
+      expand_start_java_method (mdecl);
+
+      current_this 
+       = (!METHOD_STATIC (mdecl) ? 
+          BLOCK_EXPR_DECLS (DECL_FUNCTION_BODY (mdecl)) : NULL_TREE);
+
+      if (BLOCK_EXPR_BODY (DECL_FUNCTION_BODY (mdecl)) && no_ac_found)
+       java_complete_tree (BLOCK_EXPR_BODY (DECL_FUNCTION_BODY (mdecl)));
+      /* Don't go any further if we've found error(s) during the
+         expansion */
+      if (!java_error_count)
+       source_end_java_method ();
+    }
+}
+
+/* Expand finals.  */
+
+void
+java_expand_finals ()
+{
+}
+
+/* Wrap non WFL PRIMARY around a WFL and set EXPR_WFL_QUALIFICATION to
+   a tree list node containing RIGHT. Fore coming RIGHTs will be
+   chained to this hook. LOCATION contains the location of the
+   separating `.' operator.  */
+
+static tree
+make_qualified_primary (primary, right, location)
+     tree primary, right;
+     int location;
+{
+  tree wfl;
+
+  /* We want to process THIS . xxx symbolicaly, to keep it consistent
+     with the way we're processing SUPER. A THIS from a primary as a
+     different form than a SUPER. Turn THIS into something symbolic */
+  if (TREE_CODE (primary) == JAVA_THIS_EXPR)
+    {
+      wfl = build_wfl_node (this_identifier_node, input_filename, 0, 0);
+      EXPR_WFL_LINECOL (wfl) = EXPR_WFL_LINECOL (primary);
+      wfl = make_qualified_name (wfl, right, location);
+      PRIMARY_P (wfl) = 1;
+      return wfl;
+    }
+  /* Other non WFL node are wrapped around a WFL */
+  else if (TREE_CODE (primary) != EXPR_WITH_FILE_LOCATION)
+    {
+      wfl = build_expr_wfl (NULL_TREE, ctxp->filename, 0, 0);
+      EXPR_WFL_LINECOL (wfl) = EXPR_WFL_LINECOL (primary);
+      EXPR_WFL_QUALIFICATION (wfl) = build_tree_list (primary, NULL_TREE);
+    }
+  else
+    {
+      wfl = primary;
+      if (!EXPR_WFL_QUALIFICATION (primary))
+       EXPR_WFL_QUALIFICATION (primary) = 
+         build_tree_list (primary, NULL_TREE);
+    }
+
+  EXPR_WFL_LINECOL (right) = location;
+  chainon (EXPR_WFL_QUALIFICATION (wfl), build_tree_list (right, NULL_TREE));
+  PRIMARY_P (wfl) =  1;
+  return wfl;
+}
+
+/* Simple merge of two name separated by a `.' */
+
+static tree
+merge_qualified_name (left, right)
+     tree left, right;
+{
+  tree node;
+  obstack_grow (&temporary_obstack, IDENTIFIER_POINTER (left),
+               IDENTIFIER_LENGTH (left));
+  obstack_1grow (&temporary_obstack, '.');
+  obstack_grow0 (&temporary_obstack, IDENTIFIER_POINTER (right),
+                IDENTIFIER_LENGTH (right));
+  node =  get_identifier (obstack_base (&temporary_obstack));
+  obstack_free (&temporary_obstack, obstack_base (&temporary_obstack));
+  QUALIFIED_P (node) = 1;
+  return node;
+}
+
+/* Merge the two parts of a qualified name into LEFT.  Set the
+   location information of the resulting node to LOCATION, usually
+   inherited from the location information of the `.' operator. */
+
+static tree
+make_qualified_name (left, right, location)
+     tree left, right;
+     int location;
+{
+  int qualified;
+  tree left_id = EXPR_WFL_NODE (left);
+  tree right_id = EXPR_WFL_NODE (right);
+  tree wfl, merge;
+
+  merge = merge_qualified_name (left_id, right_id);
+
+  /* Left wasn't qualified and is now qualified */
+  if (!QUALIFIED_P (left_id))
+    {
+      tree wfl = build_expr_wfl (left_id, ctxp->filename, 0, 0);
+      EXPR_WFL_LINECOL (wfl) = EXPR_WFL_LINECOL (left);
+      EXPR_WFL_QUALIFICATION (left) = build_tree_list (wfl, NULL_TREE);
+    }
+  
+  wfl = build_expr_wfl (right_id, ctxp->filename, 0, 0);
+  EXPR_WFL_LINECOL (wfl) = location;
+  chainon (EXPR_WFL_QUALIFICATION (left), build_tree_list (wfl, NULL_TREE));
+
+  EXPR_WFL_NODE (left) = merge;
+  return left;
+}
+
+/* Extract the last identifier component of the qualified in WFL. The
+   last identifier is removed from the linked list */
+
+static tree
+cut_identifier_in_qualified (wfl)
+     tree wfl;
+{
+  tree q;
+  tree previous = NULL_TREE;
+  for (q = EXPR_WFL_QUALIFICATION (wfl); ; previous = q, q = TREE_CHAIN (q))
+    if (!TREE_CHAIN (q))
+      {
+       if (!previous)
+         fatal ("Operating on a non qualified qualified WFL - "
+                "cut_identifier_in_qualified");
+       TREE_CHAIN (previous) = NULL_TREE;
+       return TREE_PURPOSE (q);
+      }
+}
+
+/* Resolve the expression name NAME. Return its decl.  */
+
+static tree
+resolve_expression_name (id)
+     tree id;
+{
+  tree name = EXPR_WFL_NODE (id);
+  tree decl;
+
+  /* 6.5.5.1: Simple expression names */
+  if (!PRIMARY_P (id) && !QUALIFIED_P (name))
+    {
+      /* 15.13.1: NAME can appear within the scope of a local variable
+         declaration */
+      if ((decl = IDENTIFIER_LOCAL_VALUE (name)))
+        return decl;
+
+      /* 15.13.1: NAME can appear within a class declaration */
+      else 
+        {
+         decl = lookup_field_wrapper (current_class, name);
+         if (decl)
+           {
+             int fs = FIELD_STATIC (decl);
+             /* Instance variable (8.3.1.1) can't appear within
+                static method, static initializer or initializer for
+                a static variable. */
+             if (!fs && METHOD_STATIC (current_function_decl))
+               {
+                 parse_error_context 
+                   (id, "Can't make a static reference to nonstatic variable "
+                    "`%s' in class `%s'",
+                    IDENTIFIER_POINTER (name),
+                    IDENTIFIER_POINTER (DECL_NAME 
+                                        (TYPE_NAME (current_class))));
+                 return error_mark_node;
+               }
+             decl = build_field_ref ((fs ? NULL_TREE : current_this),
+                                     current_class, name);
+             return (fs ? build_class_init (current_class, decl) : decl);
+           }
+         /* Fall down to error report on undefined variable */
+       }
+    }
+  /* 6.5.5.2 Qualified Expression Names */
+  else
+    {
+      qualify_ambiguous_name (id);
+      /* 15.10.1 Field Access Using a Primary and/or Expression Name */
+      /* 15.10.2: Accessing Superclass Members using super */
+      return resolve_field_access (id, NULL, NULL);
+    }
+
+  /* We've got an error here */
+  parse_error_context (id, "Undefined variable `%s'", 
+                      IDENTIFIER_POINTER (name));
+
+  return error_mark_node;
+}
+
+/* 15.10.1 Field Acess Using a Primary and/or Expression Name.
+   We return something suitable to generate the field access. We also
+   return the field decl in FIELD_DECL and its type in FIELD_TYPE.  If
+   recipient's address can be null. */
+
+static tree
+resolve_field_access (qual_wfl, field_decl, field_type)
+     tree qual_wfl;
+     tree *field_decl, *field_type;
+{
+  int is_static = 0;
+  tree field_ref;
+  tree decl, where_found, type_found;
+
+  if (resolve_qualified_expression_name (qual_wfl, &decl,
+                                        &where_found, &type_found))
+    return error_mark_node;
+
+  /* Resolve the LENGTH field of an array here */
+  if (DECL_NAME (decl) == length_identifier_node && TYPE_ARRAY_P (type_found)
+      && ! flag_emit_class_files)
+    {
+      tree length = build_java_array_length_access (where_found);
+      field_ref =
+       build_java_arraynull_check (type_found, length, int_type_node);
+    }
+  /* We might have been trying to resolve field.method(). In which
+     case, the resolution is over and decl is the answer */
+  else if (DECL_P (decl) && IDENTIFIER_LOCAL_VALUE (DECL_NAME (decl)) == decl)
+    field_ref = decl;
+  else if (DECL_P (decl))
+    {
+      is_static = DECL_P (decl) && FIELD_STATIC (decl);
+      field_ref = build_field_ref ((is_static ? NULL_TREE : where_found), 
+                                  type_found, DECL_NAME (decl));
+      if (field_ref == error_mark_node)
+       return error_mark_node;
+      if (is_static)
+       {
+         field_ref = build_class_init (type_found, field_ref);
+         /* If the static field was identified by an expression that
+            needs to be generated, make the field access a compound
+            expression whose first part of the evaluation of the
+            field selector part. */
+         if (where_found && TREE_CODE (where_found) != TYPE_DECL)
+           {
+             tree type = QUAL_DECL_TYPE (field_ref);
+             field_ref = build (COMPOUND_EXPR, type, where_found, field_ref);
+           }
+       }
+    }
+  else
+    field_ref = decl;
+
+  if (field_decl)
+    *field_decl = decl;
+  if (field_type)
+    *field_type = QUAL_DECL_TYPE (decl);
+  return field_ref;
+}
+
+/* 6.5.5.2: Qualified Expression Names */
+
+static int
+resolve_qualified_expression_name (wfl, found_decl, where_found, type_found)
+     tree wfl;
+     tree *found_decl, *type_found, *where_found;
+{
+  int from_type = 0;           /* Field search initiated from a type */
+  int from_super = 0, from_cast = 0;
+  int previous_call_static = 0;
+  int is_static;
+  tree decl = NULL_TREE, type = NULL_TREE, q;
+  *where_found = NULL_TREE;
+
+  for (q = EXPR_WFL_QUALIFICATION (wfl); q; q = TREE_CHAIN (q))
+    {
+      tree qual_wfl = QUAL_WFL (q);
+
+      /* 15.10.1 Field Access Using a Primary */
+      
+      switch (TREE_CODE (qual_wfl))
+       {
+       case CALL_EXPR:
+       case JAVA_NEW_CLASS_EXPR:
+         /* If the access to the function call is a non static field,
+            build the code to access it. */
+         if (DECL_P (decl) && !FIELD_STATIC (decl))
+           {
+             decl = maybe_access_field (decl, *where_found, type);
+             if (decl == error_mark_node)
+               return 1;
+           }
+         /* And code for the function call */
+         if (complete_function_arguments (qual_wfl))
+           return 1;
+         *where_found = 
+           patch_method_invocation_stmt (qual_wfl, decl, type, &is_static);
+         if (*where_found == error_mark_node)
+           return 1;
+         *type_found = type = QUAL_DECL_TYPE (*where_found);
+
+         /* If the previous call was static and this one is too,
+            build a compound expression to hold the two (because in
+            that case, previous function calls aren't transported as
+            forcoming function's argument. */
+         if (previous_call_static && is_static)
+           {
+             decl = build (COMPOUND_EXPR, type, decl, *where_found);
+             TREE_SIDE_EFFECTS (decl) = 1;
+           }
+         else
+           {
+             previous_call_static = is_static;
+             decl = *where_found;
+           }
+         continue;
+
+       case CONVERT_EXPR:
+         *where_found = decl = java_complete_tree (qual_wfl);
+         if (decl == error_mark_node)
+           return 1;
+         *type_found = type = QUAL_DECL_TYPE (decl);
+         from_cast = 1;
+         continue;
+
+       case ARRAY_REF:
+         /* If the access to the function call is a non static field,
+            build the code to access it. */
+         if (DECL_P (decl) && !FIELD_STATIC (decl))
+           {
+             decl = maybe_access_field (decl, *where_found, type);
+             if (decl == error_mark_node)
+               return 1;
+           }
+         /* And code for the array reference expression */
+         decl = java_complete_tree (qual_wfl);
+         if (decl == error_mark_node)
+           return 1;
+         type = QUAL_DECL_TYPE (decl);
+         continue;
+       }
+
+      /* If we fall here, we weren't processing a (static) function call. */
+      previous_call_static = 0;
+
+      /* It can be the keyword THIS */
+      if (EXPR_WFL_NODE (qual_wfl) == this_identifier_node)
+       {
+         if (!current_this)
+           {
+             parse_error_context 
+               (wfl, "Keyword `this' used outside allowed context");
+             return 1;
+           }
+         /* We have to generate code for intermediate acess */
+         *where_found = decl = current_this;
+         type = QUAL_DECL_TYPE (decl);
+         continue;
+       }
+
+      /* 15.10.2 Accessing Superclass Members using SUPER */
+      if (EXPR_WFL_NODE (qual_wfl) == super_identifier_node)
+       {
+         tree node;
+         /* Check on the restricted use of SUPER */
+         if (METHOD_STATIC (current_function_decl)
+             || current_class == object_type_node)
+           {
+             parse_error_context 
+               (wfl, "Keyword `super' used outside allowed context");
+             return 1;
+           }
+         /* Otherwise, treat SUPER as (SUPER_CLASS)THIS */
+         node = build_cast (EXPR_WFL_LINECOL (qual_wfl), 
+                            CLASSTYPE_SUPER (current_class),
+                            build_this (EXPR_WFL_LINECOL (qual_wfl)));
+         *where_found = decl = java_complete_tree (node);
+         *type_found = type = QUAL_DECL_TYPE (decl);
+         from_super = from_type = 1;
+         continue;
+       }
+
+      /* 15.13.1: Can't search for field name in packages, so we
+        assume a variable/class name was meant. */
+      if (RESOLVE_PACKAGE_NAME_P (qual_wfl))
+       {
+         if (from_super || from_cast)
+           parse_error_context 
+             ((from_cast ? qual_wfl : wfl),
+              "No variable `%s' defined in class `%s'",
+              IDENTIFIER_POINTER (EXPR_WFL_NODE (qual_wfl)),
+              lang_printable_name (type));
+         else
+           parse_error_context
+             (qual_wfl, "Undefined variable or class name: `%s'",
+              IDENTIFIER_POINTER (EXPR_WFL_NODE (qual_wfl)));
+         return 1;
+       }
+
+      /* We have a type name. It's been already resolved when the
+        expression was qualified. */
+      else if (RESOLVE_TYPE_NAME_P (qual_wfl))
+       {
+         if (!(decl = QUAL_RESOLUTION (q)))
+           return 1;           /* Error reported already */
+
+         if (not_accessible_p (TREE_TYPE (decl), decl, 0))
+           {
+             parse_error_context 
+               (qual_wfl, "Can't access %s field `%s.%s' from `%s'",
+                java_accstring_lookup (get_access_flags_from_decl (decl)),
+                IDENTIFIER_POINTER (DECL_NAME (TYPE_NAME (type))),
+                IDENTIFIER_POINTER (DECL_NAME (decl)),
+                IDENTIFIER_POINTER (DECL_NAME (TYPE_NAME (current_class))));
+             return 1;
+           }
+         
+         type = TREE_TYPE (decl);
+         from_type = 1;
+       }
+      /* We resolve and expression name */
+      else 
+       {
+         tree field_decl;
+
+         /* If there exists an early resolution, use it. That occurs
+            only once and we know that there are more things to
+            come. Don't do that when processing something after SUPER
+            (we need more thing to be put in place below */
+         if (!from_super && QUAL_RESOLUTION (q))
+           decl = QUAL_RESOLUTION (q);
+
+         /* We have to search for a field, knowing the type of its
+             container. The flag FROM_TYPE indicates that we resolved
+             the last member of the expression as a type name, which
+             means that for the resolution of this field, will check
+             on other errors than if the it was resolved as a member
+             of an other field. */
+         else
+           {
+             int is_static;
+             if (!from_type && !JREFERENCE_TYPE_P (type))
+               {
+                 parse_error_context 
+                   (qual_wfl, "Attempt to reference field `%s' in `%s %s'",
+                    IDENTIFIER_POINTER (EXPR_WFL_NODE (qual_wfl)),
+                    lang_printable_name (type),
+                    IDENTIFIER_POINTER (DECL_NAME (field_decl)));
+                 return 1;
+               }
+             
+             if (!(field_decl = 
+                   lookup_field_wrapper (type, EXPR_WFL_NODE (qual_wfl))))
+               {
+                 parse_error_context 
+                   (qual_wfl, "No variable `%s' defined in class `%s'",
+                    IDENTIFIER_POINTER (EXPR_WFL_NODE (qual_wfl)), 
+                    IDENTIFIER_POINTER (DECL_NAME (TYPE_NAME (type))));
+                 return 1;
+               }
+             
+             /* Check on accessibility here */
+             if (not_accessible_p (type, field_decl, from_super))
+               {
+                 parse_error_context 
+                   (qual_wfl,
+                    "Can't access %s field `%s.%s' from `%s'",
+                    java_accstring_lookup 
+                      (get_access_flags_from_decl (field_decl)),
+                    IDENTIFIER_POINTER (DECL_NAME (TYPE_NAME (type))),
+                    IDENTIFIER_POINTER (DECL_NAME (field_decl)),
+                    IDENTIFIER_POINTER 
+                      (DECL_NAME (TYPE_NAME (current_class))));
+                 return 1;
+               }
+             
+             /* There are things to check when fields are accessed
+                from type. There are no restrictions on a static
+                declaration of the field when it is accessed from an
+                interface */
+             is_static = FIELD_STATIC (field_decl);
+             if (!from_super && from_type 
+                 && !TYPE_INTERFACE_P (type) && !is_static)
+               {
+                 parse_error_context 
+                   (qual_wfl, "Can't make a static reference to nonstatic "
+                    "variable `%s' in class `%s'",
+                    IDENTIFIER_POINTER (EXPR_WFL_NODE (qual_wfl)),
+                    IDENTIFIER_POINTER (DECL_NAME (TYPE_NAME (type))));
+                 return 1;
+               }
+             from_cast = from_super = 0;
+
+             /* If we need to generate something to get a proper handle
+                on what this field is accessed from, do it now. */
+             if (!is_static)
+               {
+                 decl = maybe_access_field (decl, *where_found, type);
+                 if (decl == error_mark_node)
+                   return 1;
+               }
+
+             /* We want to keep the location were found it, and the type
+                we found. */
+             *where_found = decl;
+             *type_found = type;
+
+             /* This is the decl found and eventually the next one to
+                search from */
+             decl = field_decl;
+           }
+
+         from_type = 0;
+         type = QUAL_DECL_TYPE (decl);
+       }
+    }
+  *found_decl = decl;
+  return 0;
+}
+
+/* 6.6 Qualified name and access control. Returns 1 if MEMBER (a decl)
+   can't be accessed from REFERENCE (a record type). */
+
+int not_accessible_p (reference, member, from_super)
+     tree reference, member;
+     int from_super;
+{
+  int access_flag = get_access_flags_from_decl (member);
+
+  /* Access always granted for members declared public */
+  if (access_flag & ACC_PUBLIC)
+    return 0;
+  
+  /* Check access on protected members */
+  if (access_flag & ACC_PROTECTED)
+    {
+      /* Access granted if it occurs from within the package
+         containing the class in which the protected member is
+         declared */
+      if (class_in_current_package (DECL_CONTEXT (member)))
+       return 0;
+
+      if (TREE_CODE (member) == FUNCTION_DECL && DECL_CONSTRUCTOR_P (member))
+       {
+         /* Access from SUPER is granted */
+         if (from_super)
+           return 0;
+         /* Otherwise, access isn't granted */
+         return 1;
+       }
+      else
+       {
+         /* If accessed with the form `super.member', then access is
+             granted */
+         if (from_super)
+           return 0;
+
+         /* Otherwise, access is granted if occuring from the class where
+            member is declared or a subclass of it */
+         if (inherits_from_p (reference, current_class))
+           return 0;
+       }
+      return 1;
+    }
+
+  /* Check access on private members. Access is granted only if it
+     occurs from within the class in witch it is declared*/
+
+  if (access_flag & ACC_PRIVATE)
+    return (current_class == DECL_CONTEXT (member) ? 0 : 1);
+
+  /* Default access are permitted only when occuring within the
+     package in which the type (REFERENCE) is declared. In other words,
+     REFERENCE is defined in the current package */
+  if (ctxp->package)
+    return !class_in_current_package (reference);
+  
+  /* Otherwise, access is granted */
+  return 0;
+}
+
+/* Returns 1 if class was declared in the current package, 0 otherwise */
+
+static int
+class_in_current_package (class)
+     tree class;
+{
+  static tree cache = NULL_TREE;
+  int qualified_flag;
+  tree left;
+
+  if (cache == class)
+    return 1;
+
+  qualified_flag = QUALIFIED_P (DECL_NAME (TYPE_NAME (class)));
+
+  /* If the current package is empty and the name of CLASS is
+     qualified, class isn't in the current package.  If there is a
+     current package and the name of the CLASS is not qualified, class
+     isn't in the current package */
+  if (!ctxp->package && qualified_flag || ctxp->package && !qualified_flag)
+    return 0;
+
+  /* If there is not package and the name of CLASS isn't qualified,
+     they belong to the same unnamed package */
+  if (!ctxp->package && !qualified_flag)
+    return 1;
+
+  /* Compare the left part of the name of CLASS with the package name */
+  breakdown_qualified (&left, NULL, DECL_NAME (TYPE_NAME (class)));
+  if (ctxp->package == left)
+    {
+      cache = class;
+      return 1;
+    }
+  return 0;
+}
+
+/* This function may generate code to access DECL from WHERE. This is
+   done only if certain conditions meet.  */
+
+static tree
+maybe_access_field (decl, where, type)
+  tree decl, where, type;
+{
+  if (DECL_P (decl) && decl != current_this
+      && (!(TREE_CODE (decl) != PARM_DECL
+           && FIELD_STATIC (decl)))
+      && !IDENTIFIER_LOCAL_VALUE (DECL_NAME (decl)))
+    decl = build_field_ref (where ? where : current_this, 
+                           type, DECL_NAME (decl));
+  return decl;
+}
+
+/* Build a method invocation statement, by patching PATCH. If non NULL
+   and according to the situation, PRIMARY and WHERE may be
+   used. IS_STATIC is set to 1 if the invoked function is static. */
+
+static tree
+patch_method_invocation_stmt (patch, primary, where, is_static)
+     tree patch, primary, where;
+     int *is_static;
+{
+  tree wfl = TREE_OPERAND (patch, 0);
+  tree args = TREE_OPERAND (patch, 1);
+  tree name = EXPR_WFL_NODE (wfl);
+  tree list, class_type;
+  
+  /* Should be overriden if everything goes well. Otherwise, if
+     something fails, it should keep this value. It stop the
+     evaluation of a bogus assignment. See java_complete_tree,
+     MODIFY_EXPR: for the reasons why we sometimes want to keep on
+     evaluating an assignment */
+  TREE_TYPE (patch) = error_mark_node;
+
+  /* Since lookup functions are messing with line numbers, save the
+     context now.  */
+  java_parser_context_save_global ();
+
+  /* 15.11.1: Compile-Time Step 1: Determine Class or Interface to Search */
+
+  /* Resolution of qualified name, excluding constructors */
+  if (QUALIFIED_P (name) && !CALL_CONSTRUCTOR_P (patch))
+    {
+      tree class_decl, identifier, identifier_wfl;
+      /* Extract the last IDENTIFIER of the qualified
+        expression. This is a wfl and we will use it's location
+        data during error report. */
+      identifier_wfl = cut_identifier_in_qualified (wfl);
+      identifier = EXPR_WFL_NODE (identifier_wfl);
+      
+      /* Given the context, IDENTIFIER is syntactically qualified
+        as a MethodName. We need to qualify what's before */
+      qualify_ambiguous_name (wfl);
+
+      /* Package resolution are erroneous */
+      if (RESOLVE_PACKAGE_NAME_P (wfl))
+       {
+         tree remainder;
+         breakdown_qualified (&remainder, NULL, EXPR_WFL_NODE (wfl));
+         parse_error_context (wfl, "Can't search method `%s' in package "
+                              "`%s'",IDENTIFIER_POINTER (identifier),
+                              IDENTIFIER_POINTER (remainder));
+         return error_mark_node;
+       }
+      /* We're resolving a call from a type */
+      else if (RESOLVE_TYPE_NAME_P (wfl))
+       {
+         tree decl = QUAL_RESOLUTION (EXPR_WFL_QUALIFICATION (wfl));
+         tree name = DECL_NAME (decl);
+         tree type;
+
+         class_decl = resolve_and_layout (name, wfl);
+         if (CLASS_INTERFACE (decl))
+           {
+             parse_error_context
+               (identifier_wfl, "Can't make static reference to method "
+                "`%s' in interface `%s'", IDENTIFIER_POINTER (identifier), 
+                IDENTIFIER_POINTER (name));
+             return error_mark_node;
+           }
+         /* Look the method up in the type selector. The method ought
+             to be static. */
+         type = TREE_TYPE (class_decl);
+         list = lookup_method_invoke (0, wfl, type, identifier, args);
+         if (list && !METHOD_STATIC (list))
+           {
+             char *fct_name = strdup ((char *)lang_printable_name (list));
+             parse_error_context 
+               (identifier_wfl,
+                "Can't make static reference to method `%s %s' in class `%s'",
+                lang_printable_name (TREE_TYPE (TREE_TYPE (list))), fct_name, 
+                IDENTIFIER_POINTER (DECL_NAME (TYPE_NAME (type))));
+             free (fct_name);
+             return error_mark_node;
+           }
+       }
+      /* We're resolving an expression name */
+      else
+       {
+         tree field, type;
+         
+         /* 1- Find the field to which the call applies */
+         field = resolve_field_access (wfl, NULL, &type);
+         if (field == error_mark_node)
+           return error_mark_node;
+         
+         /* 2- Do the layout of the class where the last field
+            was found, so we can search it. */
+         class_decl = 
+           resolve_and_layout (DECL_NAME (TYPE_NAME (type)), NULL_TREE);
+         
+         /* 3- Retrieve a filtered list of method matches, Refine
+            if necessary. In any cases, point out errors.  */
+         list = lookup_method_invoke (0, identifier_wfl, type, 
+                                      identifier, args);
+
+         /* 4- Add the field as an argument */
+         args = tree_cons (NULL_TREE, field, args);
+       }
+
+      /* CLASS_TYPE is used during the call to not_accessible_p and
+        IDENTIFIER_WFL will be used to report any problem further */
+      class_type = TREE_TYPE (class_decl);
+      wfl = identifier_wfl;
+    }
+  /* Resolution of simple names, names generated after a primary: or
+     constructors */
+  else
+    {
+      tree class_to_search;
+      int lc;          /* Looking for Constructor */
+      
+      /* We search constructor in their target class */
+      if (CALL_CONSTRUCTOR_P (patch))
+       {
+         class_to_search = resolve_no_layout (EXPR_WFL_NODE (wfl), NULL_TREE);
+         if (!class_to_search)
+           {
+             parse_error_context 
+               (wfl, "Class `%s' not found in type declaration",
+                IDENTIFIER_POINTER (EXPR_WFL_NODE (wfl)));
+             return error_mark_node;
+           }
+         
+         /* Can't instantiate an abstract class */
+         if (CLASS_ABSTRACT (class_to_search))
+           {
+             parse_error_context 
+               (wfl, "Class `%s' is an abstract class. It can't be "
+                "instantiated", IDENTIFIER_POINTER (EXPR_WFL_NODE (wfl)));
+             return error_mark_node;
+           }
+         class_to_search = TREE_TYPE (class_to_search);
+         lc = 1;
+       }
+      /* This is a regular search in the local class, unless an
+         alternate class is specified. */
+      else
+       {
+         class_to_search = (where ? where : current_class);
+         lc = 0;
+       }
+      
+      /* NAME is a simple identifier or comes from a primary. Search
+        in the class whose declaration contain the method being
+        invoked. */
+      list = lookup_method_invoke (lc, wfl, class_to_search, name, args);
+
+      /* Don't continue if no method were found, as the next statement
+         can't be executed then. */
+      if (!list) return error_mark_node;
+
+      /* Check for static reference if non static methods */
+      if (check_for_static_method_reference (wfl, patch, list, 
+                                            class_to_search, primary))
+       return error_mark_node;
+
+      /* Non static/constructor methods are called with the current
+        object extra argument. If method is resolved as a primary,
+        use the primary otherwise use the current THIS. */
+      if (!CALL_CONSTRUCTOR_P (patch) && !METHOD_STATIC (list))
+       args = tree_cons (NULL_TREE, primary ? primary : current_this, args);
+
+      class_type = class_to_search;
+    }
+  
+  /* Merge point of all resolution schemes. If we have nothing, this
+     is an error, already signaled */
+  if (!list) return error_mark_node;
+  
+  /* Check accessibility, position the is_static flag, build and
+     return the call */
+  if (not_accessible_p (class_type, list, 0))
+    {
+      char *fct_name = strdup ((char *)lang_printable_name (list));
+      parse_error_context 
+       (wfl, "Can't access %s method `%s %s.%s' from `%s'",
+        java_accstring_lookup (get_access_flags_from_decl (list)),
+        lang_printable_name (TREE_TYPE (TREE_TYPE (list))), 
+        IDENTIFIER_POINTER (DECL_NAME (TYPE_NAME (class_type))), fct_name,
+        IDENTIFIER_POINTER (DECL_NAME (TYPE_NAME (current_class))));
+      free (fct_name);
+      return error_mark_node;
+    }
+  
+  if (is_static) 
+    *is_static = METHOD_STATIC (list);
+  java_parser_context_restore_global ();
+  return patch_invoke (patch, list, args, wfl);
+}
+
+/* Check that we're not trying to do a static reference to a method in
+   non static method. Return 1 if it's the case, 0 otherwise. */
+
+static int
+check_for_static_method_reference (wfl, node, method, where, primary)
+     tree wfl, node, method, where, primary;
+{
+  if (METHOD_STATIC (current_function_decl) 
+      && !METHOD_STATIC (method) && !primary && !CALL_CONSTRUCTOR_P (node))
+    {
+      char *fct_name = strdup ((char *)lang_printable_name (method));
+      parse_error_context 
+       (wfl, "Can't make static reference to method `%s %s' in class `%s'", 
+        lang_printable_name (TREE_TYPE (TREE_TYPE (method))), fct_name,
+        IDENTIFIER_POINTER (DECL_NAME (TYPE_NAME (where))));
+      free (fct_name);
+      return 1;
+    }
+  return 0;
+}
+
+/* Patch an invoke expression METHOD and ARGS, based on its invocation
+   mode.  */
+
+static tree
+patch_invoke (patch, method, args, cl)
+     tree patch, method, args;
+     tree cl;
+{
+  tree dtable, func;
+  tree signature = build_java_signature (TREE_TYPE (method));
+  tree original_call;
+
+  switch (invocation_mode (method, 0))
+    {
+    case INVOKE_VIRTUAL:
+      dtable = invoke_build_dtable (0, args);
+      func = build_invokevirtual (dtable, method);
+      break;
+    case INVOKE_STATIC:
+      func = build_known_method_ref (method, TREE_TYPE (method),
+                                    DECL_CONTEXT (method),
+                                    signature, args);
+      args = nreverse (args);
+      break;
+
+    default:
+      fatal ("Unknown invocation mode - build_invoke");
+      return NULL_TREE;
+    }
+
+
+  /* Ensure self_type is initialized, (invokestatic). FIXME */
+  func = build1 (NOP_EXPR, build_pointer_type (TREE_TYPE (method)), func);
+  TREE_TYPE (patch) = TREE_TYPE (TREE_TYPE (method));
+  TREE_OPERAND (patch, 0) = func;
+  TREE_OPERAND (patch, 1) = args;
+  original_call = patch;
+
+  /* We're calling a constructor. New is called an its returned value
+     is an argument to the constructor. We build a COMPOUND_EXPR and
+     use saved expression so that the overall NEW expression value is
+     a pointer to a newly created and initialized class. */
+  if (CALL_CONSTRUCTOR_P (original_call))
+    {
+      tree class = DECL_CONTEXT (method);
+      tree c1, saved_new, size, new;
+      if (!TYPE_SIZE (class))
+       safe_layout_class (class);
+      size = size_in_bytes (class);
+      new = build (CALL_EXPR, promote_type (class),
+                  build_address_of (alloc_object_node),
+                  tree_cons (NULL_TREE, build_class_ref (class),
+                             build_tree_list (NULL_TREE, 
+                                              size_in_bytes (class))),
+                  NULL_TREE);
+      saved_new = save_expr (new);
+      c1 = build_tree_list (NULL_TREE, saved_new);
+      TREE_CHAIN (c1) = TREE_OPERAND (original_call, 1);
+      TREE_OPERAND (original_call, 1) = c1;
+      TREE_SET_CODE (original_call, CALL_EXPR);
+      patch = build (COMPOUND_EXPR, TREE_TYPE (new), patch, saved_new);
+    }
+  return patch;
+}
+
+static int
+invocation_mode (method, super)
+     tree method;
+     int super;
+{
+  int access = get_access_flags_from_decl (method);
+
+  if (access & ACC_STATIC)
+    return INVOKE_STATIC;
+
+  if (CLASS_FINAL (TYPE_NAME (DECL_CONTEXT (method))))
+    return INVOKE_STATIC;
+  
+  if (super)
+    return INVOKE_SUPER;
+  
+  if (CLASS_INTERFACE (TYPE_NAME (DECL_CONTEXT (method))))
+    return INVOKE_INTERFACE;
+  
+  if (DECL_CONSTRUCTOR_P (method))
+    return INVOKE_STATIC;
+  
+  return INVOKE_VIRTUAL;
+}
+
+/* Retrieve a refined list of matching methods. */
+
+static tree
+lookup_method_invoke (lc, cl, class, name, arg_list)
+     int lc;
+     tree cl;
+     tree class, name, arg_list;
+{
+  tree method = make_node (FUNCTION_TYPE);
+  tree arg_type_list = NULL_TREE;
+  tree signature, list, node, scratch;
+
+  for (node = arg_list; node; node = TREE_CHAIN (node))
+    {
+      tree current_arg;
+      current_arg = 
+       build_tree_list (NULL_TREE,
+                        promote_type (TREE_TYPE (TREE_VALUE (node))));
+      arg_type_list = chainon (current_arg, arg_type_list);
+    }
+  TYPE_ARG_TYPES (method) = arg_type_list;
+
+  if (!lc)
+    {
+      signature = build_java_argument_signature (method);
+      list = match_java_method (class, name, signature);
+      list = refine_accessible_methods_list (lc, list);
+    }
+  else
+    {
+      TREE_TYPE (method) = void_type_node;
+      signature = build_java_signature (method);
+      list = lookup_java_constructor (class, signature);
+    }
+
+  if (!list)
+    {
+      parse_error_context (cl, "Can't find method `%s(%s)' in class `%s'",
+                          IDENTIFIER_POINTER (name),
+                          IDENTIFIER_POINTER (signature),
+                          IDENTIFIER_POINTER (DECL_NAME (TYPE_NAME (class))));
+      return NULL_TREE;
+    }
+
+  if (lc)
+    return list;
+
+  if (TREE_CHAIN (list))
+    {
+      tree most_specific_list = NULL_TREE;
+      tree current;
+      /* 15.11.2.2 Choose the Most Specific Method */
+      for (current = list; current; current = TREE_CHAIN (current))
+       {
+         tree rest;
+         tree method = TREE_VALUE (list);
+         tree class_from = DECL_CONTEXT (method);
+         for (rest = TREE_CHAIN (current); rest; rest = TREE_CHAIN (rest))
+           {
+             tree other = TREE_VALUE (rest);
+
+             /* METHOD can be declared more specific with regard to OTHER iif:
+               
+                - The class METHOD belongs can be converted to the
+                  class OTHER belongs by method invocation conversion
+                  (5.3).  Since we're dealing with classes here, it is
+                  covered by the identity conversion or the windening
+                  primitive conversion.
+               
+                - The types of the arguments of METHOD can be
+                  converted to the types of the arguments of OTHER by
+                  method invocation conversion (5.3). */
+
+             if (valid_ref_assignconv_cast_p (class_from, 
+                                              DECL_CONTEXT (other), 0)
+                 && 1)         /* Test on args non implemented */
+               most_specific_list = tree_cons (NULL_TREE, method, 
+                                               most_specific_list);
+           }
+       }
+      list = most_specific_list;
+    }
+
+  if (!list || TREE_CHAIN (list))
+    {
+      parse_error_context (cl, "Can't find method `%s(%s)' in class `%s'",
+                          IDENTIFIER_POINTER (name),
+                          IDENTIFIER_POINTER (signature),
+                          IDENTIFIER_POINTER (DECL_NAME (TYPE_NAME (class))));
+      return NULL_TREE;
+    }
+
+  /* 15.11.3 Is the Chosen Method Appropriate ? */
+  else
+    return TREE_VALUE (list);
+}
+
+/* Refine accessible methods from the raw matching method list, as
+   specified in 15.11.4.3. Return a (possibly empty) new method
+   list.  */
+
+static tree
+refine_accessible_methods_list (lc, list)
+     int lc;                   /* Looking for Constructor */
+     tree list;
+{
+#define ADD_TO_LIST_AND_CONTINUE                               \
+  {                                                            \
+    refined_list = tree_cons (NULL_TREE, method, refined_list);        \
+    continue;                                                  \
+  }
+  tree node, refined_list = NULL_TREE;
+  tree current_class_name = DECL_NAME (TYPE_NAME (current_class));
+
+  for (node = list; node; node = TREE_CHAIN (node))
+    {
+      int access, identical;
+      tree class_from, method, class_from_name;
+      
+      method = TREE_VALUE (node);
+
+      /* Constructor not retained here, unless were specifically
+       looking for them. */
+      if (lc && DECL_CONSTRUCTOR_P (method))
+       ADD_TO_LIST_AND_CONTINUE;
+
+      access = get_access_flags_from_decl (method);
+      class_from = DECL_CONTEXT (method);
+      class_from_name = DECL_NAME (TYPE_NAME (class_from));
+      
+      identical = identical_subpath_p (current_class_name, class_from_name);
+
+      /* Check accessibility of class_from from the current one: This
+        test has been already carried out when qualify_ambiguous_name
+        tried to resolve a type found in an other package. It is not
+        necessary to retest things here, the error has been already
+        reported. */
+                 
+      /* Public method are always OK */
+      if (access & ACC_PUBLIC)
+       ADD_TO_LIST_AND_CONTINUE;
+      
+      /* Protected method access is OK if classes are from the
+        same package or part of the same inheritance lineage */
+      if ((access & ACC_PROTECTED)
+         && (inherits_from_p (current_class, class_from) || identical))
+       ADD_TO_LIST_AND_CONTINUE;
+
+      /* Methods with default (package) access are OK if classes are
+        from the same default package. */
+      if (identical || 
+         (!QUALIFIED_P (class_from_name) && !QUALIFIED_P (current_class_name)))
+       ADD_TO_LIST_AND_CONTINUE;
+
+      /* Private method accessible iff current class is the node where
+        the method is defined */
+      if ((access & ACC_PRIVATE) && (class_from == current_class))
+       ADD_TO_LIST_AND_CONTINUE;
+    }
+#undef ADD_TO_LIST_AND_CONTINUE
+  return refined_list;
+}
+
+/* Qualification routines */
+
+static void
+qualify_ambiguous_name (id)
+     tree id;
+{
+  tree qual, qual_wfl, name, decl, ptr_type, saved_current_class;
+  int again, super_found = 0, this_found = 0;
+
+  /* We first qualify the first element, then derive qualification of
+     others based on the first one. If the first element is qualified
+     by a resolution (field or type), this resolution is stored in the
+     QUAL_RESOLUTION of the qual element being examined. We need to
+     save the current_class since the use of SUPER might change the
+     its value. */
+  saved_current_class = current_class;
+  qual = EXPR_WFL_QUALIFICATION (id);
+  do {
+
+    /* Simple qualified expression feature a qual_wfl that is a
+       WFL. Expression derived from a primary feature more complicated
+       things like a CALL_EXPR. Expression from primary need to be
+       worked out to extract the part on which the qualification will
+       take place. */
+    qual_wfl = QUAL_WFL (qual);
+    switch (TREE_CODE (qual_wfl))
+      {
+      case CALL_EXPR:
+       qual_wfl = TREE_OPERAND (qual_wfl, 0);
+       if (TREE_CODE (qual_wfl) != EXPR_WITH_FILE_LOCATION)
+         {
+           qual = EXPR_WFL_QUALIFICATION (qual_wfl);
+           qual_wfl = QUAL_WFL (qual);
+         }
+       break;
+      case JAVA_NEW_CLASS_EXPR:
+      case CONVERT_EXPR:
+      case ARRAY_REF:
+       qual_wfl = TREE_OPERAND (qual_wfl, 0);
+       break;
+      }
+    name = EXPR_WFL_NODE (qual_wfl);
+    ptr_type = current_class;
+    again = 0;
+    /* If we have a THIS (from a primary), we set the context accordingly */
+    if (name == this_identifier_node)
+      {
+       qual = TREE_CHAIN (qual);
+       qual_wfl = QUAL_WFL (qual);
+       name = EXPR_WFL_NODE (qual_wfl);
+       this_found = 1;
+      }
+    /* If we have a SUPER, we set the context accordingly */
+    if (name == super_identifier_node)
+      {
+       current_class = CLASSTYPE_SUPER (ptr_type);
+       /* Check that there is such a thing as a super class. If not,
+          return.  The error will be caught later on, during the
+          resolution */
+       if (!current_class)
+         {
+           current_class = saved_current_class;
+           return;
+         }
+       qual = TREE_CHAIN (qual);
+       /* Do one more interation to set things up */
+       super_found = again = 1;
+      }
+  } while (again);
+  
+  /* If name appears within the scope of a location variable
+     declaration or parameter declaration, then it is an expression
+     name. We don't carry this test out if we're in the context of the
+     use of SUPER or THIS */
+
+  if (!this_found && !super_found && (decl = IDENTIFIER_LOCAL_VALUE (name)))
+    {
+      RESOLVE_EXPRESSION_NAME_P (qual_wfl) = 1;
+      QUAL_RESOLUTION (qual) = decl;
+    }
+
+  /* If within the class/interface NAME was found to be used there
+     exists a (possibly inherited) field named NAME, then this is an
+     expression name. */
+  else if ((decl = lookup_field_wrapper (ptr_type, name)))
+    {
+      RESOLVE_EXPRESSION_NAME_P (qual_wfl) = 1;
+      QUAL_RESOLUTION (qual) = decl;
+    }
+
+  /* We reclassify NAME as a type name if:
+     - NAME is a class/interface declared within the compilation
+       unit containing NAME,
+     - NAME is imported via a single-type-import declaration,
+     - NAME is declared in an another compilation unit of the package
+       of the compilation unit containing NAME,
+     - NAME is declared by exactly on type-import-on-demand declaration
+     of the compilation unit containing NAME. */
+  else if ((decl = resolve_and_layout (name, NULL_TREE)))
+    {
+      RESOLVE_TYPE_NAME_P (qual_wfl) = 1;
+      QUAL_RESOLUTION (qual) = decl;
+    }
+
+  /* Method call are expression name */
+  else if (TREE_CODE (QUAL_WFL (qual)) == CALL_EXPR)
+    RESOLVE_EXPRESSION_NAME_P (qual_wfl) = 1;
+
+  /* Check here that NAME isn't declared by more than one
+     type-import-on-demand declaration of the compilation unit
+     containing NAME. FIXME */
+
+  /* Otherwise, NAME is reclassified as a package name */
+  else 
+    RESOLVE_PACKAGE_NAME_P (qual_wfl) = 1;
+
+  /* Propagate the qualification accross other components of the
+     qualified name */
+  for (qual = TREE_CHAIN (qual); qual;
+       qual_wfl = QUAL_WFL (qual), qual = TREE_CHAIN (qual))
+    {
+      if (RESOLVE_PACKAGE_NAME_P (qual_wfl))
+       RESOLVE_PACKAGE_NAME_P (QUAL_WFL (qual)) = 1;
+      else 
+       RESOLVE_EXPRESSION_NAME_P (QUAL_WFL (qual)) = 1;
+    }
+
+  /* Store the global qualification for the ambiguous part of ID back
+     into ID fields */
+  if (RESOLVE_EXPRESSION_NAME_P (qual_wfl))
+    RESOLVE_EXPRESSION_NAME_P (id) = 1;
+  else if (RESOLVE_TYPE_NAME_P (qual_wfl))
+    RESOLVE_TYPE_NAME_P (id) = 1;
+  else if (RESOLVE_PACKAGE_NAME_P (qual_wfl))
+    RESOLVE_PACKAGE_NAME_P (id) = 1;
+
+  /* Restore the current class */
+  current_class = saved_current_class;
+}
+
+static int
+breakdown_qualified (left, right, source)
+    tree *left, *right, source;
+{
+  char *p = IDENTIFIER_POINTER (source), *base;
+  int   l = IDENTIFIER_LENGTH (source);
+
+  /* Breakdown NAME into REMAINDER . IDENTIFIER */
+  base = p;
+  p += (l-1);
+  while (*p != '.' && p != base)
+    p--;
+
+  /* We didn't find a '.'. Return an error */
+  if (p == base)
+    return 1;
+
+  *p = '\0';
+  if (right)
+    *right = get_identifier (p+1);
+  *left = get_identifier (IDENTIFIER_POINTER (source));
+  *p = '.';
+  
+  return 0;
+}
+
+/* Return 1 if N1 and N2 have identical sub-path. */
+
+static int
+identical_subpath_p (n1, n2)
+     tree n1, n2;
+{
+  tree left1, left2;
+
+  if (!QUALIFIED_P (n1) || !QUALIFIED_P (n2))
+    return n1 == n2;
+  
+  breakdown_qualified (&left1, NULL, n1);
+  breakdown_qualified (&left2, NULL, n2);
+
+  return left1 == left2;
+}
+
+static int
+not_initialized_as_it_should_p (decl)
+     tree decl;
+{
+  if (DECL_P (decl))
+    {
+      if (TREE_CODE (decl) == FIELD_DECL
+         && METHOD_STATIC (current_function_decl))
+       return 0;
+      return DECL_P (decl) && !INITIALIZED_P (decl);
+    }
+  return 0;
+}
+
+/* Patch tree nodes in a function body. When a BLOCK is found, push
+   local variable decls if present.  */
+
+static tree
+java_complete_tree (node)
+     tree node;
+{
+  tree nn, cn, wfl_op1, wfl_op2;
+  int flag, location;
+
+  /* CONVERT_EXPR always has its type set, even though it needs to be
+     worked out */
+  if (TREE_TYPE (node) && TREE_CODE (node) != CONVERT_EXPR)
+    return node;
+
+  /* The switch block implements cases processing container nodes
+     first.  Contained nodes are always written back. Leaves come
+     next and return a value. */
+  switch (TREE_CODE (node))
+    {
+    case BLOCK:
+
+      /* 1- Block section.
+        Set the local values on decl names so we can identify them
+        faster when they're referenced. At that stage, identifiers
+        are legal so we don't check for declaration errors. */
+      for (cn = BLOCK_EXPR_DECLS (node); cn; cn = TREE_CHAIN (cn))
+       {
+         DECL_CONTEXT (cn) = current_function_decl;
+         IDENTIFIER_LOCAL_VALUE (DECL_NAME (cn)) = cn;
+         INITIALIZED_P (cn) = 0;
+       }
+      if (BLOCK_EXPR_BODY (node))
+       {
+         BLOCK_EXPR_BODY (node) = java_complete_tree (BLOCK_EXPR_BODY (node));
+         if (BLOCK_EXPR_BODY (node) == error_mark_node)
+           return error_mark_node;
+       }
+      /* Turn local bindings to null */
+      for (cn = BLOCK_EXPR_DECLS (node); cn; cn = TREE_CHAIN (cn))
+       IDENTIFIER_LOCAL_VALUE (DECL_NAME (cn)) = NULL_TREE;
+
+      TREE_TYPE (node) = void_type_node;
+      break;
+
+      /* 2- They are expressions but ultimately deal with statements */
+    case LABELED_BLOCK_EXPR:
+      PUSH_LABELED_BLOCK (node);
+      if (LABELED_BLOCK_BODY (node))
+       COMPLETE_CHECK_OP_1 (node);
+      TREE_TYPE (node) = void_type_node;
+      POP_LABELED_BLOCK ();
+      return node;
+
+    case EXIT_BLOCK_EXPR:
+      /* We don't complete operand 1, because it's the return value of
+         the EXIT_BLOCK_EXPR which doesn't exist it Java */
+      return patch_bc_statement (node);
+
+    case LOOP_EXPR:
+      PUSH_LOOP (node);
+      /* Check whether the loop was enclosed in a labeled
+         statement. If not, create one, insert the loop in it and
+         return the node */
+      nn = patch_loop_statement (node);
+      /* Anyways, walk the body of the loop */
+      TREE_OPERAND (node, 0) = java_complete_tree (TREE_OPERAND (node, 0));
+      if (TREE_OPERAND (node, 0) == error_mark_node)
+       return error_mark_node;
+      TREE_TYPE (nn) = TREE_TYPE (node) = void_type_node;
+      /* If we returned something different, that's because we
+         inserted a label. Pop the label too. */
+      if (nn != node)
+       POP_LABELED_BLOCK ();
+      POP_LOOP ();
+      return nn;
+
+    case EXIT_EXPR:
+      TREE_OPERAND (node, 0) = java_complete_tree (TREE_OPERAND (node, 0));
+      return patch_exit_expr (node);
+
+    case COND_EXPR:
+      /* Condition */
+      TREE_OPERAND (node, 0) = java_complete_tree (TREE_OPERAND (node, 0));
+      if (TREE_OPERAND (node, 0) == error_mark_node)
+       return error_mark_node;
+      /* then-else branches */
+      TREE_OPERAND (node, 1) = java_complete_tree (TREE_OPERAND (node, 1));
+      if (TREE_OPERAND (node, 1) == error_mark_node)
+       return error_mark_node;
+      TREE_OPERAND (node, 2) = java_complete_tree (TREE_OPERAND (node, 2));
+      if (TREE_OPERAND (node, 2) == error_mark_node)
+       return error_mark_node;
+      return patch_if_else_statement (node);
+      break;
+
+      /* 3- Expression section */
+    case COMPOUND_EXPR:
+      TREE_OPERAND (node, 0) = java_complete_tree (TREE_OPERAND (node, 0));
+      TREE_OPERAND (node, 1) = java_complete_tree (TREE_OPERAND (node, 1));
+      if (TREE_OPERAND (node, 1) == error_mark_node)
+       return error_mark_node;
+      TREE_TYPE (node) = TREE_TYPE (TREE_OPERAND (node, 1));
+      break;
+
+    case RETURN_EXPR:
+      return patch_return (node);
+
+    case EXPR_WITH_FILE_LOCATION:
+      if (!EXPR_WFL_NODE (node) /* Or a PRIMARY flag ? */
+         || TREE_CODE (EXPR_WFL_NODE (node)) == IDENTIFIER_NODE)
+        return resolve_expression_name (node);
+      else
+       {
+         EXPR_WFL_NODE (node) = java_complete_tree (EXPR_WFL_NODE (node));
+         TREE_SIDE_EFFECTS (node) = 1;
+         if (EXPR_WFL_NODE (node) == error_mark_node)
+           {
+             /* Its important for the evaluation of assignment that
+                this mark on the TREE_TYPE is propagated. */
+             TREE_TYPE (node) = error_mark_node;
+             return error_mark_node;
+           }
+         else
+           TREE_TYPE (node) = TREE_TYPE (EXPR_WFL_NODE (node));
+       }
+      break;
+
+    case JAVA_NEW_ARRAY_EXPR:
+      /* Patch all the dimensions */
+      flag = 0;
+      for (cn = TREE_OPERAND (node, 1); cn; cn = TREE_CHAIN (cn))
+       {
+         int location = EXPR_WFL_LINECOL (TREE_VALUE (cn));
+         tree dim = java_complete_tree (TREE_VALUE (cn));
+         if (dim == error_mark_node)
+           {
+             flag = 1;
+             continue;
+           }
+         else
+           {
+             TREE_VALUE (cn) = save_expr (dim);
+             /* Setup the location of the current dimension, for
+                later error report. */
+             TREE_PURPOSE (cn) = 
+               build_expr_wfl (NULL_TREE, input_filename, 0, 0);
+             EXPR_WFL_LINECOL (TREE_PURPOSE (cn)) = location;
+           }
+       }
+      /* They complete the array creation expression, if no errors
+         were found. */
+      return (flag ? error_mark_node : patch_newarray (node));
+
+    case JAVA_NEW_CLASS_EXPR:
+    case CALL_EXPR:
+      /* Complete function's argument first */
+      if (complete_function_arguments (node))
+       return error_mark_node;
+      else
+       return patch_method_invocation_stmt (node, NULL_TREE, NULL_TREE, NULL);
+
+    case MODIFY_EXPR:
+      /* Save potential wfls */
+      wfl_op1 = TREE_OPERAND (node, 0);
+      wfl_op2 = TREE_OPERAND (node, 1);
+      TREE_OPERAND (node, 0) = java_complete_tree (wfl_op1);
+      if (TREE_OPERAND (node, 0) == error_mark_node)
+       return error_mark_node;
+
+      if (COMPOUND_ASSIGN_P (wfl_op2))
+       {
+         tree lvalue;
+         tree other = 
+           java_complete_tree (TREE_OPERAND (wfl_op2, 0));
+
+         /* Hand stablize the lhs on both places */
+         lvalue = stabilize_reference (other); 
+         TREE_OPERAND (node, 0) = lvalue;
+         TREE_OPERAND (TREE_OPERAND (node, 1), 0) = lvalue;
+       }
+
+      /* There are cases where the type of RHS is fixed. In those
+        cases, if the evaluation of the RHS fails, we further the
+        evaluation of the assignment to detect more errors. */
+      nn = java_complete_tree (TREE_OPERAND (node, 1));
+      if (nn == error_mark_node)
+       {
+         /* It's hopeless, but we can further things on to discover
+            an error during the assignment. In any cases, the
+            assignment operation fails. */
+         if (TREE_CODE (TREE_OPERAND (node, 1)) != EXPR_WITH_FILE_LOCATION
+             && TREE_TYPE (TREE_OPERAND (node, 1)) != error_mark_node)
+           patch_assignment (node, wfl_op1, wfl_op2);
+
+         /* Now, we still mark the lhs as initialized */
+         if (DECL_P (TREE_OPERAND (node, 0)))
+           INITIALIZED_P (TREE_OPERAND (node, 0)) = 1;
+
+         return error_mark_node;
+       }
+      TREE_OPERAND (node, 1) = nn;
+      return patch_assignment (node, wfl_op1, wfl_op2);
+
+    case MULT_EXPR:
+    case PLUS_EXPR:
+    case MINUS_EXPR:
+    case LSHIFT_EXPR:
+    case RSHIFT_EXPR:
+    case URSHIFT_EXPR:
+    case BIT_AND_EXPR:
+    case BIT_XOR_EXPR:
+    case BIT_IOR_EXPR:
+    case TRUNC_MOD_EXPR:
+    case RDIV_EXPR:
+    case TRUTH_ANDIF_EXPR:
+    case TRUTH_ORIF_EXPR:
+    case EQ_EXPR: 
+    case NE_EXPR:
+    case GT_EXPR:
+    case GE_EXPR:
+    case LT_EXPR:
+    case LE_EXPR:
+      /* Operands 0 and 1 are WFL in certain cases only. patch_binop
+        knows how to handle those cases. */
+      wfl_op1 = TREE_OPERAND (node, 0);
+      wfl_op2 = TREE_OPERAND (node, 1);
+      TREE_OPERAND (node, 0) = java_complete_tree (wfl_op1);
+      if (TREE_OPERAND (node, 0) == error_mark_node)
+        return error_mark_node;
+      TREE_OPERAND (node, 1) = java_complete_tree (wfl_op2);
+      if (TREE_OPERAND (node, 1) == error_mark_node)
+       return error_mark_node;
+      return patch_binop (node, wfl_op1, wfl_op2);
+
+    case JAVA_UNARY_PLUS_EXPR:
+    case NEGATE_EXPR:
+    case TRUTH_NOT_EXPR:
+    case BIT_NOT_EXPR:
+    case PREDECREMENT_EXPR:
+    case PREINCREMENT_EXPR:
+    case POSTDECREMENT_EXPR:
+    case POSTINCREMENT_EXPR:
+    case CONVERT_EXPR:
+      /* There are cases were wfl_op1 is a WFL. patch_unaryop knows
+        how to handle those cases. */
+      wfl_op1 = TREE_OPERAND (node, 0);
+      TREE_OPERAND (node, 0) = java_complete_tree (wfl_op1);
+      if (TREE_OPERAND (node, 0) == error_mark_node)
+       return error_mark_node;
+      return patch_unaryop (node, wfl_op1);
+
+    case ARRAY_REF:
+      /* There are cases were wfl_op1 is a WFL. patch_array_ref knows
+        how to handle those cases. */
+      wfl_op1 = TREE_OPERAND (node, 0);
+      TREE_OPERAND (node, 0) = java_complete_tree (wfl_op1);
+      if (TREE_OPERAND (node, 0) == error_mark_node)
+       return error_mark_node;
+      /* The same applies to wfl_op2 */
+      wfl_op2 = TREE_OPERAND (node, 1);
+      TREE_OPERAND (node, 1) = java_complete_tree (wfl_op2);
+      if (TREE_OPERAND (node, 1) == error_mark_node)
+       return error_mark_node;
+      return patch_array_ref (node, wfl_op1, wfl_op2);
+
+    case JAVA_THIS_EXPR:
+      /* Can't use THIS in a static environment */
+      if (!current_this)
+       {
+         EXPR_WFL_LINECOL (wfl_operator) = EXPR_WFL_LINECOL (node);
+         parse_error_context (wfl_operator, "Keyword `this' used outside "
+                              "allowed context");
+         TREE_TYPE (node) = error_mark_node;
+         return error_mark_node;
+       }
+      return current_this;
+
+    case STRING_CST:
+      /* Build the internal string representation */
+      push_obstacks (&permanent_obstack, &permanent_obstack);
+      node = get_identifier (TREE_STRING_POINTER (node));
+      location = alloc_name_constant (CONSTANT_String, node);
+      node = build_ref_from_constant_pool (location);
+      TREE_TYPE (node) = promote_type (string_type_node);
+      return node;
+
+    default:
+      fatal ("No case for tree code `%s' - java_complete_tree\n",
+            tree_code_name [TREE_CODE (node)]);
+    }
+  return node;
+}
+
+/* Complete function call's argument. Return a non zero value is an
+   error was found.  */
+
+static int
+complete_function_arguments (node)
+     tree node;
+{
+  int flag = 0;
+  tree cn;
+
+  for (cn = TREE_OPERAND (node, 1); cn; cn = TREE_CHAIN (cn))
+    {
+      tree wfl = TREE_VALUE (cn), parm;
+      parm = java_complete_tree (wfl);
+      if (parm == error_mark_node)
+       {
+         flag = 1;
+         continue;
+       }
+      if (TREE_CODE (TREE_TYPE (parm)) == RECORD_TYPE)
+       TREE_VALUE (cn) = convert (promote_type (TREE_TYPE (parm)), parm);
+      else
+       TREE_VALUE (cn) = save_expr (parm);
+      if (not_initialized_as_it_should_p (parm))
+       {
+         ERROR_VARIABLE_NOT_INITIALIZED (wfl, EXPR_WFL_NODE (wfl));
+         INITIALIZED_P (parm) = 1;
+       }
+    }
+  return flag;
+}
+
+/* Sometimes (for loops and variable initialized during their
+   declaration), we want to wrap a statement around a WFL and turn it
+   debugable.  */
+
+static tree
+build_debugable_stmt (location, stmt)
+    int location;
+    tree stmt;
+{
+  if (TREE_CODE (stmt) != EXPR_WITH_FILE_LOCATION)
+    {
+      stmt = build_expr_wfl (stmt, input_filename, 0, 0);
+      EXPR_WFL_LINECOL (stmt) = location;
+    }
+  JAVA_MAYBE_GENERATE_DEBUG_INFO (stmt);
+  return stmt;
+}
+
+static tree
+build_expr_block (body, decls)
+     tree body, decls;
+{
+  tree node = make_node (BLOCK);
+  BLOCK_EXPR_DECLS (node) = decls;
+  BLOCK_EXPR_BODY (body);
+  if (body)
+    TREE_TYPE (node) = TREE_TYPE (body);
+  TREE_SIDE_EFFECTS (node) = 1;
+  return node;
+}
+
+/* Create a new function block and link its supercontext to the
+   previous block. The current function DECL is used as supercontext
+   when enter_block is called for the first time for a given
+   function. The current function body (DECL_FUNCTION_BODY) is set to
+   the newly created block.  */
+
+static block_level = 0;
+
+static tree
+enter_block ()
+{
+  tree b = build_expr_block (NULL_TREE, NULL_TREE);
+  tree fndecl = current_function_decl; 
+
+  if (!DECL_FUNCTION_BODY (fndecl))
+    {
+      BLOCK_SUPERCONTEXT (b) = fndecl;
+      DECL_FUNCTION_BODY (fndecl) = b;
+    }
+  else
+    {
+      BLOCK_SUPERCONTEXT (b) = DECL_FUNCTION_BODY (fndecl);
+      DECL_FUNCTION_BODY (fndecl) = b;
+    }
+  return b;
+}
+
+/* Exit a block by changing the current function body
+   (DECL_FUNCTION_BODY) to the current block super context, only if
+   the block being exited isn't the method's top level one.  */
+
+static tree
+exit_block ()
+{
+  tree b = DECL_FUNCTION_BODY (current_function_decl);
+
+  if (BLOCK_SUPERCONTEXT (b) != current_function_decl)
+    DECL_FUNCTION_BODY (current_function_decl) = BLOCK_SUPERCONTEXT (b);
+
+  return b;
+}
+
+/* Lookup for NAME in the nested function's blocks, all the way up to
+   the current toplevel one. It complies with Java's local variable
+   scoping rules.  */
+
+static tree
+lookup_name_in_blocks (name)
+     tree name;
+{
+  tree b = DECL_FUNCTION_BODY (current_function_decl);
+
+  while (b != current_function_decl)
+    {
+      tree current;
+
+      /* Paranoid sanity check. To be removed */
+      if (TREE_CODE (b) != BLOCK)
+       fatal ("non block expr function body - lookup_name_in_blocks");
+
+      for (current = BLOCK_EXPR_DECLS (b); current; 
+          current = TREE_CHAIN (current))
+       if (DECL_NAME (current) == name)
+         return current;
+      b = BLOCK_SUPERCONTEXT (b);
+    }
+  return NULL_TREE;
+}
+
+static void
+maybe_absorb_scoping_blocks ()
+{
+  while (BLOCK_EXPR_ORIGIN (DECL_FUNCTION_BODY (current_function_decl)))
+    {
+      tree b = exit_block ();
+      java_method_add_stmt (current_function_decl, b);
+      SOURCE_FRONTEND_DEBUG (("Absorbing scoping block at line %d", lineno));
+    }
+}
+
+\f
+/* This section of the source is reserved to build_* functions that
+   are building incomplete tree nodes and the patch_* functions that
+   are completing them.  */
+
+/* Build an incomplete CALL_EXPR node. Encapsulate it within a WFL */
+
+static tree
+build_method_invocation (name, args)
+    tree name;
+    tree args;
+{
+  tree call = build (CALL_EXPR, NULL_TREE, name, args, NULL_TREE);
+  TREE_SIDE_EFFECTS (call) = 1;
+  /* Check on cases where NAME isn't a WFL. FIXME */
+  EXPR_WFL_LINECOL (call) = EXPR_WFL_LINECOL (name);
+  return call;
+}
+
+/* Build an incomplete assignment expression. */
+
+static tree
+build_assignment (op, op_location, lhs, rhs)
+     int op, op_location;
+     tree lhs, rhs;
+{
+  tree assignment;
+  /* Build the corresponding binop if we deal with a Compound
+     Assignment operator. Mark the binop sub-tree as part of a
+     Compound Assignment expression */
+  if (op != ASSIGN_TK)
+    {
+      rhs = build_binop (BINOP_LOOKUP (op), op_location, lhs, rhs);
+      COMPOUND_ASSIGN_P (rhs) = 1;
+    }
+  assignment = build (MODIFY_EXPR, NULL_TREE, lhs, rhs);
+  TREE_SIDE_EFFECTS (assignment) = 1;
+  EXPR_WFL_LINECOL (assignment) = op_location;
+  return assignment;
+}
+
+/* Print an INTEGER_CST node in a static buffer, and return the buffer. */
+
+static char *
+print_int_node (node)
+    tree node;
+{
+  static char buffer [80];
+  if (TREE_CONSTANT_OVERFLOW (node))
+    sprintf (buffer, "<overflow>");
+    
+  if (TREE_INT_CST_HIGH (node) == 0)
+    sprintf (buffer, HOST_WIDE_INT_PRINT_UNSIGNED,
+            TREE_INT_CST_LOW (node));
+  else if (TREE_INT_CST_HIGH (node) == -1
+          && TREE_INT_CST_LOW (node) != 0)
+    {
+      buffer [0] = '-';
+      sprintf (&buffer [1], HOST_WIDE_INT_PRINT_UNSIGNED,
+              -TREE_INT_CST_LOW (node));
+    }
+  else
+    sprintf (buffer, HOST_WIDE_INT_PRINT_DOUBLE_HEX,
+            TREE_INT_CST_HIGH (node), TREE_INT_CST_LOW (node));
+
+  return buffer;
+}
+
+/* 15.25 Assignment operators. */
+
+static tree
+patch_assignment (node, wfl_op1, wfl_op2)
+     tree node;
+     tree wfl_op1;
+     tree wfl_op2;
+{
+  tree rhs = TREE_OPERAND (node, 1);
+  tree lvalue = TREE_OPERAND (node, 0);
+  tree lhs_type, rhs_type, new_rhs = NULL_TREE;
+  int all_primitive;
+  int error_found = 0;
+  int lvalue_from_array = 0;
+
+  /* Can't assign to a final. */
+  if (DECL_P (lvalue) && FIELD_FINAL (lvalue))
+    {
+      parse_error_context 
+        (wfl_op1, "Can't assign a value to the final variable `%s'",
+        IDENTIFIER_POINTER (EXPR_WFL_NODE (wfl_op1)));
+      error_found = 1;
+    }
+
+  EXPR_WFL_LINECOL (wfl_operator) = EXPR_WFL_LINECOL (node);
+
+  /* Lhs can be a named variable */
+  if (DECL_P (lvalue))
+    {
+      INITIALIZED_P (lvalue) = 1;
+      lhs_type = TREE_TYPE (lvalue);
+    }
+  /* Or Lhs can be a array acccess. Should that be lvalue ? FIXME +
+     comment on reason why */
+  else if (TREE_CODE (wfl_op1) == ARRAY_REF)
+    {
+      lhs_type = TREE_TYPE (lvalue);
+      lvalue_from_array = 1;
+    }
+  /* Or a field access */
+  else if (TREE_CODE (lvalue) == COMPONENT_REF)
+    lhs_type = TREE_TYPE (lvalue);
+  /* Or a function return slot */
+  else if (TREE_CODE (lvalue) == RESULT_DECL)
+    lhs_type = TREE_TYPE (lvalue);
+  /* Otherwise, this is an error */
+  else
+    {
+      parse_error_context (wfl_op1, "Invalid left hand side of assignment");
+      error_found = 1;
+    }
+
+  rhs_type = TREE_TYPE (rhs);
+
+  /* 5.2 Begin Assignment conversion */
+
+  /* 5.1.1 Try Identity Conversion */
+  if (lhs_type == rhs_type) 
+    new_rhs = rhs;
+  
+  /* 5.1.2 Try Widening Primitive Conversion */
+  all_primitive = JPRIMITIVE_TYPE_P (lhs_type) && JPRIMITIVE_TYPE_P (rhs_type);
+  if (all_primitive && JINTEGRAL_TYPE_P (rhs_type)
+      && ((TYPE_PRECISION (rhs_type) < TYPE_PRECISION (lhs_type))
+         || (JFLOAT_TYPE_P (lhs_type) &&
+             TYPE_PRECISION (rhs_type) == TYPE_PRECISION (lhs_type))))
+    new_rhs = convert (lhs_type, rhs);
+  else if (all_primitive && JFLOAT_TYPE_P (rhs_type)
+          && (TYPE_PRECISION (rhs_type) < TYPE_PRECISION (lhs_type)))
+    new_rhs = convert (lhs_type, rhs);
+
+  /* Try a narrowing primitive conversion: 
+       - expression is a constant expression of type int AND
+       - variable is byte, short or char AND
+       - The value of the expression is representable in the type of the 
+         variable */
+  else if (rhs_type == int_type_node && TREE_CONSTANT (rhs)
+          && (lhs_type == byte_type_node || lhs_type == char_type_node
+              || lhs_type == short_type_node))
+    {
+      if (int_fits_type_p (rhs, lhs_type))
+        new_rhs = convert (lhs_type, rhs);
+      else
+       parse_warning_context 
+         (wfl_op1, "Constant expression `%s' to wide for narrowing "
+          "primitive conversion to `%s'", 
+          print_int_node (rhs), lang_printable_name (lhs_type));
+      /* Reported a warning that will turn into an error further
+        down, so we don't return */
+    }
+
+  /* 5.2 Try a reference conversion */
+  else if (!JPRIMITIVE_TYPE_P (rhs_type) && JREFERENCE_TYPE_P (lhs_type))
+    {
+      /* `null' may be assigned to any reference type */
+      if (rhs == null_pointer_node)
+        new_rhs = null_pointer_node;
+      /* Try the reference assignment conversion */
+      else if (valid_ref_assignconv_cast_p (rhs_type, lhs_type, 0))
+       new_rhs = rhs;
+      if (new_rhs)
+       lhs_type = promote_type (rhs_type);
+    }
+
+  /* 15.25.2 If we have a compound assignment, convert RHS into the
+     type of the LHS */
+  else if (COMPOUND_ASSIGN_P (TREE_OPERAND (node, 1)))
+    new_rhs = convert (lhs_type, rhs);
+
+  /* Explicit cast required. This is an error */
+  if (!new_rhs)
+    {
+      char *t1 = strdup ((char *)lang_printable_name (TREE_TYPE (rhs)));
+      char *t2 = strdup ((char *)lang_printable_name (lhs_type));
+      tree wfl;
+      char operation [32];     /* Max size known */
+
+      /* If the assignment is part of a declaration, we use the WFL of
+        the declared variable to point out the error and call it a
+        declaration problem. If the assignment is a genuine =
+        operator, we call is a operator `=' problem, otherwise we
+        call it an assignment problem. In both of these last cases,
+        we use the WFL of the operator to indicate the error. */
+
+      if (MODIFY_EXPR_FROM_INITIALIZATION_P (node))
+       {
+         wfl = wfl_op1;
+         strcpy (operation, "declaration");
+       }
+      else
+       {
+         wfl = wfl_operator;
+         if (COMPOUND_ASSIGN_P (TREE_OPERAND (node, 1)))
+           strcpy (operation, "assignment");
+         else if (TREE_CODE (TREE_OPERAND (node, 0)) == RESULT_DECL)
+           strcpy (operation, "`return'");
+         else
+           strcpy (operation, "`='");
+       }
+
+      parse_error_context 
+       (wfl, (!can_cast_to_p (rhs_type, lhs_type) ?
+              "Incompatible type for %s. Can't convert `%s' to `%s'" :
+              "Incompatible type for %s. Explicit cast "
+              "needed to convert `%s' to `%s'"), operation, t1, t2);
+      free (t1); free (t2);
+      error_found = 1;
+    }
+
+  /* Before reporting type incompatibility errors, check that the rhs
+     is initialized, if a variable */
+  if (not_initialized_as_it_should_p (rhs))
+    {
+      ERROR_VARIABLE_NOT_INITIALIZED (wfl_op2, DECL_NAME (rhs));
+      INITIALIZED_P (rhs) = 1;
+    }
+
+  if (error_found)
+    return error_mark_node;
+
+  /* If we built a compound expression as the result of a reference
+     assignment into an array element, return it here. */
+  if (TREE_CODE (node) == COMPOUND_EXPR)
+    return node;
+      
+  TREE_OPERAND (node, 0) = lvalue;
+  TREE_OPERAND (node, 1) = new_rhs;
+  TREE_TYPE (node) = lhs_type;
+  return node;
+}
+
+/* Check that SOURCE can be converted into DEST, at least with a
+   cast. If the convertion can't occur at all, return 0 otherwise
+   1. This function is used to produce accurate error messages on the
+   reasons why an assignment failed. */
+
+static int
+can_cast_to_p (source, dest)
+     tree source;
+     tree dest;
+{
+  if (TREE_CODE (source) == POINTER_TYPE)
+    source = TREE_TYPE (source);
+  if (TREE_CODE (dest) == POINTER_TYPE)
+    dest = TREE_TYPE (dest);
+
+  if (TREE_CODE (source) == RECORD_TYPE && TREE_CODE (dest) == RECORD_TYPE)
+    return valid_ref_assignconv_cast_p (source, dest, 1);
+
+  else if (JNUMERIC_TYPE_P (source) && JNUMERIC_TYPE_P (dest))
+    return 1;
+
+  return 0;
+}
+
+/* Check that something of SOURCE type can be assigned or cast to
+   something of DEST type at runtime. Return 1 if the operation is
+   valid, 0 otherwise. If CAST is set to 1, we're treating the case
+   were SOURCE is cast into DEST, which borrows a lot of the
+   assignment check. */
+
+static int
+valid_ref_assignconv_cast_p (source, dest, cast)
+     tree source;
+     tree dest;
+     int cast;
+{
+  if (TREE_CODE (source) == POINTER_TYPE)
+    source = TREE_TYPE (source);
+  if (TREE_CODE (dest) == POINTER_TYPE)
+    dest = TREE_TYPE (dest);
+  /* Case where SOURCE is a class type */
+  if (TYPE_CLASS_P (source))
+    {
+      if (TYPE_CLASS_P (dest))
+       return  source == dest || inherits_from_p (source, dest)
+         || cast && inherits_from_p (dest, source);
+      if (TYPE_INTERFACE_P (dest))
+       {
+         /* If doing a cast and SOURCE is final, the operation is
+             always correct a compile time (because even if SOURCE
+             does not implement DEST, a subclass of SOURCE might). */
+         if (cast && !CLASS_FINAL (TYPE_NAME (source)))
+           return 1;
+         /* Otherwise, SOURCE must implement DEST */
+         return interface_of_p (dest, source);
+       }
+      /* DEST is an array, cast permited if SOURCE is of Object type */
+      return (cast && source == object_type_node ? 1 : 0);
+    }
+  if (TYPE_INTERFACE_P (source))
+    {
+      if (TYPE_CLASS_P (dest))
+       {
+         /* If not casting, DEST must be the Object type */
+         if (!cast)
+           return dest == object_type_node;
+         /* We're doing a cast. The cast is always valid is class
+            DEST is not final, otherwise, DEST must implement SOURCE */
+         else if (!CLASS_FINAL (TYPE_NAME (source)))
+           return 1;
+         else
+           return interface_of_p (source, dest);
+       }
+      if (TYPE_INTERFACE_P (dest))
+       {
+         /* If doing a cast, then if SOURCE and DEST contain method
+             with the same signature but different return type, then
+             this is a (compile time) error */
+         if (cast)
+           {
+             tree method_source, method_dest;
+             tree source_type;
+             tree source_sig, dest_sig;
+             tree source_name;
+             for (method_source = TYPE_METHODS (source); method_source; 
+                  method_source = TREE_CHAIN (method_source))
+               {
+                 source_sig = 
+                   build_java_argument_signature (TREE_TYPE (method_source));
+                 source_type = TREE_TYPE (TREE_TYPE (method_source));
+                 source_name = DECL_NAME (method_source);
+                 for (method_dest = TYPE_METHODS (dest);
+                      method_dest; method_dest = TREE_CHAIN (method_dest))
+                   if (source_sig == 
+                       build_java_argument_signature (TREE_TYPE (method_dest))
+                       && source_name == DECL_NAME (method_dest)
+                       && source_type != TREE_TYPE (TREE_TYPE (method_dest)))
+                     return 0;
+               }
+             return 1;
+           }
+         else
+           return source == dest || interface_of_p (dest, source);
+       }
+      else                     /* Array */
+       return 0;
+    }
+  if (TYPE_ARRAY_P (source))
+    {
+      if (TYPE_CLASS_P (dest))
+       return dest == object_type_node;
+      if (TYPE_INTERFACE_P (dest))
+       return 0;               /* Install test on Clonable. FIXME */
+      else                     /* Arrays */
+       {
+         tree source_element_type = TYPE_ARRAY_ELEMENT (source);
+         tree dest_element_type = TYPE_ARRAY_ELEMENT (dest);
+         
+         if (source_element_type == dest_element_type)
+           return 1;
+         return valid_ref_assignconv_cast_p (source_element_type,
+                                             dest_element_type, cast);
+       }
+      return 0;
+    }
+  return 0;
+}
+
+/* Build an incomplete binop expression. */
+
+static tree
+build_binop (op, op_location, op1, op2)
+     enum tree_code op;
+     int op_location;
+     tree op1, op2;
+{
+  tree wfl;
+
+  /* URSHIFT_EXPR is not part of what GCC understands, we can't directly build
+     a node with it */
+  tree binop = 
+    build ((op == URSHIFT_EXPR ? RSHIFT_EXPR : op), NULL_TREE, op1, op2);
+  if (op == URSHIFT_EXPR)
+    TREE_SET_CODE (binop, op);
+
+  TREE_SIDE_EFFECTS (binop) = 1;
+  /* Store the location of the operator, for better error report. The
+     string of the operator will be rebuild based on the OP value. */
+  EXPR_WFL_LINECOL (binop) = op_location;
+  return binop;
+}
+
+/* Build the string of the operator retained by NODE. If NODE is part
+   of a compound expression, add an '=' at the end of the string. This
+   function is called when an error needs to be reported on an
+   operator. The string is returned as a pointer to a static character
+   buffer. */
+
+static char *
+operator_string (node)
+     tree node;
+{
+#define BUILD_OPERATOR_STRING(S)                                       \
+  {                                                                    \
+    sprintf (buffer, "%s%s", S, (COMPOUND_ASSIGN_P (node) ? "=" : ""));        \
+    return buffer;                                                     \
+  }
+  
+  static char buffer [10];
+  switch (TREE_CODE (node))
+    {
+    case MULT_EXPR: BUILD_OPERATOR_STRING ("*");
+    case RDIV_EXPR: BUILD_OPERATOR_STRING ("/");
+    case TRUNC_MOD_EXPR: BUILD_OPERATOR_STRING ("%");
+    case PLUS_EXPR: BUILD_OPERATOR_STRING ("+");
+    case MINUS_EXPR: BUILD_OPERATOR_STRING ("-");
+    case LSHIFT_EXPR: BUILD_OPERATOR_STRING ("<<");
+    case RSHIFT_EXPR: BUILD_OPERATOR_STRING (">>");
+    case URSHIFT_EXPR: BUILD_OPERATOR_STRING (">>>");
+    case BIT_AND_EXPR: BUILD_OPERATOR_STRING ("&");
+    case BIT_XOR_EXPR: BUILD_OPERATOR_STRING ("^");
+    case BIT_IOR_EXPR: BUILD_OPERATOR_STRING ("|");
+    case TRUTH_ANDIF_EXPR: BUILD_OPERATOR_STRING ("&&");
+    case TRUTH_ORIF_EXPR: BUILD_OPERATOR_STRING ("||");
+    case EQ_EXPR: BUILD_OPERATOR_STRING ("==");
+    case NE_EXPR: BUILD_OPERATOR_STRING ("!=");
+    case GT_EXPR: BUILD_OPERATOR_STRING (">");
+    case GE_EXPR: BUILD_OPERATOR_STRING (">=");
+    case LT_EXPR: BUILD_OPERATOR_STRING ("<");
+    case LE_EXPR: BUILD_OPERATOR_STRING ("<=");
+    case JAVA_UNARY_PLUS_EXPR: BUILD_OPERATOR_STRING ("+");
+    case NEGATE_EXPR: BUILD_OPERATOR_STRING ("-");
+    case TRUTH_NOT_EXPR: BUILD_OPERATOR_STRING ("!");
+    case BIT_NOT_EXPR: BUILD_OPERATOR_STRING ("~");
+    case PREINCREMENT_EXPR:    /* Fall through */
+    case POSTINCREMENT_EXPR: BUILD_OPERATOR_STRING ("++");
+    case PREDECREMENT_EXPR:    /* Fall through */
+    case POSTDECREMENT_EXPR: BUILD_OPERATOR_STRING ("--");
+    default:
+      fatal ("unregistered operator %s - operator_string",
+            tree_code_name [TREE_CODE (node)]);
+    }
+  return NULL;
+#undef BUILD_OPERATOR_STRING
+}
+
+/* Binary operators (15.16 up to 15.18). We return error_mark_node on
+   errors but we modify NODE so that it contains the type computed
+   according to the expression, when it's fixed. Otherwise, we write
+   error_mark_node as the type. It allows us to further the analysis
+   of remaining nodes and detects more errors in certain cases.  */
+
+static tree
+patch_binop (node, wfl_op1, wfl_op2)
+     tree node;
+     tree wfl_op1;
+     tree wfl_op2;
+{
+  tree op1 = TREE_OPERAND (node, 0);
+  tree op2 = TREE_OPERAND (node, 1);
+  tree op1_type = TREE_TYPE (op1);
+  tree op2_type = TREE_TYPE (op2);
+  tree prom_type;
+  int code = TREE_CODE (node);
+  /* If 1, tell the routine that we have to return error_mark_node
+     after checking for the initialization of the RHS */
+  int error_found = 0;
+
+  /* Figure what is going to be checked first for initialization prior
+     its use. If NODE is part of a compound assignment, we check the
+     second operand first, otherwise the first one first. We also
+     initialize the matching WFL for the error report. `cfi' stands
+     for Check For Initialization */
+  tree cfi = (COMPOUND_ASSIGN_P (node) ? op2 : op1);
+  tree cfi_wfl = (COMPOUND_ASSIGN_P (node) ? wfl_op2 : wfl_op1);
+
+  EXPR_WFL_LINECOL (wfl_operator) = EXPR_WFL_LINECOL (node);
+
+  /* Check initialization of LHS first. We then silence further error
+     message if the variable wasn't initialized */
+  if (not_initialized_as_it_should_p (cfi))
+    {
+      ERROR_VARIABLE_NOT_INITIALIZED (cfi_wfl, DECL_NAME (cfi));
+      INITIALIZED_P (op1) = 1;
+    }
+
+  switch (code)
+    {
+    /* 15.16 Multiplicative operators */
+    case MULT_EXPR:            /* 15.16.1 Multiplication Operator * */
+    case RDIV_EXPR:            /* 15.16.2 Division Operator / */
+    case TRUNC_MOD_EXPR:       /* 15.16.3 Remainder operator % */
+      if (!JPRIMITIVE_TYPE_P (op1_type) || !JPRIMITIVE_TYPE_P (op2_type))
+       {
+         if (!JPRIMITIVE_TYPE_P (op1_type))
+           ERROR_CANT_CONVERT_TO_NUMERIC (wfl_operator, node, op1_type);
+         if (!JPRIMITIVE_TYPE_P (op2_type) && (op1_type != op2_type))
+           ERROR_CANT_CONVERT_TO_NUMERIC (wfl_operator, node, op2_type);
+         TREE_TYPE (node) = error_mark_node;
+         error_found = 1;
+         break;
+       }
+      prom_type = binary_numeric_promotion (op1_type, op2_type, &op1, &op2);
+      /* Change the division operator if necessary */
+      if (code == RDIV_EXPR && TREE_CODE (prom_type) == INTEGER_TYPE)
+       TREE_SET_CODE (node, TRUNC_DIV_EXPR);
+      /* This one is more complicated. FLOATs are processed by a function
+        call to soft_fmod. */
+      if (code == TRUNC_MOD_EXPR)
+       return build_java_binop (TRUNC_MOD_EXPR, prom_type, op1, op2);
+      break;
+
+    /* 15.17 Additive Operators */
+    case PLUS_EXPR:            /* 15.17.1 String Concatenation Operator + */
+      if (JSTRING_TYPE_P (op1_type) || JSTRING_TYPE_P (op2_type))
+       fatal ("operator `+' non implemented on String - patch_binop");
+    case MINUS_EXPR:           /* 15.17.2 Additive Operators (+ and -) for
+                                  Numeric Types */
+      if (!JPRIMITIVE_TYPE_P (op1_type) || !JPRIMITIVE_TYPE_P (op2_type))
+       {
+         if (!JPRIMITIVE_TYPE_P (op1_type))
+           ERROR_CANT_CONVERT_TO_NUMERIC (wfl_operator, node, op1_type);
+         if (!JPRIMITIVE_TYPE_P (op2_type) && (op1_type != op2_type))
+           ERROR_CANT_CONVERT_TO_NUMERIC (wfl_operator, node, op2_type);
+         TREE_TYPE (node) = error_mark_node;
+         error_found = 1;
+         break;
+       }
+      prom_type = binary_numeric_promotion (op1_type, op2_type, &op1, &op2);
+      break;
+
+    /* 15.18 Shift Operators */
+    case LSHIFT_EXPR:
+    case RSHIFT_EXPR:
+    case URSHIFT_EXPR:
+      if (!JINTEGRAL_TYPE_P (op1_type) || !JINTEGRAL_TYPE_P (op2_type))
+       {
+         if (!JINTEGRAL_TYPE_P (op1_type))
+           ERROR_CAST_NEEDED_TO_INTEGRAL (wfl_operator, node, op1_type);
+         else
+           parse_error_context 
+             (wfl_operator, (JPRIMITIVE_TYPE_P (op2_type) ? 
+              "Incompatible type for `%s'. Explicit cast needed to convert "
+              "shift distance from `%s' to integral" : 
+              "Incompatible type for `%s'. Can't convert shift distance from "
+              "`%s' to integral"), 
+              operator_string (node), lang_printable_name (op2_type));
+         TREE_TYPE (node) = error_mark_node;
+         error_found = 1;
+         break;
+       }
+
+      /* Unary numeric promotion (5.6.1) is performed on each operand
+         separatly */
+      op1 = convert (promote_type (op1_type), op1);
+      op2 = convert (promote_type (op2_type), op2);
+
+      /* The type of the shift expression is the type of the promoted
+         type of the left-hand operand */
+      prom_type = TREE_TYPE (op1);
+
+      /* Shift int only up to 0x1f and long up to 0x3f */
+      if (prom_type == int_type_node)
+       op2 = fold (build (BIT_AND_EXPR, int_type_node, op2, 
+                          build_int_2 (0x1f, 0)));
+      else
+       op2 = fold (build (BIT_AND_EXPR, int_type_node, op2, 
+                          build_int_2 (0x3f, 0)));
+
+      /* The >>> operator is a >> operating on unsigned quantities */
+      if (code == URSHIFT_EXPR)
+       {
+         op1 = convert (unsigned_type (prom_type), op1);
+         TREE_SET_CODE (node, RSHIFT_EXPR);
+       }
+      break;
+      
+
+      /* 15.21 Bitwise and Logical Operators */
+    case BIT_AND_EXPR:
+    case BIT_XOR_EXPR:
+    case BIT_IOR_EXPR:
+      if (JINTEGRAL_TYPE_P (op1_type) && JINTEGRAL_TYPE_P (op2_type))
+       /* Binary numeric promotion is performed on both operand and the
+          expression retain that type */
+       prom_type = binary_numeric_promotion (op1_type, op2_type, &op1, &op2);
+
+      else if (TREE_CODE (op1_type) == BOOLEAN_TYPE 
+              && TREE_CODE (op1_type) == BOOLEAN_TYPE)
+       /* The type of the bitwise operator expression is BOOLEAN */
+       prom_type = boolean_type_node;
+      else
+       {
+         if (!JINTEGRAL_TYPE_P (op1_type))
+           ERROR_CAST_NEEDED_TO_INTEGRAL (wfl_operator, node, op1_type);
+         if (!JINTEGRAL_TYPE_P (op2_type) && (op1_type != op2_type))
+           ERROR_CAST_NEEDED_TO_INTEGRAL (wfl_operator, node, op2_type);
+         TREE_TYPE (node) = error_mark_node;
+         error_found = 1;
+         /* Insert a break here if adding thing before the switch's
+             break for this case */
+       }
+      break;
+
+      /* 15.22 Conditional-And Operator */
+    case TRUTH_ANDIF_EXPR:
+      /* 15.23 Conditional-Or Operator */
+    case TRUTH_ORIF_EXPR:
+      /* Operands must be of BOOLEAN type */
+      if (TREE_CODE (op1_type) != BOOLEAN_TYPE || 
+         TREE_CODE (op2_type) != BOOLEAN_TYPE)
+       {
+         if (TREE_CODE (op1_type) != BOOLEAN_TYPE)
+           ERROR_CANT_CONVERT_TO_BOOLEAN (wfl_operator, node, op1_type);
+         if (TREE_CODE (op2_type) != BOOLEAN_TYPE && (op1_type != op2_type))
+           ERROR_CANT_CONVERT_TO_BOOLEAN (wfl_operator, node, op2_type);
+         TREE_TYPE (node) = boolean_type_node;
+         error_found = 1;
+         break;
+       }
+      /* The type of the conditional operators is BOOLEAN */
+      prom_type = boolean_type_node;
+      break;
+
+      /* 15.19.1 Numerical Comparison Operators <, <=, >, >= */
+    case LT_EXPR:
+    case GT_EXPR:
+    case LE_EXPR:
+    case GE_EXPR:
+      /* The type of each of the operands must be a primitive numeric
+         type */
+      if (!JNUMERIC_TYPE_P (op1_type) || ! JNUMERIC_TYPE_P (op2_type))
+       {
+         if (!JNUMERIC_TYPE_P (op1_type))
+           ERROR_CANT_CONVERT_TO_NUMERIC (wfl_operator, node, op1_type);
+         if (!JNUMERIC_TYPE_P (op2_type) && (op1_type != op2_type))
+           ERROR_CANT_CONVERT_TO_NUMERIC (wfl_operator, node, op2_type);
+         TREE_TYPE (node) = boolean_type_node;
+         error_found = 1;
+         break;
+       }
+      /* Binary numeric promotion is performed on the operands */
+      binary_numeric_promotion (op1_type, op2_type, &op1, &op2);
+      /* The type of the relation expression is always BOOLEAN */
+      prom_type = boolean_type_node;
+      break;
+
+      /* 15.20 Equality Operator */
+    case EQ_EXPR:
+    case NE_EXPR:
+      /* 15.20.1 Numerical Equality Operators == and != */
+      /* Binary numeric promotion is performed on the operands */
+      if (JPRIMITIVE_TYPE_P (op1_type) && JPRIMITIVE_TYPE_P (op2_type))
+       binary_numeric_promotion (op1_type, op2_type, &op1, &op2);
+      
+      /* 15.20.2 Boolean Equality Operators == and != */
+      else if (TREE_CODE (op1_type) == BOOLEAN_TYPE &&
+         TREE_CODE (op2_type) == BOOLEAN_TYPE)
+       ;                       /* Nothing to do here */
+      
+      /* 15.20.3 Reference Equality Operators == and != */
+      /* Types have to be either references or the null type */
+      else if ((op1 == null_pointer_node || op2 == null_pointer_node 
+               || JREFERENCE_TYPE_P (op1_type) 
+               || JREFERENCE_TYPE_P (op2_type))
+              && ((op1_type == op2_type)
+                  /* The should use a can_cast_to_p() */
+                  ))
+       ;                       /* Nothing to do here */
+         
+      /* Else we have an error figure what can't be converted into
+        what and report the error */
+      else
+       {
+         char *t1;
+         t1 = strdup ((char *)lang_printable_name (op1_type));
+         parse_error_context 
+           (wfl_operator, "Incompatible type for `%s'. Can't convert `%s' "
+            "to `%s'", operator_string (node), t1, 
+            lang_printable_name (op2_type));
+         free (t1);
+         TREE_TYPE (node) = boolean_type_node;
+         error_found = 1;
+         break;
+       }
+      prom_type = boolean_type_node;
+      break;
+    }
+
+  /* Then check the initialization of the RHS. We don't do that if
+     we're dealing with a node that is part of a compound
+     assignment. We then silence further error message if the variable
+     wasn't initialized */
+  if (not_initialized_as_it_should_p (op2) && !COMPOUND_ASSIGN_P (node))
+    {
+      ERROR_VARIABLE_NOT_INITIALIZED (wfl_op2, DECL_NAME (op2));
+      INITIALIZED_P (op2) = 1;
+    }
+
+  if (error_found)
+    return error_mark_node;
+
+  TREE_OPERAND (node, 0) = op1;
+  TREE_OPERAND (node, 1) = op2;
+  TREE_TYPE (node) = prom_type;
+  return fold (node);
+}
+
+/* Build an incomplete unary operator expression. Unary `+' node is
+   build as a CONV_EXPR, even though its tree code is overridden by a
+   JAVA_UNARY_PLUS_EXPR that isn't a tree code, to differentiate it during
+   the walk. */
+
+static tree
+build_unaryop (op_token, op_location, op1)
+     int op_token, op_location;
+     tree op1;
+{
+  enum tree_code op;
+  tree unaryop;
+  switch (op_token)
+    {
+    case PLUS_TK: op = CONVERT_EXPR; break;
+    case MINUS_TK: op = NEGATE_EXPR; break;
+    case NEG_TK: op = TRUTH_NOT_EXPR; break;
+    case NOT_TK: op = BIT_NOT_EXPR; break;
+    default: fatal ("Unknown token `%d' for unary operator - build_unaryop",
+                   op_token);
+    }
+
+  unaryop = build1 (op, NULL_TREE, op1);
+  if (op_token == PLUS_TK)
+    TREE_SET_CODE (unaryop, JAVA_UNARY_PLUS_EXPR);
+
+  TREE_SIDE_EFFECTS (unaryop) = 1;
+  /* Store the location of the operator, for better error report. The
+     string of the operator will be rebuild based on the OP value. */
+  EXPR_WFL_LINECOL (unaryop) = op_location;
+  return unaryop;
+}
+
+/* Special case for the ++/-- operators, since they require an extra
+   argument to build, which is set to NULL and patched
+   later. IS_POST_P is 1 if the operator, 0 otherwise.  */
+
+static tree
+build_incdec (op_token, op_location, op1, is_post_p)
+     int op_token, op_location;
+     tree op1;
+     int is_post_p;
+{
+  static enum tree_code lookup [2][2] = 
+    {
+      { PREDECREMENT_EXPR, PREINCREMENT_EXPR, },
+      { POSTDECREMENT_EXPR, POSTINCREMENT_EXPR, },
+    };
+  tree node = build (lookup [is_post_p][(op_token - DECR_TK)], 
+                    NULL_TREE, op1, NULL_TREE);
+  TREE_SIDE_EFFECTS (node) = 1;
+  /* Store the location of the operator, for better error report. The
+     string of the operator will be rebuild based on the OP value. */
+  EXPR_WFL_LINECOL (node) = op_location;
+  return node;
+}     
+
+/* Build an incomplete cast operator, based on the use of the
+   CONVERT_EXPR. Note that TREE_TYPE of the constructed node is
+   set. java_complete_tree is trained to walk a CONVERT_EXPR even
+   though its type is already set.  */
+
+static tree
+build_cast (location, type, exp)
+     int location;
+     tree type, exp;
+{
+  tree node = build1 (CONVERT_EXPR, type, exp);
+  EXPR_WFL_LINECOL (node) = location;
+  return node;
+}
+
+/* 15.14 Unary operators. We return error_mark_node in case of error,
+   but preserve the type of NODE if the type is fixed.  */
+
+static tree
+patch_unaryop (node, wfl_op)
+     tree node;
+     tree wfl_op;
+{
+  tree op = TREE_OPERAND (node, 0);
+  tree op_type = TREE_TYPE (op);
+  tree prom_type, value;
+  int code = TREE_CODE (node);
+  int error_found = 0;
+
+  EXPR_WFL_LINECOL (wfl_operator) = EXPR_WFL_LINECOL (node);
+
+  switch (code)
+    {
+      /* 15.13.2 Postfix Increment Operator ++ */
+    case POSTINCREMENT_EXPR:
+      /* 15.13.3 Postfix Increment Operator -- */
+    case POSTDECREMENT_EXPR:
+      /* 15.14.1 Prefix Increment Operator ++ */
+    case PREINCREMENT_EXPR:
+      /* 15.14.2 Prefix Decrement Operator -- */
+    case PREDECREMENT_EXPR:
+      if (!DECL_P (op))
+       {
+         parse_error_context (wfl_operator, "Invalid argument to `%s'",
+                              operator_string (node));
+         TREE_TYPE (node) = error_mark_node;
+         error_found = 1;
+       }
+      else if (FIELD_FINAL (op))
+       {
+         parse_error_context 
+           (wfl_op, "Can't assign a value to the final variable `%s'",
+            IDENTIFIER_POINTER (EXPR_WFL_NODE (wfl_op)));
+         TREE_TYPE (node) = error_mark_node;
+         error_found = 1;
+       }
+      /* From now on, we know that op if a variable and that it has a
+         valid wfl. We use wfl_op to locate errors related to the
+         ++/-- operand. */
+      else if (!JNUMERIC_TYPE_P (op_type))
+       {
+         parse_error_context
+           (wfl_op, "Invalid argument type `%s' to `%s'",
+            lang_printable_name (op_type), operator_string (node));
+         TREE_TYPE (node) = error_mark_node;
+         error_found = 1;
+       }
+      else
+       {
+         /* Before the addition, binary numeric promotion if performed on
+            both operands */
+         value = integer_one_node;
+         prom_type = binary_numeric_promotion (op_type, TREE_TYPE (value), 
+                                               &op, &value);
+         /* And write the promoted increment back */
+         TREE_OPERAND (node, 1) = value;
+       }
+      break;
+
+      /* 15.14.3 Unary Plus Operator + */
+    case JAVA_UNARY_PLUS_EXPR:
+      /* 15.14.4 Unary Minus Operator - */
+    case NEGATE_EXPR:
+      if (!JNUMERIC_TYPE_P (op_type))
+       {
+         ERROR_CANT_CONVERT_TO_NUMERIC (wfl_operator, node, op_type);
+         TREE_TYPE (node) = error_mark_node;
+         error_found = 1;
+       }
+      /* Unary numeric promotion is performed on operand */
+      else
+       {
+         prom_type = promote_type (op_type);
+         op = convert (prom_type, op);
+         if (code == JAVA_UNARY_PLUS_EXPR)
+           node = op;
+       }
+      break;
+
+      /* 15.14.5 Bitwise Complement Operator ~ */
+    case BIT_NOT_EXPR:
+      if (!JINTEGRAL_TYPE_P (op_type))
+       {
+         ERROR_CAST_NEEDED_TO_INTEGRAL (wfl_operator, node, op_type);
+         TREE_TYPE (node) = error_mark_node;
+         error_found = 1;
+       }
+      else
+       {
+         prom_type = promote_type (op_type);
+         op = convert (prom_type, op);
+       }
+      break;
+
+      /* 15.14.6 Logical Complement Operator ! */
+    case TRUTH_NOT_EXPR:
+      if (TREE_CODE (op_type) != BOOLEAN_TYPE)
+       {
+         ERROR_CANT_CONVERT_TO_BOOLEAN (wfl_operator, node, op_type);
+         TREE_TYPE (node) = boolean_type_node;
+         error_found = 1;
+       }
+      else
+       prom_type = boolean_type_node;
+      break;
+
+      /* 15.15 Cast Expression */
+    case CONVERT_EXPR:
+      value = patch_cast (node, wfl_op, wfl_operator);
+      if (value == error_mark_node)
+       {
+         TREE_TYPE (node) = error_mark_node;
+         error_found = 1;
+       }
+      else
+       node = value;
+      break;
+    }
+  
+  /* Check variable initialization */
+  if (not_initialized_as_it_should_p (op))
+    {
+      ERROR_VARIABLE_NOT_INITIALIZED (wfl_op, DECL_NAME (op));
+      INITIALIZED_P (op) = 1;
+    }
+  
+  if (error_found)
+    return error_mark_node;
+  /* In the case of JAVA_UNARY_PLUS_EXPR, we replaced NODE by a new one */
+  else if (code != JAVA_UNARY_PLUS_EXPR && code != CONVERT_EXPR)
+    {
+      TREE_OPERAND (node, 0) = op;
+      TREE_TYPE (node) = prom_type;
+    }
+  return fold (node);
+}
+
+/* Generic type resolution that sometimes takes place during node
+   patching. Returned the resolved type or generate an error
+   message. Return the resolved type or NULL_TREE.  */
+
+static tree
+resolve_type_during_patch (type)
+     tree type;
+{
+  if (unresolved_type_p (type, NULL))
+    {
+      tree type_decl = resolve_no_layout (EXPR_WFL_NODE (type), NULL_TREE);
+      if (!type_decl)
+       {
+         parse_error_context (type, 
+                              "Class `%s' not found in type declaration",
+                              IDENTIFIER_POINTER (EXPR_WFL_NODE (type)));
+         return NULL_TREE;
+       }
+      else
+       return TREE_TYPE (type_decl);
+    }
+  return type;
+}
+/* 5.5 Casting Conversion. error_mark_node is returned if an error is
+   found. Otherwise NODE or something meant to replace it is returned.  */
+
+static tree
+patch_cast (node, wfl_op, wfl_operator)
+     tree node;
+     tree wfl_op;
+     tree wfl_operator;
+{
+  tree op = TREE_OPERAND (node, 0);
+  tree op_type = TREE_TYPE (op);
+  tree cast_type = TREE_TYPE (node);
+  char *t1;
+
+  /* First resolve OP_TYPE if unresolved */
+  if (!(cast_type = resolve_type_during_patch (cast_type)))
+    return error_mark_node;
+
+  /* Check on cast that are proven correct at compile time */
+  if (JNUMERIC_TYPE_P (cast_type) && JNUMERIC_TYPE_P (op_type))
+    {
+      static tree convert_narrow ();
+      /* Same type */
+      if (cast_type == op_type)
+       return node;
+
+      /* Try widening/narowwing convertion. Potentially, things need
+        to be worked out in gcc so we implement the extreme cases
+        correctly. fold_convert() needs to be fixed. */
+      return convert (cast_type, op);
+    }
+
+  /* The remaining legal casts involve conversion between reference
+     types. Check for their compile time correctness. */
+  if (JREFERENCE_TYPE_P (op_type) && JREFERENCE_TYPE_P (cast_type) 
+      && valid_ref_assignconv_cast_p (op_type, cast_type, 1))
+    {
+      TREE_TYPE (node) = promote_type (cast_type);
+      /* Now, the case can be determined correct at compile time if
+         OP_TYPE can be converted into CAST_TYPE by assignment
+         conversion (5.2) */
+
+      if (valid_ref_assignconv_cast_p (op_type, cast_type, 0))
+       return node;
+
+      /* The cast requires a run-time check */
+      return build (CALL_EXPR, promote_type (cast_type),
+                   build_address_of (soft_checkcast_node),
+                   tree_cons (NULL_TREE, build_class_ref (cast_type),
+                              build_tree_list (NULL_TREE, op)),
+                   NULL_TREE);
+    }
+
+  /* Any other casts are proven incorrect at compile time */
+  t1 = strdup ((char *)lang_printable_name (op_type));
+  parse_error_context (wfl_operator, "Invalid cast from `%s' to `%s'",
+                      t1, lang_printable_name (cast_type));
+  free (t1);
+  return error_mark_node;
+}
+
+/* Build an ARRAY_REF incomplete tree node. Note that operand 1 isn't
+   a list of indices. */
+static tree
+build_array_ref (location, array, index)
+     int location;
+     tree array, index;
+{
+  tree node = build (ARRAY_REF, NULL_TREE, array, index);
+  EXPR_WFL_LINECOL (node) = location;
+  return node;
+}
+
+/* 15.12 Array Access Expression */
+
+static tree
+patch_array_ref (node, wfl_array, wfl_index)
+     tree node, wfl_array, wfl_index;
+{
+  tree array = TREE_OPERAND (node, 0);
+  tree array_type  = TREE_TYPE (array);
+  tree index = TREE_OPERAND (node, 1);
+  tree index_type = TREE_TYPE (index);
+  tree promoted_index_type;
+  int error_found = 0;
+
+  EXPR_WFL_LINECOL (wfl_operator) = EXPR_WFL_LINECOL (node);
+
+  if (not_initialized_as_it_should_p (array))
+    {
+      ERROR_VARIABLE_NOT_INITIALIZED (wfl_array, DECL_NAME (array));
+      INITIALIZED_P (array) = 1;
+    }
+  if (! flag_emit_class_files)
+    array = save_expr (array);
+
+  if (TREE_CODE (array_type) == POINTER_TYPE)
+    array_type = TREE_TYPE (array_type);
+
+  /* The array reference must be an array */
+  if (!TYPE_ARRAY_P (array_type))
+    {
+      parse_error_context 
+       (wfl_operator, "`[]' can only be applied to arrays. It can't be "
+        "applied to `%s'", lang_printable_name (array_type));
+      TREE_TYPE (node) = error_mark_node;
+      error_found = 1;
+    }
+
+  /* The array index underdoes unary numeric promotion. The promoted
+     type must be int */
+  promoted_index_type = promote_type (index_type);
+  if (promoted_index_type != int_type_node)
+    {
+      int could_cast = can_cast_to_p (index_type, int_type_node);
+      parse_error_context 
+       (wfl_operator, 
+        (could_cast ? "Incompatible type for `[]'. Explicit cast needed to "
+         "convert `%s' to `int'" : "Incompatible type for `[]'. "
+         "Can't convert `%s' to `int'"),
+        lang_printable_name (index_type));
+      TREE_TYPE (node) = error_mark_node;
+      error_found = 1;
+    }
+
+  /* Now if the index is a var/parm decl, check on its initialization */
+  if (not_initialized_as_it_should_p (index))
+    {
+      ERROR_VARIABLE_NOT_INITIALIZED (wfl_index, DECL_NAME (index));
+      INITIALIZED_P (index) = 1;
+    }
+
+  if (error_found)
+    return error_mark_node;
+  index = convert (promoted_index_type, index);
+
+  if (TREE_CODE (array_type) == RECORD_TYPE)
+    array_type = promote_type (TYPE_ARRAY_ELEMENT (array_type));
+  if (flag_emit_class_files)
+    {
+      TREE_OPERAND (node, 0)= array;
+      TREE_OPERAND (node, 1)= index;
+    }
+  else
+    node = build_java_arrayaccess (array, array_type, index);
+  TREE_TYPE (node) = array_type;
+  return node;
+}
+
+/* 15.9 Array Creation Expressions */
+
+static tree
+build_newarray_node (type, dims, extra_dims)
+     tree type;
+     tree dims;
+     int extra_dims;
+{
+  tree node =
+    build (CALL_EXPR, NULL_TREE, type, nreverse (dims), 
+          build_int_2 (extra_dims, 0));
+  TREE_SET_CODE (node, JAVA_NEW_ARRAY_EXPR);
+  return node;
+}
+
+static tree
+patch_newarray (node)
+     tree node;
+{
+  tree type = TREE_OPERAND (node, 0);
+  tree dims = TREE_OPERAND (node, 1);
+  tree cdim, array_type;
+  int error_found = 0;
+  int ndims = 0;
+  int xdims = TREE_INT_CST_LOW (TREE_OPERAND (node, 2));
+  int total_dims;
+
+  /* Dimension types are verified. It's better for the types to be
+     verified in order. */
+  for (cdim = dims, ndims = 0; cdim; cdim = TREE_CHAIN (cdim), ndims++ )
+    {
+      int dim_error = 0;
+      tree dim = TREE_VALUE (cdim);
+
+      /* Dim might have been saved during its evaluation */
+      dim = (TREE_CODE (dim) == SAVE_EXPR ? dim = TREE_OPERAND (dim, 0) : dim);
+
+      /* The type of each specified dimension must be an integral type. */
+      if (!JINTEGRAL_TYPE_P (TREE_TYPE (dim)))
+       dim_error = 1;
+
+      /* Each expression undergoes an unary numeric promotion (5.6.1) and the
+        promoted type must be int. */
+      else
+       {
+         dim = convert (promote_type (TREE_TYPE (dim)), dim);
+         if (TREE_TYPE (dim) != int_type_node)
+           dim_error = 1;
+       }
+
+      /* Report errors on types here */
+      if (dim_error)
+       {
+         parse_error_context 
+           (TREE_PURPOSE (cdim), 
+            "Incompatible type for dimension in array creation expression. "
+            "%s convert `%s' to `int'", 
+            (can_cast_to_p (TREE_TYPE (dim), int_type_node) ?
+             "Explicit cast needed to" : "Can't"),
+            lang_printable_name (TREE_TYPE (dim)));
+         error_found = 1;
+       }
+
+      /* Check for uninitialized variables */
+      if (not_initialized_as_it_should_p (dim))
+       {
+         ERROR_VARIABLE_NOT_INITIALIZED (TREE_PURPOSE (cdim), 
+                                         DECL_NAME (dim));
+         INITIALIZED_P (dim) = 1;
+         error_found = 1;
+       }
+
+      TREE_PURPOSE (cdim) = NULL_TREE;
+    }
+
+  /* Resolve array base type if unresolved */
+  if (!(type = resolve_type_during_patch (type)))
+    error_found = 1;
+
+  if (error_found)
+    {
+      /* We don't want further evaluation of this bogus array creation
+         operation */
+      TREE_TYPE (node) = error_mark_node;
+      return error_mark_node;
+    }
+
+  /* The node is transformed into a function call. Things are done
+     differently according to the number of dimensions. If the number
+     of dimension is equal to 1, then the nature of the base type
+     (primitive or not) matters. */
+  total_dims = xdims + ndims;
+  if (total_dims == 1)
+    {
+      if (JPRIMITIVE_TYPE_P (type))
+       {
+         int type_code;
+         if (type == boolean_type_node)
+           type_code = 4;
+         else if (type == char_type_node)
+           type_code = 5;
+         else if (type == float_type_node)
+           type_code = 6;
+         else if (type == double_type_node)
+           type_code = 7;
+         else if (type == byte_type_node)
+           type_code = 8;
+         else if (type == short_type_node)
+           type_code = 9;
+         else if (type == int_type_node)
+           type_code = 10;
+         else if (type == long_type_node)
+           type_code = 11;
+         else
+           fatal ("Can't compute type code - patch_newarray");
+         return build_newarray (type_code, TREE_VALUE (dims));
+       }
+      else
+       return build_anewarray (type, TREE_VALUE (dims));
+    }
+  
+  /* Add extra dimensions as unknown dimensions */
+  while (xdims--)
+    dims = 
+      chainon (dims, build_tree_list (NULL_TREE, integer_negative_one_node));
+  dims = chainon (dims, build_tree_list (NULL_TREE, integer_zero_node));
+
+  /* Can't reuse what's already written in expr.c because it uses the
+     JVM stack representation. Provide a build_multianewarray. FIXME */
+  array_type = type;
+  for (cdim = TREE_CHAIN (dims); cdim; cdim = TREE_CHAIN (cdim))
+    array_type = build_java_array_type (promote_type (array_type), 
+                                       TREE_CODE (cdim) == INTEGER_CST ?
+                                       TREE_INT_CST_LOW (cdim) : -1);
+  return build (CALL_EXPR,
+               promote_type (array_type),
+               build_address_of (soft_multianewarray_node),
+               tree_cons (NULL_TREE, build_class_ref (array_type),
+                          tree_cons (NULL_TREE, 
+                                     build_int_2 (total_dims, 0), dims )),
+               NULL_TREE);
+}
+
+static tree
+build_this (location)
+     int location;
+{
+  tree node = build_wfl_node (this_identifier_node, input_filename, 0, 0);
+  TREE_SET_CODE (node, JAVA_THIS_EXPR);
+  EXPR_WFL_LINECOL (node) = location;
+  return node;
+}
+
+/* 14.15 The return statement. It builds a modify expression that
+   assigns the returned value to the RESULT_DECL that hold the value
+   to be returned. */
+
+static tree
+build_return (location, op)
+     int location;
+     tree op;
+{
+  tree node = build1 (RETURN_EXPR, NULL_TREE, op);
+  EXPR_WFL_LINECOL (node) = location;
+  return node;
+}
+
+static tree
+patch_return (node)
+     tree node;
+{
+  tree return_exp = TREE_OPERAND (node, 0);
+  tree meth = current_function_decl;
+  tree mtype = TREE_TYPE (TREE_TYPE (current_function_decl));
+  tree modify;
+  int error_found = 0;
+
+  TREE_TYPE (node) = error_mark_node;
+  EXPR_WFL_LINECOL (wfl_operator) = EXPR_WFL_LINECOL (node);
+
+  /* It's invalid to have a return value within a function that is
+     declared with the keyword void or that is a constructor */
+  if (return_exp && (mtype == void_type_node || DECL_CONSTRUCTOR_P (meth)))
+    error_found = 1;
+
+  /* It's invalid to have a no return value within a function that
+     isn't declared with the keyword `void' */
+  if (!return_exp && (mtype != void_type_node && !DECL_CONSTRUCTOR_P (meth)))
+    error_found = 2;
+
+  if (error_found)
+    {
+      char *t = strdup ((char *)lang_printable_name (mtype));
+      parse_error_context (wfl_operator, "`return' with%s value from `%s %s'",
+                          (error_found == 1 ? "" : "out"), t,
+                          lang_printable_name (meth));
+      free (t);
+      return error_mark_node;
+    }
+
+  /* If we have a return_exp, build a modify expression and expand it */
+  if (return_exp)
+    {
+      modify = build (MODIFY_EXPR, NULL_TREE, DECL_RESULT (meth), return_exp);
+      EXPR_WFL_LINECOL (modify) = EXPR_WFL_LINECOL (node);
+      modify = java_complete_tree (modify);
+      if (modify != error_mark_node)
+       {
+         TREE_SIDE_EFFECTS (modify) = 1;
+         TREE_OPERAND (node, 0) = modify;
+       }
+      else
+       return error_mark_node;
+    }
+  TREE_TYPE (node) = void_type_node;
+  TREE_SIDE_EFFECTS (node) = 1;
+  return node;
+}
+
+/* 14.8 The if Statement */
+
+static tree
+build_if_else_statement (location, expression, if_body, else_body)
+     int location;
+     tree expression, if_body, else_body;
+{
+  tree node;
+  /* FIXME: make else body be a void node, where this function is
+     called */
+  if (!else_body)
+    else_body = build (COMPOUND_EXPR, void_type_node, NULL_TREE, NULL_TREE);
+  node = build (COND_EXPR, NULL_TREE, expression, if_body, else_body);
+  EXPR_WFL_LINECOL (node) = location;
+  return node;
+}
+
+static tree
+patch_if_else_statement (node)
+     tree node;
+{
+  tree expression = TREE_OPERAND (node, 0);
+
+  TREE_TYPE (node) = error_mark_node;
+  EXPR_WFL_LINECOL (wfl_operator) = EXPR_WFL_LINECOL (node);
+
+  /* The type of expression must be boolean */
+  if (TREE_TYPE (expression) != boolean_type_node)
+    {
+      parse_error_context 
+       (wfl_operator, 
+        "Incompatible type for `if'. Can't convert `%s' to `boolean'", 
+        lang_printable_name (TREE_TYPE (expression)));
+      return error_mark_node;
+    }
+  
+  TREE_TYPE (node) = void_type_node;
+  TREE_SIDE_EFFECTS (node) = 1;
+  return node;
+}
+
+/* 14.6 Labeled Statements */
+
+/* Action taken when a lableled statement is parsed. a new
+   LABELED_BLOCK_EXPR is created. No statement is attached to the
+   label, yet.  */
+
+static tree
+build_labeled_block (location, label, wfl)
+     int location;
+     tree label, wfl;
+{
+  tree label_name = merge_qualified_name (label_id, label);
+  tree label_decl, node;
+
+  /* Issue a warning if we try to reuse a label that was previously
+     declared */
+  if (IDENTIFIER_LOCAL_VALUE (label_name))
+    {
+      EXPR_WFL_LINECOL (wfl_operator) = location;
+      parse_warning_context (wfl_operator, "Declaration of `%s' shadows "
+                            "a previous declaration",
+                            IDENTIFIER_POINTER (label));
+      EXPR_WFL_LINECOL (wfl_operator) = 
+        EXPR_WFL_LINECOL (IDENTIFIER_LOCAL_VALUE (label_name));
+      parse_warning_context (wfl_operator, "This is the location of the "
+                            "previous declaration of label `%s'",
+                            IDENTIFIER_POINTER (label));
+      java_warning_count--;
+    }
+
+  label_decl = create_label_decl (label_name);
+  node = build (LABELED_BLOCK_EXPR, NULL_TREE, label_decl, NULL_TREE);
+  EXPR_WFL_LINECOL (node) = location;
+  TREE_SIDE_EFFECTS (node) = 1;
+  return node;
+}
+
+/* Generate a label crafting a unique name for it. This is used to
+   implicitely label loops that aren't the body part of labeled
+   statement.  */
+
+static tree
+generate_labeled_block ()
+{
+  static int l_number = 0;
+  char buf [20];
+  tree label_name;
+  
+  sprintf (buf, "$a%d", l_number++);
+  return build_labeled_block (0, get_identifier (buf), NULL_TREE);
+}
+
+/* A labeled statement LBE is attached a statement. If the statement
+   happens to be a loop, a link from the loop back to the label is
+   installed.  */
+
+static tree
+complete_labeled_statement (lbe, statement)
+     tree lbe;                 /* Labeled block expr */
+     tree statement;
+{
+  /* In anyways, tie the loop to its statement */
+  LABELED_BLOCK_BODY (lbe) = statement;
+
+  /* Ok, if statement is a for loop, we have to attach the labeled
+     statement to the block the for loop belongs to and return the
+     block instead */
+  if (TREE_CODE (statement) == LOOP_EXPR && IS_FOR_LOOP_P (statement))
+    {
+      java_method_add_stmt (current_function_decl, lbe);
+      return exit_block ();
+    }
+
+  return lbe;
+}
+
+/* 14.10, 14.11, 14.12 Loop Statements */
+
+/* Create an empty LOOP_EXPR and make it the last in the nested loop
+   list. */
+
+static tree
+build_new_loop (loop_body)
+     tree loop_body;
+{
+  tree loop =  build (LOOP_EXPR, NULL_TREE, loop_body);
+  TREE_SIDE_EFFECTS (loop) = 1;
+  PUSH_LOOP (loop);
+  return loop;
+}
+
+/* Create a loop body according to the following structure:
+     COMPOUND_EXPR
+       COMPOUND_EXPR           (loop main body)
+         EXIT_EXPR             (this order is for while/for loops.
+         LABELED_BLOCK_EXPR      the order is reversed for do loops)
+           LABEL_DECL           (continue occurding here branche at the 
+           BODY                         end of this labeled block)
+       INCREMENT               (if any)
+
+  REVERSED, if non zero, tells that the loop condition expr comes
+  after the body, like in the do-while loop.  */
+
+static tree
+build_loop_body (location, condition, reversed)
+     int location;
+     tree condition;
+     int reversed;
+{
+  tree first, second, label, body;
+
+  condition = build (EXIT_EXPR, NULL_TREE, condition); /* Force walk */
+  EXPR_WFL_LINECOL (condition) = location; /* For accurate error report */
+  condition = build_debugable_stmt (location, condition);
+  TREE_SIDE_EFFECTS (condition) = 1;
+
+  body = generate_labeled_block ();
+  first = (reversed ? body : condition);
+  second = (reversed ? condition : body);
+  return 
+    build (COMPOUND_EXPR, NULL_TREE, 
+          build (COMPOUND_EXPR, NULL_TREE, first, second), size_zero_node);
+}
+
+/* Install CONDITION (if any) and loop BODY (using REVERSED to tell
+   their order) on the current loop. Unlink the current loop from the
+   loop list.  */
+
+static tree
+complete_loop_body (location, condition, body, reversed)
+     int location;
+     tree condition, body;
+     int reversed;
+{
+  tree to_return = ctxp->current_loop;
+  tree loop_body = LOOP_EXPR_BODY (to_return);
+  if (condition)
+    {
+      tree cnode = LOOP_EXPR_BODY_CONDITION_EXPR (loop_body, reversed);
+      /* We wrapped the EXIT_EXPR around a WFL so we can debug it.
+         The real EXIT_EXPR is one operand further. */
+      EXPR_WFL_LINECOL (cnode) = location;
+      /* This one is for accurate error reports */
+      EXPR_WFL_LINECOL (TREE_OPERAND (cnode, 0)) = location;
+      TREE_OPERAND (TREE_OPERAND (cnode, 0), 0) = condition;
+    }
+  LOOP_EXPR_BODY_BODY_EXPR (loop_body, reversed) = body;
+  POP_LOOP ();
+  return to_return;
+}
+
+/* Tailored version of complete_loop_body for FOR loops, when FOR
+   loops feature the condition part */
+
+static tree
+complete_for_loop (location, condition, update, body)
+    int location;
+    tree condition, update, body;
+{
+  /* Put the condition and the loop body in place */
+  tree loop = complete_loop_body (location, condition, body, 0);
+  /* LOOP is the current loop which has been now popped of the loop
+     stack. Install the update block */
+  LOOP_EXPR_BODY_UPDATE_BLOCK (LOOP_EXPR_BODY (loop)) = update;
+  return loop;
+}
+
+/* If the loop isn't surrounded by a labeled statement, create one and
+   insert LOOP as it's body.  */
+
+static tree
+patch_loop_statement (loop)
+     tree loop;
+{
+  tree cbl, loop_label, to_return_as_loop;
+
+  if (LOOP_HAS_LABEL_P (loop))
+    {
+      loop_label = ctxp->current_labeled_block;
+      to_return_as_loop = loop;
+    }
+  else
+    {
+      loop_label = generate_labeled_block ();
+      LABELED_BLOCK_BODY (loop_label) = loop;
+      PUSH_LABELED_BLOCK (loop_label);
+      to_return_as_loop = loop_label;
+    }
+  TREE_TYPE (to_return_as_loop) = void_type_node;
+  return to_return_as_loop;
+}
+
+/* 14.13, 14.14: break and continue Statements */
+
+/* Build a break or a continue statement. a null NAME indicates an
+   unlabeled break/continue statement.  */
+
+static tree
+build_bc_statement (location, is_break, name)
+     int location, is_break;
+     tree name;
+{
+  tree break_continue, label_block_expr = NULL_TREE;
+
+  if (name)
+    {
+      if (!(label_block_expr = IDENTIFIER_LOCAL_VALUE 
+           (merge_qualified_name (label_id, EXPR_WFL_NODE (name)))))
+       /* Null means that we don't have a target for this named
+          break/continue. In this case, we make the target to be the
+          label name, so that the error can be reported accuratly in
+          patch_bc_statement. */
+       label_block_expr = EXPR_WFL_NODE (name);
+    }
+  /* Unlabeled break/continue will be handled during the
+     break/continue patch operation */
+  break_continue 
+    = build (EXIT_BLOCK_EXPR, NULL_TREE, label_block_expr, NULL_TREE);
+
+  IS_BREAK_STMT_P (break_continue) = is_break;
+  TREE_SIDE_EFFECTS (break_continue) = 1;
+  EXPR_WFL_LINECOL (break_continue) = location;
+  return break_continue;
+}
+
+/* Verification of a break/continue statement. */
+
+static tree
+patch_bc_statement (node)
+     tree node;
+{
+  tree bc_label = EXIT_BLOCK_LABELED_BLOCK (node), target_stmt;
+  int is_unlabeled = 0;
+ EXPR_WFL_LINECOL (wfl_operator) = EXPR_WFL_LINECOL (node);
+  /* Not having a target means that the break/continue statement is
+     unlabeled. We try to find a descent label for it */
+  if (!bc_label)
+    {
+      is_unlabeled = 1;
+      /* There should be a loop to branch to */
+      if (ctxp->current_loop)
+       {
+         /* At that stage, we're in the loop body, which is
+            encapsulated around a LABELED_BLOCK_EXPR. So searching
+            the current loop label requires us to consider the
+            labeled block before the current one. */
+         if (!LOOP_HAS_LABEL_SKIP_P (ctxp->current_loop))
+           fatal ("unlabeled loop has not installed label -- "
+                  "patch_bc_statement");
+         bc_label = TREE_CHAIN (ctxp->current_labeled_block);
+       }
+      /* Not having a loop to break/continue to is an error */
+      else
+       {
+         parse_error_context (wfl_operator, "`%s' must be in loop%s",
+                              (IS_BREAK_STMT_P (node) ? "break" : "continue"),
+                              (IS_BREAK_STMT_P (node) ? " or switch" : ""));
+         return error_mark_node;
+       }
+    }
+  /* Having an identifier here means that the target is unknown. */
+  else if (TREE_CODE (bc_label) == IDENTIFIER_NODE)
+    {
+      parse_error_context (wfl_operator, "No label definition found for `%s'",
+                          IDENTIFIER_POINTER (bc_label));
+      return error_mark_node;
+    }
+
+  /* Find the statement we're targeting. */
+  target_stmt = LABELED_BLOCK_BODY (bc_label);
+
+  /* 14.13 The break Statement */
+  if (IS_BREAK_STMT_P (node))
+    {
+      /* Named break are always fine, as far as they have a target
+         (already verified). Anonymous break need to target
+         while/do/for/switch */
+      if (is_unlabeled &&
+         !(TREE_CODE (target_stmt) == LOOP_EXPR   /* do/while/for */
+           || 0))                                 /* switch FIXME */
+       {
+         parse_error_context (wfl_operator, 
+                              "`break' must be in loop or switch");
+         return error_mark_node;
+       }
+      /* If previously unlabeled, install the new found label */
+      if (is_unlabeled)
+       EXIT_BLOCK_LABELED_BLOCK (node) = bc_label;
+    }
+  /* 14.14 The continue Statement */
+  /* The continue statement must always target a loop */
+  else 
+    {
+      if (TREE_CODE (target_stmt) != LOOP_EXPR) /* do/while/for */
+       {
+         parse_error_context (wfl_operator, "`continue' must be in loop");
+         return error_mark_node;
+       }
+      /* Everything looks good. We can fix the `continue' jump to go
+         at the place in the loop were the continue is. The continue
+        is the current labeled block, by construction. */
+      EXIT_BLOCK_LABELED_BLOCK (node) = ctxp->current_labeled_block;
+    }
+
+  /* Our break/continue don't return values. */
+  TREE_TYPE (node) = void_type_node;
+  /* Encapsulate the break within a compound statement so that it's
+     expanded all the times by expand_expr (and not clobered
+     sometimes, like after a if statement) */
+  node = add_stmt_to_compound (NULL_TREE, void_type_node, node);
+  TREE_SIDE_EFFECTS (node) = 1;
+  return node;
+}
+
+/* Process the exit expression belonging to a loop. Its type must be
+   boolean.  */
+
+static tree
+patch_exit_expr (node)
+     tree node;
+{
+  tree expression = TREE_OPERAND (node, 0);
+  TREE_TYPE (node) = error_mark_node;
+  EXPR_WFL_LINECOL (wfl_operator) = EXPR_WFL_LINECOL (node);
+
+  /* The type of expression must be boolean */
+  if (TREE_TYPE (expression) != boolean_type_node)
+    {
+      parse_error_context 
+       (wfl_operator, 
+        "Incompatible type for loop conditional. Can't convert `%s' to "
+        "`boolean'", 
+        lang_printable_name (TREE_TYPE (expression)));
+      return error_mark_node;
+    }
+  /* Now we know things are allright, invert the condition, fold and
+     return */
+  TREE_OPERAND (node, 0) = 
+    fold (build1 (TRUTH_NOT_EXPR, boolean_type_node, expression));
+  TREE_TYPE (node) = void_type_node;
+  return node;
+}
diff --git a/gcc/java/parse.h b/gcc/java/parse.h
new file mode 100644 (file)
index 0000000..50d5140
--- /dev/null
@@ -0,0 +1,599 @@
+/* Language parser definitions for the GNU compiler for the Java(TM) language.
+   Copyright (C) 1997, 1998 Free Software Foundation, Inc.
+   Contributed by Alexandre Petit-Bianco (apbianco@cygnus.com)
+
+This file is part of GNU CC.
+
+GNU CC is free software; you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation; either version 2, or (at your option)
+any later version.
+
+GNU CC is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with GNU CC; see the file COPYING.  If not, write to
+the Free Software Foundation, 59 Temple Place - Suite 330,
+Boston, MA 02111-1307, USA.
+
+Java and all Java-based marks are trademarks or registered trademarks
+of Sun Microsystems, Inc. in the United States and other countries.
+The Free Software Foundation is independent of Sun Microsystems, Inc.  */
+
+#ifndef JV_LANG_H
+#define JV_LANG_H
+
+#include "lex.h"
+
+/* Extern global variable declarations */
+extern int java_error_count;
+extern struct obstack temporary_obstack;
+extern struct obstack permanent_obstack;
+extern int quiet_flag;
+
+#ifndef JC1_LITE
+/* Function extern to java/ */
+extern int int_fits_type_p PROTO ((tree, tree));
+extern tree stabilize_reference PROTO ((tree));
+#endif
+
+/* Macros for verbose debug info  */
+#ifdef  VERBOSE_SKELETON
+#define RULE( rule ) printf ( "jv_yacc:%d: rule %s\n", lineno, rule )
+#else
+#define RULE( rule )
+#endif
+
+#ifdef SOURCE_FRONTEND_DEBUG
+#undef SOURCE_FRONTEND_DEBUG
+#define SOURCE_FRONTEND_DEBUG(X)                               \
+  {if (!quiet_flag) {printf ("* "); printf X; putchar ('\n');} }
+#else
+#define SOURCE_FRONTEND_DEBUG(X)
+#endif
+
+/* Macro for error recovering  */
+#ifdef YYDEBUG
+#define RECOVERED                                      \
+  { if (!quiet_flag) {printf ("** Recovered\n");} }
+#define DRECOVERED(s)                                          \
+  { if (!quiet_flag) {printf ("** Recovered (%s)\n", #s);}}
+#else
+#define RECOVERED
+#define DRECOVERED(s)
+#endif
+
+#define DRECOVER(s) {yyerrok; DRECOVERED(s)}
+#define RECOVER     {yyerrok; RECOVERED}
+
+#define YYERROR_NOW ctxp->java_error_flag = 1
+#define YYNOT_TWICE if (ctxp->prevent_ese != lineno)
+
+/* Accepted modifiers */
+#define CLASS_MODIFIERS ACC_PUBLIC|ACC_ABSTRACT|ACC_FINAL
+#define FIELD_MODIFIERS ACC_PUBLIC|ACC_PROTECTED|ACC_PRIVATE|ACC_FINAL| \
+                        ACC_STATIC|ACC_TRANSIENT|ACC_VOLATILE
+#define METHOD_MODIFIERS ACC_PUBLIC|ACC_PROTECTED|ACC_PRIVATE|ACC_ABSTRACT| \
+                        ACC_STATIC|ACC_FINAL|ACC_SYNCHRONIZED|ACC_NATIVE
+#define INTERFACE_MODIFIERS ACC_PUBLIC|ACC_ABSTRACT
+#define INTERFACE_METHOD_MODIFIERS ACC_PUBLIC|ACC_ABSTRACT
+#define INTERFACE_FIELD_MODIFIERS ACC_PUBLIC|ACC_STATIC|ACC_FINAL
+
+/* Getting a modifier WFL */
+#define MODIFIER_WFL(M)   (ctxp->modifier_ctx [(M) - PUBLIC_TK])
+
+/* Check on modifiers */
+#define THIS_MODIFIER_ONLY(f, m, v, count, l)                          \
+  if ((f) & (m))                                                       \
+    {                                                                  \
+      tree node = ctxp->modifier_ctx [v];                              \
+      if ((l)                                                          \
+         && ((EXPR_WFL_COLNO (node) > EXPR_WFL_COLNO (l))              \
+             || (EXPR_WFL_LINENO (node) > EXPR_WFL_LINENO (l))))       \
+        l = node;                                                      \
+      else if (!(l))                                                   \
+        l = node;                                                      \
+      count++;                                                         \
+    }
+
+#define ABSTRACT_CHECK(flag, v, cl, s)                         \
+  if ((flag) & (v))                                            \
+    parse_error_context (cl, s " method can't be abstract");
+
+/* Misc. */
+#define exit_java_complete_class()             \
+  {                                            \
+    pop_obstacks ();                           \
+    return;                                    \
+  }
+
+#define CLASS_OR_INTERFACE(decl, s1, s2)                       \
+   (decl ?                                                     \
+    ((get_access_flags_from_decl (TYPE_NAME (TREE_TYPE (decl)))        \
+      & ACC_INTERFACE) ?                                       \
+     s2 : s1) : ((s1 [0]=='S'|| s1 [0]=='s') ?                 \
+                (s1 [0]=='S' ? "Supertype" : "supertype") :    \
+                (s1 [0] > 'A' ? "Type" : "type")))
+
+/* Pedantic warning on obsolete modifiers. Note: when cl is NULL,
+   flags was set artificially, such as for a interface method */
+#define OBSOLETE_MODIFIER_WARNING(cl, flags, modifier, format, arg)          \
+  {                                                                          \
+    if ((cl) && ((flags) & (modifier)))                                             \
+      parse_warning_context (cl,                                             \
+                            "Discouraged redundant use of `%s' modifier "   \
+                            "in declaration of " format,                    \
+                            java_accstring_lookup (modifier), arg);         \
+  }
+
+/* Quickly build a temporary pointer on hypothetical type NAME. */
+#define BUILD_PTR_FROM_NAME(ptr, name)         \
+  {                                            \
+    ptr = build (POINTER_TYPE, NULL_TREE);     \
+    TYPE_NAME (ptr) = name;                    \
+  }
+
+#define INCOMPLETE_TYPE_P(NODE)                                        \
+  ((TREE_CODE (NODE) == TREE_LIST)                             \
+   && (TREE_CODE (TREE_PURPOSE (NODE)) == POINTER_TYPE)        \
+   && (TREE_TYPE (TREE_PURPOSE (NODE)) == NULL_TREE))
+
+/* Set the EMIT_LINE_NOTE flag of a EXPR_WLF to 1 if debug information
+   are requested. Works in the context of a parser rule. */
+#define JAVA_MAYBE_GENERATE_DEBUG_INFO(node)           \
+  (debug_info_level != DINFO_LEVEL_NONE ?              \
+    EXPR_WFL_EMIT_LINE_NOTE (node) = 1, node : node)
+
+/* Types classification, according to the JLS, section 4.2 */
+#define JFLOAT_TYPE_P(TYPE)      (TREE_CODE ((TYPE)) == REAL_TYPE)
+#define JINTEGRAL_TYPE_P(TYPE)   ((TREE_CODE ((TYPE)) == INTEGER_TYPE) \
+                                 || (TREE_CODE ((TYPE)) == CHAR_TYPE))
+#define JNUMERIC_TYPE_P(TYPE)    (JFLOAT_TYPE_P ((TYPE))       \
+                                 || JINTEGRAL_TYPE_P ((TYPE)))
+#define JPRIMITIVE_TYPE_P(TYPE)  (JNUMERIC_TYPE_P ((TYPE))                \
+                                 || (TREE_CODE ((TYPE)) == BOOLEAN_TYPE))
+
+/* Not defined in the LRM */
+#define JSTRING_TYPE_P(TYPE) ((TYPE) == string_type_node ||            \
+                        (TREE_CODE (TYPE) == POINTER_TYPE &&           \
+                         TREE_TYPE (op1_type) == string_type_node))
+
+#define JREFERENCE_TYPE_P(TYPE) (TREE_CODE (TYPE) == RECORD_TYPE ||    \
+                                (TREE_CODE (TYPE) == POINTER_TYPE &&   \
+                                TREE_CODE (TREE_TYPE (TYPE)) == RECORD_TYPE))
+
+/* Other predicate */
+#define DECL_P(NODE) (NODE && (TREE_CODE (NODE) == PARM_DECL           \
+                              || TREE_CODE (NODE) == VAR_DECL          \
+                              || TREE_CODE (NODE) == FIELD_DECL))
+
+#define TYPE_INTERFACE_P(TYPE)                                         \
+  (CLASS_P (TYPE) && CLASS_INTERFACE (TYPE_NAME (TYPE)))
+
+#define TYPE_CLASS_P(TYPE) (CLASS_P (TYPE)                             \
+                           && !CLASS_INTERFACE (TYPE_NAME (TYPE))      \
+                           && !TYPE_ARRAY_P (TYPE))
+
+/* Standard error messages */
+#define ERROR_CANT_CONVERT_TO_BOOLEAN(OPERATOR, NODE, TYPE)            \
+  parse_error_context                                                  \
+    ((OPERATOR), "Incompatible type for `%s'. Can't convert `%s' to "  \
+     "boolean", operator_string ((NODE)), lang_printable_name ((TYPE)))
+
+#define ERROR_CANT_CONVERT_TO_NUMERIC(OPERATOR, NODE, TYPE)            \
+  parse_error_context                                                  \
+    ((OPERATOR), "Incompatible type for `%s'. Can't convert `%s' to "  \
+     "numeric type", operator_string ((NODE)), lang_printable_name ((TYPE)))
+
+#define ERROR_CAST_NEEDED_TO_INTEGRAL(OPERATOR, NODE, TYPE)            \
+  parse_error_context                                                  \
+    ((OPERATOR), (JPRIMITIVE_TYPE_P (TYPE) ?                           \
+     "Incompatible type for `%s'. Explicit cast needed to convert "    \
+      "`%s' to integral" : "Incompatible type for `%s'. Can't convert "        \
+      "`%s' to integral"), operator_string ((NODE)),                   \
+      lang_printable_name ((TYPE)))
+
+#define ERROR_VARIABLE_NOT_INITIALIZED(WFL, V)                 \
+  parse_error_context                                          \
+    ((WFL), "Variable `%s' may not have been initialized",     \
+     IDENTIFIER_POINTER (V))
+
+/* Definition for loop handling. This Java's own definition of a loop
+   body. See parse.y for documentation. It's valid once you hold a
+   loop's body (LOOP_EXPR_BODY) */
+
+/* The loop main block is the one hold the condition and the loop body */
+#define LOOP_EXPR_BODY_MAIN_BLOCK(NODE) TREE_OPERAND (NODE, 0)
+/* And then there is the loop update block */
+#define LOOP_EXPR_BODY_UPDATE_BLOCK(NODE) TREE_OPERAND (NODE, 1)
+
+/* Inside the loop main block, there is the loop condition and the
+   loop body. They may be reversed if the loop being described is a
+   do-while loop. NOTE: if you use a WFL around the EXIT_EXPR so you
+   can issue debug info for it, the EXIT_EXPR will be one operand
+   further. */
+#define LOOP_EXPR_BODY_CONDITION_EXPR(NODE, R)                         \
+  TREE_OPERAND (LOOP_EXPR_BODY_MAIN_BLOCK (NODE), (R ? 1 : 0))
+
+/* Here is the labeled block the loop real body is encapsulated in */
+#define LOOP_EXPR_BODY_LABELED_BODY(NODE, R)                   \
+  TREE_OPERAND (LOOP_EXPR_BODY_MAIN_BLOCK (NODE), (R ? 0 : 1))
+/* And here is the loop's real body */
+#define LOOP_EXPR_BODY_BODY_EXPR(NODE, R)                      \
+  LABELED_BLOCK_BODY (LOOP_EXPR_BODY_LABELED_BODY(NODE, R))
+
+/* Does a loop have a label ? */
+#define LOOP_HAS_LABEL_P(LOOP)                                 \
+  (ctxp->current_labeled_block                                 \
+   && LABELED_BLOCK_BODY (ctxp->current_labeled_block) == (LOOP))
+
+/* Same operation than the one performed above, but considering the
+   previous labeled block */
+#define LOOP_HAS_LABEL_SKIP_P(LOOP)                                         \
+  (ctxp->current_labeled_block                                              \
+   && TREE_CHAIN (ctxp->current_labeled_block)                              \
+   && LABELED_BLOCK_BODY (TREE_CHAIN (ctxp->current_labeled_block)) == (LOOP))
+
+#define PUSH_LABELED_BLOCK(B)                          \
+  {                                                    \
+    TREE_CHAIN (B) = ctxp->current_labeled_block;      \
+    ctxp->current_labeled_block = (B);                 \
+  }
+#define POP_LABELED_BLOCK()                                            \
+  ctxp->current_labeled_block = TREE_CHAIN (ctxp->current_labeled_block)
+
+#define PUSH_LOOP(L)                           \
+  {                                            \
+    TREE_CHAIN (L) = ctxp->current_loop;       \
+    ctxp->current_loop = (L);                  \
+  }
+#define POP_LOOP() ctxp->current_loop = TREE_CHAIN (ctxp->current_loop)
+
+
+/* Invocation modes, as returned by invocation_mode (). */
+enum {
+  INVOKE_STATIC,
+  INVOKE_NONVIRTUAL,
+  INVOKE_SUPER,
+  INVOKE_INTERFACE,
+  INVOKE_VIRTUAL,
+};
+
+/* We need the resolution stuff only if we compile jc1 */
+#ifndef JC1_LITE
+
+/* Unresolved type identifiers handling. When we process the source
+   code, we blindly accept an unknown type identifier and try to
+   resolve it later. When an unknown type identifier is encountered
+   and used, we record in a struct jdep element what the incomplete
+   type is and what it should patch. Later, java_complete_class will
+   process all classes known to have unresolved type
+   dependencies. Within each of these classes, this routine will
+   process unresolved type dependencies (JDEP_TO_RESOLVE), patch what
+   needs to be patched in the dependent tree node (JDEP_GET_PATCH,
+   JDEP_APPLY_PATCH) and perform other actions dictated by the context
+   of the patch (JDEP_KIND). The ideas are: we patch only what needs
+   to be patched, and with java_complete_class called at the right
+   time, we will start processing incomplete function bodies tree
+   nodes with everything external to function's bodies already
+   completed, it makes things much simpler. */
+
+enum jdep_code {
+  JDEP_NO_PATCH,               /* Must be first */
+  JDEP_SUPER,                  /* Patch the type of one type
+                                  supertype. Requires some check
+                                  before it's done */
+  JDEP_FIELD,                  /* Patch the type of a class field */
+
+  /* JDEP_{METHOD,METHOD_RETURN,METHOD_END} to be kept in order */
+  JDEP_METHOD,                 /* Mark the beginning of the patching
+                                  of a method declaration, including
+                                  it's arguments */
+  JDEP_METHOD_RETURN,          /* Mark the beginning of the patching
+                                  of a method declaration. Arguments
+                                  aren't patched, only the returned
+                                  type is */
+  JDEP_METHOD_END,             /* Mark the end of the patching of a
+                                  method declaration. It indicates
+                                  that it's time to compute and
+                                  install a new signature */
+
+  JDEP_INTERFACE,              /* Patch the type of a Class/interface
+                                  extension */
+  JDEP_VARIABLE,               /* Patch the type of a variable declaration */
+  JDEP_PARM,                   /* Patch the type of a parm declaration */
+  JDEP_TYPE,                   /* Patch a random tree node type,
+                                   without the need for any specific
+                                   actions */
+};
+
+typedef struct _jdep {
+#ifdef ONLY_INT_FIELDS
+  int  kind : 8;               /* Type of patch */
+#else
+  enum jdep_code kind : 8;
+#endif
+
+  int  flag0 : 1;              /* Some flags */
+  tree decl;                   /* Tied decl/or WFL */
+  tree solv;                   /* What to solve */
+  tree wfl;                    /* Where thing to resolve where found */
+  tree misc;                   /* Miscellaneous info (optional). */
+  tree *patch;                 /* Address of a location to patch */
+  struct _jdep *next;          /* Linked list */
+} jdep;
+
+
+#define JDEP_DECL(J)          ((J)->decl)
+#define JDEP_DECL_WFL(J)      ((J)->decl)
+#define JDEP_KIND(J)          ((J)->kind)
+#define JDEP_SOLV(J)          ((J)->solv)
+#define JDEP_WFL(J)           ((J)->wfl)
+#define JDEP_MISC(J)          ((J)->misc)
+#define JDEP_CLASS(J)         ((J)->class)
+#define JDEP_APPLY_PATCH(J,P) (*(J)->patch = (P))
+#define JDEP_GET_PATCH(J)     ((J)->patch)
+#define JDEP_CHAIN(J)         ((J)->next)
+#define JDEP_TO_RESOLVE(J)    (TREE_PURPOSE ((J)->solv))
+#define JDEP_RESOLVED_DECL(J) ((J)->solv ? TREE_PURPOSE ((J)->solv):NULL_TREE)
+#define JDEP_RESOLVED(J, D)                    \
+  {                                            \
+    TREE_PURPOSE ((J)->solv) = D;              \
+    TREE_VALUE ((J)->solv) = (J)->solv;                \
+  }
+#define JDEP_RESOLVED_P(J)    (!(J)->solv ||                           \
+                              TREE_VALUE ((J)->solv) == (J)->solv)
+
+typedef struct _jdeplist {
+  jdep *first;
+  jdep *last;
+  struct _jdeplist *next;
+} jdeplist;
+static jdeplist *reverse_jdep_list ();
+
+#endif /* JC1_LITE */
+
+#define CLASSD_FIRST(CD) ((CD)->first)
+#define CLASSD_LAST(CD)  ((CD)->last)
+#define CLASSD_CHAIN(CD) ((CD)->next)
+
+#define JDEP_INSERT(L,J)                       \
+  {                                            \
+    if (!(L)->first)                           \
+      (L)->last = (L)->first = (J);            \
+    else                                       \
+      {                                                \
+       JDEP_CHAIN ((L)->last) = (J);           \
+       (L)->last = (J);                        \
+      }                                                \
+  }
+
+/* Insert a DECL in the current block */
+#define BLOCK_CHAIN_DECL(NODE)                                             \
+  {                                                                        \
+    TREE_CHAIN ((NODE)) =                                                  \
+      BLOCK_EXPR_DECLS (DECL_FUNCTION_BODY (current_function_decl));       \
+    BLOCK_EXPR_DECLS (DECL_FUNCTION_BODY (current_function_decl)) = (NODE); \
+  }
+
+#define BLOCK_EXPR_DECLS(NODE)  BLOCK_VARS(NODE)
+#define BLOCK_EXPR_BODY(NODE)   BLOCK_SUBBLOCKS(NODE)
+#define BLOCK_EXPR_ORIGIN(NODE) BLOCK_ABSTRACT_ORIGIN(NODE)
+
+/* Merge an other line to the source line number of a decl. Used to
+   remember function's end. */
+#define DECL_SOURCE_LINE_MERGE(DECL,NO) DECL_SOURCE_LINE(DECL) |= (NO << 16)
+
+/* Retrieve those two info separately. */
+#define DECL_SOURCE_LINE_FIRST(DECL)    (DECL_SOURCE_LINE(DECL) & 0x0000ffff)
+#define DECL_SOURCE_LINE_LAST(DECL)     (DECL_SOURCE_LINE(DECL) >> 16)
+
+/* Build a WFL for expression nodes */
+#define BUILD_EXPR_WFL(NODE, WFL)                                      \
+  build_expr_wfl ((NODE), input_filename, EXPR_WFL_LINENO ((WFL)),     \
+                 EXPR_WFL_COLNO ((WFL)))
+
+#define EXPR_WFL_QUALIFICATION(WFL) TREE_OPERAND ((WFL), 1)
+#define QUAL_WFL(NODE) TREE_PURPOSE (NODE)
+#define QUAL_RESOLUTION(NODE) TREE_VALUE (NODE)
+#define QUAL_DECL_TYPE(NODE)                           \
+  (TREE_CODE (TREE_TYPE (NODE)) == POINTER_TYPE ?      \
+   TREE_TYPE (TREE_TYPE (NODE)) : TREE_TYPE (NODE))
+
+/* Handy macros for the walk operation */
+#define COMPLETE_CHECK_OP(NODE, N)                     \
+{                                                      \
+  TREE_OPERAND ((NODE), (N)) =                                 \
+    java_complete_tree (TREE_OPERAND ((NODE), (N)));   \
+  if (TREE_OPERAND ((NODE), (N)) == error_mark_node)   \
+    return error_mark_node;                            \
+}
+#define COMPLETE_CHECK_OP_0(NODE) COMPLETE_CHECK_OP(NODE, 0)
+#define COMPLETE_CHECK_OP_1(NODE) COMPLETE_CHECK_OP(NODE, 1)
+
+/* Parser context data structure. */
+struct parser_ctxt {
+
+  char *filename;                  /* Current filename */
+  FILE *finput;                            /* Current file input stream */
+  struct parser_ctxt *next;
+
+  struct java_line *p_line, *c_line; /* Previous and current line */
+  java_lc elc;                      /* Error's line column info */
+  unicode_t unget_utf8_value;        /* An unget utf8 value */
+  int ccb_indent;                   /* Keep track of {} indent, lexer */
+  int first_ccb_indent1;            /* First { at ident level 1 */
+  int last_ccb_indent1;                     /* Last } at ident level 1 */
+  int parser_ccb_indent;            /* Keep track of {} indent, parser */
+  int osb_number;                   /* Keep track of ['s */
+  int minus_seen;                   /* Integral literal overflow */
+  int lineno;                      /* Current lineno */
+  int java_error_flag;             /* Report error when true */
+
+  /* This section is defined only if we compile jc1 */
+#ifndef JC1_LITE
+  tree modifier_ctx [11];           /* WFL of modifiers */
+  tree current_class;              /* Current class */
+  tree current_function_decl;      /* Current function decl, save/restore */
+
+  JCF *current_jcf;                /* CU jcf */
+
+  int prevent_ese;                 /* Prevent expression statement error */
+  int class_err;                   /* Flag to report certain errors */
+
+  int formal_parameter_number;     /* Number of parameters found */
+  int interface_number;                    /* # itfs declared to extend an itf def */
+
+  tree package;                            /* Defined package ID */
+
+  tree  incomplete_class;          /* List of non-complete classes */
+  tree  current_parsed_class;      /* Class currently parsed */
+  tree  class_list;                /* List of classes in a CU */
+  jdeplist *classd_list;           /* Classe dependencies in a CU */
+  
+  tree non_static_initialized;     /* List of non static initialized fields */
+  tree static_initialized;         /* List of static non final initialized */
+
+  tree import_list;                /* List of import */
+  tree import_demand_list;         /* List of import on demand */
+
+  tree current_loop;                /* List of the currently nested loops */
+  tree current_labeled_block;       /* List of currently nested
+                                       labeled blocks. */
+
+  int pending_block;                /* Pending block to close */
+#endif /* JC1_LITE */
+};
+
+/* Functions declarations */
+#ifndef JC1_LITE
+static char *java_accstring_lookup PROTO ((int));
+static void  parse_error PROTO ((char *));
+static void  redefinition_error PROTO ((char *,tree, tree, tree));
+static void  check_modifiers PROTO ((char *, int, int));
+static tree  create_class PROTO ((int, tree, tree, tree));
+static tree  create_interface PROTO ((int, tree, tree));
+static tree  find_field PROTO ((tree, tree));
+static tree lookup_field_wrapper PROTO ((tree, tree));
+static int   duplicate_declaration_error PROTO ((tree, tree, tree, tree));
+static void  register_fields PROTO ((int, tree, tree));
+static tree parser_qualified_classname PROTO ((tree));
+static int  parser_check_super PROTO ((tree, tree, tree));
+static int  parser_check_super_interface PROTO ((tree, tree, tree));
+static void check_modifiers_consistency PROTO ((int));
+static tree lookup_cl PROTO ((tree));
+static tree lookup_java_method2 PROTO ((tree, tree, int));
+static tree method_header PROTO ((int, tree, tree, tree));
+static tree method_declarator PROTO ((tree, tree));
+static void parse_error_context VPROTO ((tree cl, char *msg, ...));
+static void parse_warning_context VPROTO ((tree cl, char *msg, ...));
+static void complete_class_report_errors PROTO ((jdep *));
+static int process_imports PROTO ((void));
+static void read_import_dir PROTO ((tree));
+static int find_in_imports_on_demand PROTO ((tree));
+static int find_in_imports PROTO ((tree));
+static int check_pkg_class_access PROTO ((tree, tree));
+static tree resolve_class PROTO ((tree, tree, tree));
+static tree do_resolve_class PROTO ((tree, tree, tree));
+static void declare_local_variables PROTO ((int, tree, tree));
+static void source_start_java_method PROTO ((tree));
+static void source_end_java_method PROTO ((void));
+static void expand_start_java_method PROTO ((tree));
+static tree find_name_in_single_imports PROTO ((tree));
+static void check_abstract_method_header PROTO ((tree));
+static tree lookup_java_interface_method2 PROTO ((tree, tree));
+static tree resolve_expression_name PROTO ((tree));
+static tree maybe_create_class_interface_decl PROTO ((tree, tree, tree));
+static int check_class_interface_creation PROTO ((int, int, tree, tree, tree, tree));
+static tree patch_method_invocation_stmt PROTO ((tree, tree, tree, int *));
+static int breakdown_qualified PROTO ((tree *, tree *, tree));
+static tree resolve_and_layout PROTO ((tree, tree));
+static tree resolve_no_layout PROTO ((tree, tree));
+static int identical_subpath_p PROTO ((tree, tree));
+static int invocation_mode PROTO ((tree, int));
+static tree refine_accessible_methods_list PROTO ((int, tree));
+static tree patch_invoke PROTO ((tree, tree, tree, tree));
+static tree lookup_method_invoke PROTO ((int, tree, tree, tree, tree));
+static tree register_incomplete_type PROTO ((int, tree, tree, tree));
+static tree obtain_incomplete_type PROTO ((tree));
+static tree java_complete_tree PROTO ((tree));
+static void java_complete_expand_method PROTO ((tree));
+static int  unresolved_type_p PROTO ((tree, tree *));
+static void create_jdep_list PROTO ((struct parser_ctxt *));
+static tree build_expr_block PROTO ((tree, tree));
+static tree enter_block PROTO ((void));
+static tree exit_block PROTO ((void));
+static tree lookup_name_in_blocks PROTO ((tree));
+static void maybe_absorb_scoping_blocks PROTO ((void));
+static tree build_method_invocation PROTO ((tree, tree));
+static tree build_assignment PROTO ((int, int, tree, tree));
+static tree build_binop PROTO ((enum tree_code, int, tree, tree));
+static tree patch_assignment PROTO ((tree, tree, tree ));
+static tree patch_binop PROTO ((tree, tree, tree));
+static tree build_unaryop PROTO ((int, int, tree));
+static tree build_incdec PROTO ((int, int, tree, int));
+static tree patch_unaryop PROTO ((tree, tree));
+static tree build_cast PROTO ((int, tree, tree));
+static tree patch_cast PROTO ((tree, tree, tree));
+static int valid_ref_assignconv_cast_p PROTO ((tree, tree, int));
+static int can_cast_to_p PROTO ((tree, tree));
+static tree build_unresolved_array_type PROTO ((tree));
+static tree build_array_ref PROTO ((int, tree, tree));
+static tree patch_array_ref PROTO ((tree, tree, tree));
+static tree make_qualified_name PROTO ((tree, tree, int));
+static tree merge_qualified_name PROTO ((tree, tree));
+static tree make_qualified_primary PROTO ((tree, tree, int));
+static int resolve_qualified_expression_name PROTO ((tree, tree *, tree *, tree *));
+static void qualify_ambiguous_name PROTO ((tree));
+static void maybe_generate_clinit PROTO ((void));
+static tree resolve_field_access PROTO ((tree, tree *, tree *));
+static tree build_newarray_node PROTO ((tree, tree, int));
+static tree patch_newarray PROTO ((tree));
+static tree resolve_type_during_patch PROTO ((tree));
+static int not_initialized_as_it_should_p PROTO ((tree));
+static tree build_this PROTO ((int));
+static tree build_return PROTO ((int, tree));
+static tree patch_return PROTO ((tree));
+static tree maybe_access_field PROTO ((tree, tree, tree));
+static int complete_function_arguments PROTO ((tree));
+static int check_for_static_method_reference PROTO ((tree, tree, tree, tree, tree));
+static int not_accessible_p PROTO ((tree, tree, int));
+static int class_in_current_package PROTO ((tree));
+static tree build_if_else_statement PROTO ((int, tree, tree, tree));
+static tree patch_if_else_statement PROTO ((tree));
+static tree add_stmt_to_compound PROTO ((tree, tree, tree));
+static tree patch_exit_expr PROTO ((tree));
+static tree build_labeled_block PROTO ((int, tree, tree));
+static tree generate_labeled_block PROTO (());
+static tree complete_labeled_statement PROTO ((tree, tree));
+static tree build_bc_statement PROTO ((int, int, tree));
+static tree patch_bc_statement PROTO ((tree));
+static tree patch_loop_statement PROTO ((tree));
+static tree build_new_loop PROTO ((tree));
+static tree build_loop_body PROTO ((int, tree, int));
+static tree complete_loop_body PROTO ((int, tree, tree, int));
+static tree build_debugable_stmt PROTO ((int, tree));
+static tree complete_for_loop PROTO ((int, tree, tree, tree));
+
+void safe_layout_class PROTO ((tree));
+void java_complete_class PROTO ((void));
+void java_check_circular_reference PROTO ((void));
+void java_check_final PROTO ((void));
+void java_check_methods PROTO ((void));
+void java_layout_classes PROTO ((void));
+tree java_method_add_stmt PROTO ((tree, tree));
+char *java_get_line_col PROTO ((char *, int, int));
+#endif /* JC1_LITE */
+
+/* Always in use, no matter what you compile */
+
+void java_push_parser_context PROTO ((void));
+void java_init_lex PROTO ((void));
+int yyparse PROTO ((void));
+int yylex ();
+void yyerror PROTO ((char *));
+
+#endif
diff --git a/gcc/java/parse.y b/gcc/java/parse.y
new file mode 100644 (file)
index 0000000..50a4679
--- /dev/null
@@ -0,0 +1,8035 @@
+/* Source code parsing and tree node generation for the GNU compiler
+   for the Java(TM) language.
+   Copyright (C) 1997, 1998 Free Software Foundation, Inc.
+   Contributed by Alexandre Petit-Bianco (apbianco@cygnus.com)
+
+This file is part of GNU CC.
+
+GNU CC is free software; you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation; either version 2, or (at your option)
+any later version.
+
+GNU CC is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with GNU CC; see the file COPYING.  If not, write to
+the Free Software Foundation, 59 Temple Place - Suite 330,
+Boston, MA 02111-1307, USA.
+
+Java and all Java-based marks are trademarks or registered trademarks
+of Sun Microsystems, Inc. in the United States and other countries.
+The Free Software Foundation is independent of Sun Microsystems, Inc.  */
+
+/* This file parses java source code and issues a tree node image
+suitable for code generation (byte code and targeted CPU assembly
+language).
+
+The grammar conforms to the Java grammar described in "The Java(TM)
+Language Specification. J. Gosling, B. Joy, G. Steele. Addison Wesley
+1996, ISBN 0-201-63451-1"
+
+The following modifications were brought to the original grammar:
+
+method_body: added the rule '| block SC_TK'
+constructor_declaration: added two rules to accept SC_TK. 
+static_initializer: added the rule 'static block SC_TK'. 
+
+Note: All the extra rules described above should go away when the
+      empty_statement rule will work.
+
+statement_nsi: 'nsi' should be read no_short_if.
+
+Some rules have been modified to support JDK1.1 inner classes
+definitions and other extensions.  */
+
+%{
+#include <stdio.h>
+#include <stdlib.h>
+#include <string.h>
+#include <dirent.h>
+#ifdef __STDC__
+#include <stdarg.h>
+#else
+#include <varargs.h>
+#endif
+
+#include "config.h"
+#include "tree.h"
+#include "rtl.h"
+#include "obstack.h"
+#include "flags.h"
+#include "java-tree.h"
+#include "jcf.h"
+#include "lex.h"
+#include "parse.h"
+#include "zipfile.h"
+
+/* Number of error found so far. */
+int java_error_count; 
+/* Number of warning found so far. */
+int java_warning_count;
+
+/* The current parser context */
+static struct parser_ctxt *ctxp;
+
+/* binop_lookup maps token to tree_code. It is used where binary
+   operations are involved and required by the parser. RDIV_EXPR
+   covers both integral/floating point division. The code is changed
+   once the type of both operator is worked out.  */
+
+static enum tree_code binop_lookup[19] = 
+  { 
+    PLUS_EXPR, MINUS_EXPR, MULT_EXPR, RDIV_EXPR, TRUNC_MOD_EXPR,
+    LSHIFT_EXPR, RSHIFT_EXPR, URSHIFT_EXPR, 
+    BIT_AND_EXPR, BIT_XOR_EXPR, BIT_IOR_EXPR,
+    TRUTH_ANDIF_EXPR, TRUTH_ORIF_EXPR,
+    EQ_EXPR, NE_EXPR, GT_EXPR, GE_EXPR, LT_EXPR, LE_EXPR,
+   };
+#define BINOP_LOOKUP(VALUE)                                            \
+  binop_lookup [((VALUE) - PLUS_TK)%                                   \
+               (sizeof (binop_lookup) / sizeof (binop_lookup[0]))]
+
+/* Fake WFL used to report error message. It is initialized once if
+   needed and reused with it's location information is overriden.  */
+static tree wfl_operator = NULL_TREE;
+
+/* The "$L" identifier we use to create labels.  */
+static tree label_id;
+%}
+
+%union {
+  tree node;
+  int sub_token;
+  struct {
+    int token;
+    int location;
+  } operator;
+  int value;
+}
+
+%pure_parser
+
+/* Things defined here have to match the order of what's in the
+   binop_lookup table.  */
+
+%token   PLUS_TK         MINUS_TK        MULT_TK         DIV_TK    REM_TK
+%token   LS_TK           SRS_TK          ZRS_TK
+%token   AND_TK          XOR_TK          OR_TK
+%token   BOOL_AND_TK BOOL_OR_TK 
+%token   EQ_TK NEQ_TK GT_TK GTE_TK LT_TK LTE_TK
+
+/* This maps to the same binop_lookup entry than the token above */
+
+%token   PLUS_ASSIGN_TK  MINUS_ASSIGN_TK MULT_ASSIGN_TK DIV_ASSIGN_TK
+%token   REM_ASSIGN_TK   
+%token   LS_ASSIGN_TK    SRS_ASSIGN_TK   ZRS_ASSIGN_TK
+%token   AND_ASSIGN_TK   XOR_ASSIGN_TK   OR_ASSIGN_TK
+
+
+/* Modifier TOKEN have to be kept in this order. Don't scramble it */
+
+%token   PUBLIC_TK       PRIVATE_TK         PROTECTED_TK
+%token   STATIC_TK       FINAL_TK           SYNCHRONIZED_TK
+%token   VOLATILE_TK     TRANSIENT_TK       NATIVE_TK
+%token   PAD_TK          ABSTRACT_TK        MODIFIER_TK
+
+/* Keep those two in order, too */
+%token   DECR_TK INCR_TK
+
+/* From now one, things can be in any order */
+
+%token   DEFAULT_TK      IF_TK              THROW_TK
+%token   BOOLEAN_TK      DO_TK              IMPLEMENTS_TK
+%token   THROWS_TK       BREAK_TK           IMPORT_TK       
+%token   ELSE_TK         INSTANCEOF_TK      RETURN_TK
+%token   VOID_TK         CATCH_TK           INTERFACE_TK
+%token   CASE_TK         EXTENDS_TK         FINALLY_TK
+%token   SUPER_TK        WHILE_TK           CLASS_TK
+%token   SWITCH_TK       CONST_TK           TRY_TK
+%token   FOR_TK          NEW_TK             CONTINUE_TK
+%token   GOTO_TK         PACKAGE_TK         THIS_TK
+
+%token   BYTE_TK         SHORT_TK           INT_TK            LONG_TK
+%token   CHAR_TK         INTEGRAL_TK
+
+%token   FLOAT_TK        DOUBLE_TK          FP_TK
+
+%token   ID_TK
+
+%token   REL_QM_TK         REL_CL_TK NOT_TK  NEG_TK
+
+%token   ASSIGN_ANY_TK   ASSIGN_TK
+%token   OP_TK  CP_TK  OCB_TK  CCB_TK  OSB_TK  CSB_TK  SC_TK  C_TK DOT_TK
+
+%token   STRING_LIT_TK   CHAR_LIT_TK        INT_LIT_TK        FP_LIT_TK
+%token   TRUE_TK         FALSE_TK           BOOL_LIT_TK       NULL_TK
+
+%type    <value>       modifiers MODIFIER_TK
+
+%type    <node>                super ID_TK identifier
+%type    <node>                name simple_name qualified_name
+%type   <node>         class_declaration type_declaration compilation_unit
+                       field_declaration method_declaration extends_interfaces
+                        interfaces interface_type_list
+                        interface_declaration class_member_declaration
+                        import_declarations package_declaration 
+                        type_declarations interface_body
+                       interface_member_declaration constant_declaration
+                       interface_member_declarations interface_type
+                       abstract_method_declaration interface_type_list
+%type   <node>         class_body_declaration class_member_declaration
+                       static_initializer constructor_declaration block
+%type   <node>         class_body_declarations
+%type    <node>                class_or_interface_type class_type class_type_list
+                       constructor_declarator explicit_constructor_invocation
+%type    <node>         dim_expr dim_exprs this_or_super
+
+%type   <node>         variable_declarator_id variable_declarator
+                       variable_declarators variable_initializer
+                       variable_initializers
+
+%type   <node>         class_body
+%type   <node>         block_statement local_variable_declaration_statement
+                       block_statements local_variable_declaration
+%type   <node>         statement statement_without_trailing_substatement
+                       labeled_statement if_then_statement label_decl
+                       if_then_else_statement while_statement for_statement
+                       statement_nsi labeled_statement_nsi do_statement
+                       if_then_else_statement_nsi while_statement_nsi
+                       for_statement_nsi statement_expression_list for_init
+                       for_update statement_expression expression_statement
+                       primary_no_new_array expression primary
+                       array_creation_expression array_type
+                       class_instance_creation_expression field_access
+                       method_invocation array_access something_dot_new
+                       argument_list postfix_expression while_expression 
+                       post_increment_expression post_decrement_expression
+                       unary_expression_not_plus_minus unary_expression
+                       pre_increment_expression pre_decrement_expression
+                       unary_expression_not_plus_minus cast_expression
+                       multiplicative_expression additive_expression
+                       shift_expression relational_expression 
+                       equality_expression and_expression 
+                       exclusive_or_expression inclusive_or_expression
+                       conditional_and_expression conditional_or_expression
+                       conditional_expression assignment_expression
+                       left_hand_side assignment for_header for_begin
+                       constant_expression do_statement_begin empty_statement
+%type    <node>         return_statement break_statement continue_statement
+
+%type    <operator>     ASSIGN_TK      MULT_ASSIGN_TK  DIV_ASSIGN_TK  
+%type    <operator>     REM_ASSIGN_TK  PLUS_ASSIGN_TK  MINUS_ASSIGN_TK
+%type    <operator>     LS_ASSIGN_TK   SRS_ASSIGN_TK   ZRS_ASSIGN_TK
+%type    <operator>     AND_ASSIGN_TK  XOR_ASSIGN_TK   OR_ASSIGN_TK
+%type    <operator>     ASSIGN_ANY_TK  assignment_operator
+%token   <operator>     EQ_TK GTE_TK ZRS_TK SRS_TK GT_TK LTE_TK LS_TK 
+%token   <operator>     BOOL_AND_TK AND_TK BOOL_OR_TK OR_TK INCR_TK PLUS_TK
+%token   <operator>     DECR_TK MINUS_TK MULT_TK DIV_TK XOR_TK REM_TK NEQ_TK
+%token   <operator>     NEG_TK REL_QM_TK REL_CL_TK NOT_TK LT_TK
+%token   <operator>     OP_TK OSB_TK DOT_TK
+%type    <operator>    THIS_TK SUPER_TK RETURN_TK BREAK_TK CONTINUE_TK
+
+%type   <node>         method_body 
+       
+%type    <node>                literal INT_LIT_TK FP_LIT_TK BOOL_LIT_TK CHAR_LIT_TK
+                       STRING_LIT_TK NULL_TK VOID_TK
+
+%type   <node>         IF_TK WHILE_TK FOR_TK
+
+%type    <node>         formal_parameter_list formal_parameter
+                        method_declarator method_header
+
+%type   <node>         primitive_type reference_type type
+                       BOOLEAN_TK INTEGRAL_TK FP_TK
+
+%%
+/* 19.2 Production from 2.3: The Syntactic Grammar  */
+goal:
+       compilation_unit
+               {}
+;
+
+/* 19.3 Productions from 3: Lexical structure  */
+literal:
+       INT_LIT_TK
+|      FP_LIT_TK
+|      BOOL_LIT_TK
+|      CHAR_LIT_TK
+|       STRING_LIT_TK
+|       NULL_TK
+;
+
+/* 19.4 Productions from 4: Types, Values and Variables  */
+type:
+       primitive_type
+|      reference_type
+;
+
+primitive_type:
+       INTEGRAL_TK
+|      FP_TK
+|      BOOLEAN_TK
+;
+
+reference_type:
+       class_or_interface_type
+|      array_type
+;
+
+class_or_interface_type:
+       name
+;
+
+class_type:
+       class_or_interface_type /* Default rule */
+;
+
+interface_type:
+        class_or_interface_type
+;
+
+array_type:
+       primitive_type OSB_TK CSB_TK
+               { 
+                 $$ = build_java_array_type ($1, -1);
+                 CLASS_LOADED_P ($$) = 1;
+               }
+|      name OSB_TK CSB_TK
+               { $$ = build_unresolved_array_type ($1); }
+|      array_type OSB_TK CSB_TK
+               { $$ = build_unresolved_array_type ($1); }
+|      primitive_type OSB_TK error
+               {RULE ("']' expected"); RECOVER;}
+|      array_type OSB_TK error
+               {RULE ("']' expected"); RECOVER;}
+;
+
+/* 19.5 Productions from 6: Names  */
+name:
+       simple_name             /* Default rule */
+|      qualified_name          /* Default rule */
+;
+
+simple_name:
+       identifier              /* Default rule */
+;
+
+qualified_name:
+       name DOT_TK identifier
+               { $$ = make_qualified_name ($1, $3, $2.location); }
+;
+
+identifier:
+       ID_TK
+;
+
+/* 19.6: Production from 7: Packages  */
+compilation_unit:
+               {$$ = NULL;}
+|      package_declaration
+|      import_declarations
+|      type_declarations
+|       package_declaration import_declarations
+|       package_declaration type_declarations
+|       import_declarations type_declarations
+|       package_declaration import_declarations type_declarations
+;
+
+import_declarations:
+       import_declaration
+               {
+                 $$ = NULL;
+               }
+|      import_declarations import_declaration
+               {
+                 $$ = NULL;
+               }
+;
+
+type_declarations:
+       type_declaration
+|      type_declarations type_declaration
+;
+
+package_declaration:
+       PACKAGE_TK name SC_TK
+               { ctxp->package = EXPR_WFL_NODE ($2); }
+|      PACKAGE_TK error
+               {yyerror ("Missing name"); RECOVER;}
+|      PACKAGE_TK name error
+               {yyerror ("';' expected"); RECOVER;}
+;
+
+import_declaration:
+       single_type_import_declaration
+|      type_import_on_demand_declaration
+;
+
+single_type_import_declaration:
+       IMPORT_TK name SC_TK
+               {
+                 tree name = EXPR_WFL_NODE ($2), node, last_name;
+                 int   i = IDENTIFIER_LENGTH (name)-1;
+                 char *last = &IDENTIFIER_POINTER (name)[i];
+                 while (last != IDENTIFIER_POINTER (name))
+                   {
+                     if (last [0] == '.')
+                       break;
+                     last--;
+                   }
+                 last_name = get_identifier (++last);
+                 if (IS_A_SINGLE_IMPORT_CLASSFILE_NAME_P (last_name))
+                   {
+                     tree err = find_name_in_single_imports (last_name);
+                     if (err && err != name)
+                       parse_error_context
+                         ($2, "Ambiguous class: `%s' and `%s'",
+                          IDENTIFIER_POINTER (name), 
+                          IDENTIFIER_POINTER (err));
+                   }
+                 else
+                   {
+                     IS_A_SINGLE_IMPORT_CLASSFILE_NAME_P (last_name) = 1;
+                     node = build_tree_list ($2, last_name);
+                     TREE_CHAIN (node) = ctxp->import_list;
+                     ctxp->import_list = node;
+                   }
+               }
+|      IMPORT_TK error
+               {yyerror ("Missing name"); RECOVER;}
+|      IMPORT_TK name error
+               {yyerror ("';' expected"); RECOVER;}
+;
+
+type_import_on_demand_declaration:
+       IMPORT_TK name DOT_TK MULT_TK SC_TK
+               {
+                 tree name = EXPR_WFL_NODE ($2);
+                 tree node = build_tree_list ($2, NULL_TREE);
+                 if (!IS_AN_IMPORT_ON_DEMAND_P (name))
+                   {
+                     read_import_dir ($2);
+                     IS_AN_IMPORT_ON_DEMAND_P (name) = 1;
+                   }
+                 TREE_CHAIN (node) = ctxp->import_demand_list;
+                 ctxp->import_demand_list = node;
+               }
+|      IMPORT_TK name DOT_TK error
+               {yyerror ("'*' expected"); RECOVER;}
+|      IMPORT_TK name DOT_TK MULT_TK error
+               {yyerror ("';' expected"); RECOVER;}
+;
+
+type_declaration:
+       class_declaration
+               {
+                 maybe_generate_clinit ();
+                 $$ = $1;
+               }
+|      interface_declaration
+|      SC_TK
+               { $$ = NULL; }
+|      error
+               {
+                 YYERROR_NOW;
+                 yyerror ("Class or interface declaration expected");
+               }
+;
+
+/* 19.7 Shortened from the original:
+   modifiers: modifier | modifiers modifier
+   modifier: any of public...  */
+modifiers:
+       MODIFIER_TK
+               {
+                 $$ = (1 << $1);
+               }
+|      modifiers MODIFIER_TK
+               {
+                 int acc = (1 << $2);
+                 if ($$ & acc)
+                   parse_error_context 
+                     (ctxp->modifier_ctx [$2], "Modifier `%s' declared twice",
+                      java_accstring_lookup (acc));
+                 else
+                   {
+                     $$ |= acc;
+                   }
+               }
+;
+
+/* 19.8.1 Production from $8.1: Class Declaration */
+class_declaration:
+       modifiers CLASS_TK identifier super interfaces
+               { create_class ($1, $3, $4, $5); }
+       class_body
+               { 
+                 $$ = $7;
+               }
+|      CLASS_TK identifier super interfaces 
+               { create_class (0, $2, $3, $4); }
+       class_body
+               {       
+                 $$ = $6;
+               }
+|      modifiers CLASS_TK error
+               {yyerror ("Missing class name"); RECOVER;}
+|      CLASS_TK error
+               {yyerror ("Missing class name"); RECOVER;}
+|       CLASS_TK identifier error
+               {if (!ctxp->class_err) yyerror ("'{' expected"); DRECOVER(class1);}
+|       modifiers CLASS_TK identifier error
+               {if (!ctxp->class_err) yyerror ("'{' expected"); RECOVER;}
+;
+
+super:
+               { $$ = NULL; }
+|      EXTENDS_TK class_type
+               { $$ = $2; }
+|      EXTENDS_TK class_type error
+               {yyerror ("'{' expected"); ctxp->class_err=1;}
+|      EXTENDS_TK error
+               {yyerror ("Missing super class name"); ctxp->class_err=1;}
+;
+
+interfaces:
+               { $$ = NULL_TREE; }
+|      IMPLEMENTS_TK interface_type_list
+               { $$ = $2; }
+|      IMPLEMENTS_TK error
+               {
+                 ctxp->class_err=1;
+                 yyerror ("Missing interface name"); 
+               }
+;
+
+interface_type_list:
+       interface_type
+               { 
+                 ctxp->interface_number = 1;
+                 $$ = build_tree_list ($1, NULL_TREE);
+               }
+|      interface_type_list C_TK interface_type
+               { 
+                 ctxp->interface_number++;
+                 $$ = chainon ($1, build_tree_list ($3, NULL_TREE));
+               }
+|      interface_type_list C_TK error
+               {yyerror ("Missing interface name"); RECOVER;}
+;
+
+class_body:
+       OCB_TK CCB_TK
+               { $$ = ctxp->current_parsed_class; }
+|      OCB_TK class_body_declarations CCB_TK
+               { $$ = ctxp->current_parsed_class; }
+;
+
+class_body_declarations:
+       class_body_declaration
+|      class_body_declarations class_body_declaration
+;
+
+class_body_declaration:
+       class_member_declaration
+|      static_initializer
+|      constructor_declaration
+|      block                   /* Added, JDK1.1, instance initializer */
+;
+
+class_member_declaration:
+       field_declaration
+|      method_declaration
+|      class_declaration       /* Added, JDK1.1 inner classes */
+|      interface_declaration   /* Added, JDK1.1 inner classes */
+;
+
+/* 19.8.2 Productions from 8.3: Field Declarations  */
+field_declaration:
+       type variable_declarators SC_TK
+               { register_fields (0, $1, $2); }
+|      modifiers type variable_declarators SC_TK
+               {
+                 int acc_count = 0;
+
+                 check_modifiers 
+                   ("Illegal modifier `%s' for field declaration",
+                    $1, FIELD_MODIFIERS);
+                 check_modifiers_consistency ($1);
+                 register_fields ($1, $2, $3);
+               }
+;
+
+variable_declarators:
+       /* Should we use build_decl_list () instead ? FIXME */
+       variable_declarator     /* Default rule */
+|      variable_declarators C_TK variable_declarator
+               { $$ = chainon ($1, $3); }
+|      variable_declarators C_TK error
+               {yyerror ("Missing term"); RECOVER;}
+;
+
+variable_declarator:
+       variable_declarator_id
+               { $$ = build_tree_list ($1, NULL_TREE); }
+|      variable_declarator_id ASSIGN_TK variable_initializer
+               { 
+                 if (java_error_count)
+                   $3 = NULL_TREE;
+                 $$ = build_tree_list 
+                   ($1, build_assignment ($2.token, $2.location, $1, $3));
+               }
+|      variable_declarator_id ASSIGN_TK error
+               {
+                 yyerror ("Missing variable initializer");
+                 $$ = build_tree_list ($1, NULL_TREE);
+                 RECOVER;
+               }
+|      variable_declarator_id ASSIGN_TK variable_initializer error
+               {
+                 yyerror ("';' expected");
+                 $$ = build_tree_list ($1, NULL_TREE);
+                 RECOVER;
+               }
+;
+
+variable_declarator_id:
+       identifier
+|      variable_declarator_id OSB_TK CSB_TK
+               {
+                 $$ = NULL;    /* FIXME */
+               }
+|      identifier error
+               {yyerror ("Invalid declaration"); DRECOVER(vdi);}
+|      variable_declarator_id OSB_TK error
+               {yyerror ("']' expected"); DRECOVER(vdi);}
+|      variable_declarator_id CSB_TK error
+               {yyerror ("Unbalanced ']'"); DRECOVER(vdi);}
+;
+
+variable_initializer:
+       expression
+|      array_initializer
+               { $$ = NULL; }
+;
+
+/* 19.8.3 Productions from 8.4: Method Declarations  */
+method_declaration:
+       method_header 
+               {
+                 current_function_decl = $1;
+                 source_start_java_method (current_function_decl);
+               }
+       method_body
+               { 
+                 BLOCK_EXPR_BODY 
+                   (DECL_FUNCTION_BODY (current_function_decl)) = $3;
+                 maybe_absorb_scoping_blocks ();
+                 exit_block (); /* Exit function's body. */
+
+                 /* Merge last line of the function with first line,
+                    directly in the function decl. It will be used to
+                    emit correct debug info. */
+                 DECL_SOURCE_LINE_MERGE (current_function_decl,
+                                         ctxp->last_ccb_indent1);
+               }
+|      method_header error
+               {YYNOT_TWICE yyerror ("'{' expected"); RECOVER;}
+;
+
+method_header: 
+       type method_declarator throws
+               { $$ = method_header (0, $1, $2, NULL); }
+|      VOID_TK method_declarator throws
+               { $$ = method_header (0, void_type_node, $2, NULL); }
+|      modifiers type method_declarator throws
+               { $$ = method_header ($1, $2, $3, NULL); }
+|      modifiers VOID_TK method_declarator throws
+               { $$ = method_header ($1, void_type_node, $3, NULL); }
+|      type error
+               {RECOVER;}
+|      modifiers type error
+               {RECOVER;}
+|      VOID_TK error
+               {yyerror ("Identifier expected"); RECOVER;}
+|      modifiers VOID_TK error
+               {yyerror ("Identifier expected"); RECOVER;}
+|      modifiers error
+               {
+                 yyerror ("Invalid method declaration, return type required");
+                 RECOVER;
+               }
+;
+
+method_declarator:
+       identifier OP_TK CP_TK
+               { $$ = method_declarator ($1, NULL_TREE); }
+|      identifier OP_TK formal_parameter_list CP_TK
+               { $$ = method_declarator ($1, $3); }
+|      method_declarator OSB_TK CSB_TK
+               {
+                 /* Issue a warning here: obsolete declaration. FIXME */
+                 $$ = NULL;    /* FIXME */
+               }
+|      identifier OP_TK error
+               {yyerror ("')' expected"); DRECOVER(method_declarator);}
+|      method_declarator OSB_TK error
+               {yyerror ("']' expected"); RECOVER;}
+;
+
+formal_parameter_list:
+       formal_parameter
+               {
+                 ctxp->formal_parameter_number = 1;
+               }
+|      formal_parameter_list C_TK formal_parameter
+               {
+                 ctxp->formal_parameter_number += 1;
+                 $$ = chainon ($1, $3);
+               }
+|      formal_parameter_list C_TK error
+               {yyerror ("Missing formal parameter term"); RECOVER;}
+;
+
+formal_parameter:
+       type variable_declarator_id
+               {
+                 $$ = build_tree_list ($2, $1);
+               }
+|      modifiers type variable_declarator_id /* Added, JDK1.1 final locals */
+               {
+                 SOURCE_FRONTEND_DEBUG (("Modifiers: %d", $1));
+                 $$ = NULL;    /* FIXME */
+               }
+|      type error
+               {yyerror ("Missing identifier"); RECOVER;}
+|      modifiers type error
+               {
+                 SOURCE_FRONTEND_DEBUG (("Modifiers: %d", $1));
+                 yyerror ("Missing identifier"); RECOVER;
+               }
+;
+
+throws:
+|      THROWS_TK class_type_list
+|      THROWS_TK error
+               {yyerror ("Missing class type term"); RECOVER;}
+;
+
+class_type_list:
+       class_type
+|      class_type_list C_TK class_type
+|      class_type_list C_TK error
+               {yyerror ("Missing class type term"); RECOVER;}
+;
+
+method_body:
+       block
+|      block SC_TK
+|      SC_TK
+               { $$ = NULL_TREE; } /* Probably not the right thing to do. */
+;
+
+/* 19.8.4 Productions from 8.5: Static Initializers  */
+static_initializer:
+       static block
+               {
+                 RULE ("STATIC_INITIALIZER");
+               }
+|      static block SC_TK      /* Shouldn't be here. FIXME */
+               {
+                 RULE ("STATIC_INITIALIZER");
+               }
+;
+
+static:                                /* Test lval.sub_token here */
+       MODIFIER_TK
+               {
+                 SOURCE_FRONTEND_DEBUG (("Modifiers: %d", $1));
+               }
+;
+
+/* 19.8.5 Productions from 8.6: Constructor Declarations  */
+/* NOTE FOR FURTHER WORK ON CONSTRUCTORS:
+   - If a forbidded modifier is found, the the error is either the use of
+     a forbidded modifier for a constructor OR bogus attempt to declare a
+     method without having specified the return type. FIXME */
+constructor_declaration:
+       constructor_declarator throws constructor_body
+               {
+                 RULE ("CONSTRUCTOR_DECLARATION");
+               }
+|      modifiers constructor_declarator throws constructor_body
+               {
+                 SOURCE_FRONTEND_DEBUG (("Modifiers: %d", $1));
+                 RULE ("CONSTRUCTOR_DECLARATION (modifier)");
+               }
+/* extra SC_TK, FIXME */
+|      constructor_declarator throws constructor_body SC_TK
+               {
+                 RULE ("CONSTRUCTOR_DECLARATION");
+               }
+/* extra SC_TK, FIXME */
+|      modifiers constructor_declarator throws constructor_body SC_TK
+               {
+                 SOURCE_FRONTEND_DEBUG (("Modifiers: %d", $1));
+                 RULE ("CONSTRUCTOR_DECLARATION (modifier)");
+               }
+/* I'm not happy with the SC_TK addition. It isn't in the grammer and should
+   probably be matched by and empty statement. But it doesn't work. FIXME */
+;
+
+constructor_declarator:
+       simple_name OP_TK CP_TK
+|      simple_name OP_TK formal_parameter_list CP_TK
+;
+
+constructor_body:
+       OCB_TK CCB_TK
+|      OCB_TK explicit_constructor_invocation CCB_TK
+|      OCB_TK block_statements CCB_TK
+|       OCB_TK explicit_constructor_invocation block_statements CCB_TK
+;
+
+/* Error recovery for that rule moved down expression_statement: rule.  */
+explicit_constructor_invocation:
+       this_or_super OP_TK CP_TK SC_TK
+|      this_or_super OP_TK argument_list CP_TK SC_TK
+        /* Added, JDK1.1 inner classes. Modified because the rule
+          'primary' couldn't work.  */
+|      name DOT_TK SUPER_TK OP_TK argument_list CP_TK SC_TK
+|      name DOT_TK SUPER_TK OP_TK CP_TK SC_TK
+               {RULE ("explicit_constructor_invocation (X.super)");}
+;
+
+this_or_super:                 /* Added, simplifies error diagnostics */
+       THIS_TK
+               {
+                 tree wfl = build_wfl_node (this_identifier_node, input_filename, 0, 0);
+                 EXPR_WFL_LINECOL (wfl) = $1.location;
+                 $$ = wfl;
+               }
+|      SUPER_TK
+               {
+                 tree wfl = build_wfl_node (super_identifier_node, input_filename, 0, 0);
+                 EXPR_WFL_LINECOL (wfl) = $1.location;
+                 $$ = wfl;
+               }
+;
+
+/* 19.9 Productions from 9: Interfaces  */
+/* 19.9.1 Productions from 9.1: Interfaces Declarations  */
+interface_declaration:
+       INTERFACE_TK identifier
+               { create_interface (0, $2, NULL_TREE); }
+       interface_body
+               {
+                 $$ = $4;
+               }
+|      modifiers INTERFACE_TK identifier
+               { create_interface ($1, $3, NULL_TREE); }
+       interface_body
+               {
+                 $$ = $5;
+               }
+|      INTERFACE_TK identifier extends_interfaces
+               { create_interface (0, $2, $3); }
+       interface_body
+               {
+                 $$ = $5;
+               }
+|      modifiers INTERFACE_TK identifier extends_interfaces
+               { create_interface ($1, $3, $4); }
+       interface_body
+               {
+                 $$ = $6;
+               }
+|      INTERFACE_TK identifier error
+               {yyerror ("(here)'{' expected"); RECOVER;}
+|      modifiers INTERFACE_TK identifier error
+               {yyerror ("(there)'{' expected"); RECOVER;}
+;
+
+extends_interfaces:
+       EXTENDS_TK interface_type
+               { 
+                 ctxp->interface_number = 1;
+                 $$ = build_tree_list ($2, NULL_TREE);
+               }
+|      extends_interfaces C_TK interface_type
+               { 
+                 ctxp->interface_number++;
+                 $$ = chainon ($1, build_tree_list ($3, NULL_TREE));
+               }
+|      EXTENDS_TK error
+               {yyerror ("Invalid interface type"); RECOVER;}
+|      extends_interfaces C_TK error
+               {yyerror ("Missing term"); RECOVER;}
+;
+
+interface_body:
+       OCB_TK CCB_TK
+               { $$ = NULL_TREE; }
+|      OCB_TK interface_member_declarations CCB_TK
+               { $$ = NULL_TREE; }
+;
+
+interface_member_declarations:
+       interface_member_declaration
+|      interface_member_declarations interface_member_declaration
+;
+
+interface_member_declaration:
+       constant_declaration
+|      abstract_method_declaration
+|      class_declaration       /* Added, JDK1.1 inner classes */
+|      interface_declaration   /* Added, JDK1.1 inner classes */
+;
+
+constant_declaration:
+       field_declaration
+;
+
+abstract_method_declaration:
+       method_header SC_TK
+               { 
+                 check_abstract_method_header ($1);
+                 current_function_decl = NULL_TREE; /* FIXME ? */
+               }
+|      method_header error
+               {yyerror ("';' expected"); RECOVER;}
+;
+
+/* 19.10 Productions from 10: Arrays  */
+array_initializer:
+       OCB_TK CCB_TK
+               {
+                 RULE ("ARRAY_INITIALIZER (empty)");
+               }
+|      OCB_TK variable_initializers CCB_TK
+               {
+                 RULE ("ARRAY_INITIALIZER (variable)");
+               }
+|      OCB_TK C_TK CCB_TK
+               {
+                 RULE ("ARRAY_INITIALIZER (,)");
+               }
+|      OCB_TK variable_initializers C_TK CCB_TK
+               {
+                 RULE ("ARRAY_INITIALIZER (variable, ,)");
+               }
+;
+
+variable_initializers:
+       variable_initializer
+|      variable_initializers C_TK variable_initializer
+|      variable_initializers C_TK error
+               {yyerror ("Missing term"); RECOVER;}
+;
+
+/* 19.11 Production from 14: Blocks and Statements  */
+block:
+       OCB_TK CCB_TK
+               { $$ = size_zero_node; }
+|      OCB_TK 
+               { enter_block (); }
+       block_statements
+       CCB_TK
+               { 
+                 maybe_absorb_scoping_blocks ();
+                 $$ = exit_block ();
+               }
+;
+
+block_statements:
+       block_statement
+|      block_statements block_statement
+;
+
+block_statement:
+       local_variable_declaration_statement
+|      statement
+               { $$ = java_method_add_stmt (current_function_decl, $1); }
+|      class_declaration       /* Added, JDK1.1 inner classes */
+;
+
+local_variable_declaration_statement:
+       local_variable_declaration SC_TK /* Can't catch missing ';' here */
+;
+
+local_variable_declaration:
+       type variable_declarators
+               { declare_local_variables (0, $1, $2); }
+|      modifiers type variable_declarators /* Added, JDK1.1 final locals */
+               { declare_local_variables ($1, $2, $3); }
+;
+
+statement:
+       statement_without_trailing_substatement
+|      labeled_statement
+               { RULE ("STATEMENT (labeled)"); }
+|      if_then_statement
+               { RULE ("STATEMENT (if-then)"); }
+|      if_then_else_statement
+               { RULE ("STATEMENT (if-then-else)"); }
+|      while_statement
+               { RULE ("STATEMENT (while)"); }
+|      for_statement
+               { 
+                 /* If the for loop is unlabeled, we must return the
+                    block it was defined it. It our last chance to
+                    get a hold on it. */
+                 if (!LOOP_HAS_LABEL_P ($$))
+                   $$ = exit_block ();
+               }
+;
+
+statement_nsi:
+       statement_without_trailing_substatement
+|      labeled_statement_nsi
+               { RULE ("NSI STATEMENT (labeled)"); }
+|      if_then_else_statement_nsi
+               { RULE ("NSI STATEMENT (if-then-else)"); }
+|      while_statement_nsi
+               { RULE ("NSI STATEMENT (while)"); }
+|      for_statement_nsi
+               { RULE ("NSI STATEMENT (for)"); }
+;
+
+statement_without_trailing_substatement:
+       block
+               { RULE ("STATEMENT (block)"); }
+|      empty_statement
+               { RULE ("STATEMENT (empty)"); }
+|      expression_statement
+               { RULE ("STATEMENT (expression)"); }
+|      switch_statement
+               { RULE ("STATEMENT (switch)"); }
+|      do_statement
+               { RULE ("STATEMENT (do)"); }
+|      break_statement
+               { RULE ("STATEMENT (break)"); }
+|      continue_statement
+               { RULE ("STATEMENT (continue)"); }
+|      return_statement
+|      synchronized_statement
+               { RULE ("STATEMENT (synchronized)"); }
+|      throw_statement
+               { RULE ("STATEMENT (throw)"); }
+|      try_statement
+               { RULE ("STATEMENT (try)"); }
+;
+
+empty_statement:
+       SC_TK
+               { $$ = size_zero_node; }
+;
+
+label_decl:
+       identifier REL_CL_TK
+               {
+                 $$ = build_labeled_block (EXPR_WFL_LINECOL ($1), 
+                                           EXPR_WFL_NODE ($1), $1);
+                 pushlevel (2);
+                 push_labeled_block ($$);
+                 PUSH_LABELED_BLOCK ($$);
+               }
+;
+
+labeled_statement:
+       label_decl statement
+               { 
+                 $$ = complete_labeled_statement ($1, $2);
+                 pop_labeled_block ();
+                 POP_LABELED_BLOCK ();
+               }
+|      identifier error
+               {yyerror ("':' expected"); RECOVER;}
+;
+
+labeled_statement_nsi:
+       label_decl statement_nsi
+               { 
+                 $$ = complete_labeled_statement ($1, $2);
+                 pop_labeled_block ();
+                 POP_LABELED_BLOCK ();
+               }
+;
+
+/* We concentrate here a bunch of error handling rules that we couldn't write
+   earlier, because expression_statement catches a missing ';'.  */
+expression_statement:
+       statement_expression SC_TK
+               {
+                 /* We have a statement. Generate a WFL around it so
+                    we can debug it */
+                 $$ = build_expr_wfl ($1, input_filename, lineno, 0);
+                 /* We know we have a statement, so set the debug
+                     info to be eventually generate here. */
+                 $$ = JAVA_MAYBE_GENERATE_DEBUG_INFO ($$);
+               }
+|      error SC_TK 
+               {
+                 if (ctxp->prevent_ese != lineno)
+                   yyerror ("Invalid expression statement");
+                 DRECOVER (expr_stmt);
+               }
+|      error OCB_TK
+               {
+                 if (ctxp->prevent_ese != lineno)
+                   yyerror ("Invalid expression statement");
+                 DRECOVER (expr_stmt);
+               }
+|      error CCB_TK
+               {
+                 if (ctxp->prevent_ese != lineno)
+                   yyerror ("Invalid expression statement");
+                 DRECOVER (expr_stmt);
+               }
+|       this_or_super OP_TK error
+               {yyerror ("')' expected"); RECOVER;}
+|       this_or_super OP_TK CP_TK error
+               {yyerror ("';' expected"); RECOVER;}
+|       this_or_super OP_TK argument_list error
+               {yyerror ("')' expected"); RECOVER;}
+|       this_or_super OP_TK argument_list CP_TK error
+               {yyerror ("';' expected"); RECOVER;}
+|      name DOT_TK SUPER_TK error
+               {yyerror ("'(' expected"); RECOVER;}
+|      name DOT_TK SUPER_TK OP_TK error
+               {yyerror ("')' expected"); RECOVER;}
+|      name DOT_TK SUPER_TK OP_TK argument_list error
+               {yyerror ("')' expected"); RECOVER;}
+|      name DOT_TK SUPER_TK OP_TK argument_list CP_TK error
+               {yyerror ("';' expected"); RECOVER;}
+|      name DOT_TK SUPER_TK OP_TK CP_TK error
+               {yyerror ("';' expected"); RECOVER;}
+;
+
+statement_expression: 
+       assignment
+|      pre_increment_expression
+               {
+                 RULE ("++INCREMENT");
+               }
+|      pre_decrement_expression
+               {
+                 RULE ("--DECREMENT");
+               }
+|      post_increment_expression
+               {
+                 RULE ("INCREMENT++");
+               }
+|      post_decrement_expression
+               {
+                 RULE ("DECREMENT--");
+               }
+|      method_invocation
+|      class_instance_creation_expression
+               {
+                 RULE ("INSTANCE CREATION");
+               }
+;
+
+if_then_statement:
+       IF_TK OP_TK expression CP_TK statement
+       { $$ = build_if_else_statement ($2.location, $3, $5, NULL_TREE); }
+|      IF_TK error
+               {yyerror ("'(' expected"); RECOVER;}
+|      IF_TK OP_TK error
+               {yyerror ("Missing term"); RECOVER;}
+|      IF_TK OP_TK expression error
+               {yyerror ("')' expected"); RECOVER;}
+;
+
+if_then_else_statement:
+       IF_TK OP_TK expression CP_TK statement_nsi ELSE_TK statement
+       { $$ = build_if_else_statement ($2.location, $3, $5, $7); }
+;
+
+if_then_else_statement_nsi:
+       IF_TK OP_TK expression CP_TK statement_nsi ELSE_TK statement_nsi
+       { $$ = build_if_else_statement ($2.location, $3, $5, $7); }
+;
+
+switch_statement:
+       SWITCH_TK OP_TK expression CP_TK switch_block
+|      SWITCH_TK error
+               {yyerror ("'(' expected"); RECOVER;}
+|      SWITCH_TK OP_TK error
+               {yyerror ("Missing term or ')'"); DRECOVER(switch_statement);}
+|      SWITCH_TK OP_TK expression CP_TK error
+               {yyerror ("'{' expected"); RECOVER;}
+;
+
+switch_block:
+       OCB_TK CCB_TK
+|      OCB_TK switch_labels CCB_TK
+|      OCB_TK switch_block_statement_groups CCB_TK
+|      OCB_TK switch_block_statement_groups switch_labels CCB_TK
+;
+
+switch_block_statement_groups: 
+       switch_block_statement_group
+|      switch_block_statement_groups switch_block_statement_group
+;
+
+switch_block_statement_group:
+       switch_labels block_statements
+;
+
+
+switch_labels:
+       switch_label
+|      switch_labels switch_label
+;
+
+switch_label:
+       CASE_TK constant_expression REL_CL_TK
+|      DEFAULT_TK REL_CL_TK
+|      CASE_TK error
+               {yyerror ("Missing or invalid constant expression"); RECOVER;}
+|      CASE_TK constant_expression error
+               {yyerror ("':' expected"); RECOVER;}
+|      DEFAULT_TK error
+               {yyerror ("':' expected"); RECOVER;}
+;
+
+while_expression:
+       WHILE_TK OP_TK expression CP_TK
+               { 
+                 tree body = build_loop_body ($2.location, $3, 0);
+                 $$ = build_new_loop (body);
+               }
+;
+
+while_statement:
+       while_expression statement
+               { $$ = complete_loop_body (0, NULL_TREE, $2, 0); }
+|      WHILE_TK error
+               {YYERROR_NOW; yyerror ("'(' expected"); RECOVER;}
+|      WHILE_TK OP_TK error
+               {yyerror ("Missing term and ')' expected"); RECOVER;}
+|      WHILE_TK OP_TK expression error
+               {yyerror ("')' expected"); RECOVER;}
+;
+
+while_statement_nsi:
+       while_expression statement_nsi
+               { $$ = complete_loop_body (0, NULL_TREE, $2, 0); }
+;
+
+do_statement_begin:
+       DO_TK
+               { 
+                 tree body = build_loop_body (0, NULL_TREE, 1);
+                 $$ = build_new_loop (body);
+               }
+       /* Need error handing here. FIXME */
+;
+
+do_statement: 
+       do_statement_begin statement WHILE_TK OP_TK expression CP_TK SC_TK
+               { $$ = complete_loop_body ($4.location, $5, $2, 1); }
+;
+
+for_statement:
+       for_begin SC_TK expression SC_TK for_update CP_TK statement
+               { $$ = complete_for_loop (EXPR_WFL_LINECOL ($3), $3, $5, $7);}
+|      for_begin SC_TK SC_TK for_update CP_TK statement
+               { 
+                 $$ = complete_for_loop (0, NULL_TREE, $4, $6);
+                 /* We have not condition, so we get rid of the EXIT_EXPR */
+                 LOOP_EXPR_BODY_CONDITION_EXPR (LOOP_EXPR_BODY ($$), 0) = 
+                   size_zero_node;
+               }
+|      for_begin SC_TK error
+               {yyerror ("Invalid control expression"); RECOVER;}
+|      for_begin SC_TK expression SC_TK error
+               {yyerror ("Invalid update expression"); RECOVER;}
+|      for_begin SC_TK SC_TK error
+               {yyerror ("Invalid update expression"); RECOVER;}
+;
+
+for_statement_nsi:
+       for_begin SC_TK expression SC_TK for_update CP_TK statement_nsi
+               { $$ = complete_for_loop (EXPR_WFL_LINECOL ($3), $3, $5, $7);}
+|      for_begin SC_TK SC_TK for_update CP_TK statement_nsi
+               { 
+                 $$ = complete_for_loop (0, NULL_TREE, $4, $6);
+                 /* We have not condition, so we get rid of the EXIT_EXPR */
+                 LOOP_EXPR_BODY_CONDITION_EXPR (LOOP_EXPR_BODY ($$), 0) = 
+                   size_zero_node;
+               }
+;
+
+for_header:
+       FOR_TK OP_TK
+               { 
+                 /* This scope defined for local variable that may be
+                     defined within the scope of the for loop */
+                 enter_block (); 
+               }
+|      FOR_TK error
+               {yyerror ("'(' expected"); DRECOVER(for_1);}
+|      FOR_TK OP_TK error
+               {yyerror ("Invalid init statement"); RECOVER;}
+;
+
+for_begin:
+       for_header for_init
+               { 
+                 /* We now declare the loop body. The loop is
+                     declared as a for loop. */
+                 tree body = build_loop_body (0, NULL_TREE, 0);
+                 $$ =  build_new_loop (body);
+                 IS_FOR_LOOP_P ($$) = 1;
+                 /* The loop is added to the current block the for
+                     statement is defined within */
+                 java_method_add_stmt (current_function_decl, $$);
+               }
+;
+for_init:                      /* Can be empty */
+               { $$ = size_zero_node; }
+|      statement_expression_list
+               { 
+                 /* Init statement recorded within the previously
+                     defined block scope */
+                 $$ = java_method_add_stmt (current_function_decl, $1);
+               }
+|      local_variable_declaration
+               { 
+                 /* Local variable are recorded within the previously
+                    defined block scope */
+                 $$ = NULL_TREE;
+               }
+|      statement_expression_list error
+               {yyerror ("';' expected"); DRECOVER(for_init_1);}
+;
+
+for_update:                    /* Can be empty */
+               {$$ = size_zero_node;}
+|      statement_expression_list
+               { $$ = build_debugable_stmt (BUILD_LOCATION (), $1); }
+;
+
+statement_expression_list:
+       statement_expression
+               { $$ = add_stmt_to_compound (NULL_TREE, NULL_TREE, $1); }
+|      statement_expression_list C_TK statement_expression
+               { $$ = add_stmt_to_compound ($1, NULL_TREE, $3); }
+|      statement_expression_list C_TK error
+               {yyerror ("Missing term"); RECOVER;}
+;
+
+break_statement:
+       BREAK_TK SC_TK
+               { $$ = build_bc_statement ($1.location, 1, NULL_TREE); }
+|      BREAK_TK identifier SC_TK
+               { $$ = build_bc_statement ($1.location, 1, $2); }
+|      BREAK_TK error
+               {yyerror ("Missing term"); RECOVER;}
+|      BREAK_TK identifier error
+               {yyerror ("';' expected"); RECOVER;}
+;
+
+continue_statement:
+       CONTINUE_TK SC_TK
+               { $$ = build_bc_statement ($1.location, 0, NULL_TREE); }
+|       CONTINUE_TK identifier SC_TK
+               { $$ = build_bc_statement ($1.location, 0, $2); }
+|      CONTINUE_TK error
+               {yyerror ("Missing term"); RECOVER;}
+|      CONTINUE_TK identifier error
+               {yyerror ("';' expected"); RECOVER;}
+;
+
+return_statement:
+       RETURN_TK SC_TK
+               { $$ = build_return ($1.location, NULL_TREE); }
+|      RETURN_TK expression SC_TK
+               { $$ = build_return ($1.location, $2); }
+|      RETURN_TK error
+               {yyerror ("Missing term"); RECOVER;}
+|      RETURN_TK expression error
+               {yyerror ("';' expected"); RECOVER;}
+;
+
+throw_statement:
+       THROW_TK expression SC_TK
+|      THROW_TK error
+               {yyerror ("Missing term"); RECOVER;}
+|      THROW_TK expression error
+               {yyerror ("';' expected"); RECOVER;}
+;
+
+synchronized_statement:
+       synchronized OP_TK expression CP_TK block
+|      synchronized OP_TK expression CP_TK error
+               {yyerror ("'{' expected"); RECOVER;}
+|      synchronized error
+               {yyerror ("'(' expected"); RECOVER;}
+|      synchronized OP_TK error CP_TK
+               {yyerror ("Missing term"); RECOVER;}
+|      synchronized OP_TK error
+               {yyerror ("Missing term"); RECOVER;}
+;
+
+synchronized:                  /* Test lval.sub_token here */
+       MODIFIER_TK
+               {
+                 SOURCE_FRONTEND_DEBUG (("Modifiers: %d", $1));
+               }
+;
+
+try_statement:
+       TRY_TK block catches
+|      TRY_TK block finally
+|      TRY_TK block catches finally
+|      TRY_TK error
+               {yyerror ("'{' expected"); DRECOVER (try_statement);}
+;
+
+catches:
+       catch_clause
+|      catches catch_clause
+;
+
+catch_clause:
+       CATCH_TK OP_TK formal_parameter CP_TK block
+|      CATCH_TK error
+               {yyerror ("'(' expected"); RECOVER;}
+|      CATCH_TK OP_TK error CP_TK /* That's for () */
+               {yyerror ("Missing term"); DRECOVER (1);}
+|      CATCH_TK OP_TK error 
+               {yyerror ("Missing term"); DRECOVER (2);}
+;
+
+finally:
+       FINALLY_TK block
+|      FINALLY_TK error
+               {yyerror ("'{' expected"); RECOVER; }
+;
+
+/* 19.12 Production from 15: Expressions  */
+primary:
+       primary_no_new_array
+|      array_creation_expression
+;
+
+primary_no_new_array:
+       literal
+|      THIS_TK
+               { $$ = build_this ($1.location); }
+|      OP_TK expression CP_TK
+               {$$ = $2;}
+|      class_instance_creation_expression
+|      field_access
+|      method_invocation
+|      array_access
+       /* type DOT_TK CLASS_TK doens't work. So we split the rule
+          'type' into its components. Missing is something for array,
+          which will complete the reference_type part. FIXME */
+|      name DOT_TK CLASS_TK           /* Added, JDK1.1 class literals */
+|      primitive_type DOT_TK CLASS_TK /* Added, JDK1.1 class literals */
+|      VOID_TK DOT_TK CLASS_TK        /* Added, JDK1.1 class literals */
+        /* Added, JDK1.1 inner classes. Documentation is wrong
+           refering to a 'ClassName' (class_name) rule that doesn't
+           exist. Used name instead.  */
+|      name DOT_TK THIS_TK
+|      OP_TK expression error 
+               {yyerror ("')' expected"); RECOVER;}
+|      name DOT_TK error
+               {yyerror ("'class' or 'this' expected" ); RECOVER;}
+|      primitive_type DOT_TK error
+               {yyerror ("'class' expected" ); RECOVER;}
+|      VOID_TK DOT_TK error
+               {yyerror ("'class' expected" ); RECOVER;}
+;
+
+class_instance_creation_expression:
+       NEW_TK class_type OP_TK argument_list CP_TK
+               {
+                 $$ = build_method_invocation ($2, $4);
+                 TREE_SET_CODE ($$, JAVA_NEW_CLASS_EXPR);
+               }
+|      NEW_TK class_type OP_TK CP_TK
+               {
+                 $$ = build_method_invocation ($2, NULL_TREE);
+                 TREE_SET_CODE ($$, JAVA_NEW_CLASS_EXPR);
+               }
+        /* Added, JDK1.1 inner classes but modified to use
+           'class_type' instead of 'TypeName' (type_name) mentionned
+           in the documentation but doesn't exist. */
+|      NEW_TK class_type OP_TK argument_list CP_TK class_body
+{$$ = $2;}
+|      NEW_TK class_type OP_TK CP_TK class_body         
+{$$ = $2;}
+        /* Added, JDK1.1 inner classes, modified to use name or
+          primary instead of primary solely which couldn't work in
+          all situations.  */
+|      something_dot_new identifier OP_TK CP_TK
+|      something_dot_new identifier OP_TK CP_TK class_body
+|      something_dot_new identifier OP_TK argument_list CP_TK
+|      something_dot_new identifier OP_TK argument_list CP_TK class_body
+|      NEW_TK error SC_TK 
+               {yyerror ("'(' expected"); DRECOVER(new_1);}
+|      NEW_TK class_type error
+               {yyerror ("'(' expected"); RECOVER;}
+|      NEW_TK class_type OP_TK error
+               {yyerror ("')' or term expected"); RECOVER;}
+|      NEW_TK class_type OP_TK argument_list error
+               {yyerror ("')' expected"); RECOVER;}
+|      something_dot_new error
+               {YYERROR_NOW; yyerror ("Identifier expected"); RECOVER;}
+|      something_dot_new identifier error
+               {yyerror ("'(' expected"); RECOVER;}
+;
+
+something_dot_new:             /* Added, not part of the specs. */
+       name DOT_TK NEW_TK
+|      primary DOT_TK NEW_TK
+;
+
+argument_list:
+       expression
+               { 
+                 $$ = tree_cons (NULL_TREE, $1, NULL_TREE);
+                 ctxp->formal_parameter_number = 1; 
+               }
+|      argument_list C_TK expression
+               {
+                 ctxp->formal_parameter_number += 1;
+                 $$ = tree_cons (NULL_TREE, $3, $1);
+               }
+|      argument_list C_TK error
+               {yyerror ("Missing term"); RECOVER;}
+;
+
+array_creation_expression:
+       NEW_TK primitive_type dim_exprs
+               { $$ = build_newarray_node ($2, $3, 0); }
+|      NEW_TK class_or_interface_type dim_exprs
+               { $$ = build_newarray_node ($2, $3, 0); }
+|      NEW_TK primitive_type dim_exprs dims
+               { $$ = build_newarray_node ($2, $3, ctxp->osb_number); }
+|      NEW_TK class_or_interface_type dim_exprs dims
+               { $$ = build_newarray_node ($2, $3, ctxp->osb_number); }
+        /* Added, JDK1.1 anonymous array. Initial documentation rule
+           modified */
+|      NEW_TK class_or_interface_type dims array_initializer
+{$$ = $2;}
+|      NEW_TK primitive_type dims array_initializer
+{$$ = $2;}
+|      NEW_TK error CSB_TK
+               {yyerror ("'[' expected"); DRECOVER ("]");}
+|      NEW_TK error OSB_TK
+               {yyerror ("']' expected"); RECOVER;}
+;
+
+dim_exprs:
+       dim_expr
+               { $$ = build_tree_list (NULL_TREE, $1); }
+|      dim_exprs dim_expr
+               { $$ = tree_cons (NULL_TREE, $2, $$); }
+;
+
+dim_expr:
+       OSB_TK expression CSB_TK
+               { 
+                 EXPR_WFL_LINECOL ($2) = $1.location;
+                 $$ = $2;
+               }
+|      OSB_TK expression error
+               {yyerror ("']' expected"); RECOVER;}
+|      OSB_TK error
+               {
+                 yyerror ("Missing term");
+                 yyerror ("']' expected");
+                 RECOVER;
+               }
+;
+
+dims:                          
+       OSB_TK CSB_TK
+               { ctxp->osb_number = 1; }
+|      dims OSB_TK CSB_TK
+               { ctxp->osb_number++; }
+|      dims OSB_TK error
+               { yyerror ("']' expected"); RECOVER;}
+;
+
+field_access:
+       primary DOT_TK identifier
+               { $$ = make_qualified_primary ($1, $3, $2.location); }
+|      SUPER_TK DOT_TK identifier
+               {
+                 tree super_wfl = 
+                   build_wfl_node (super_identifier_node, input_filename, 0, 0);
+                 EXPR_WFL_LINECOL (super_wfl) = $1.location;
+                 $$ = make_qualified_name (super_wfl, $3, $2.location);
+               }
+|      SUPER_TK error
+               {yyerror ("Field expected"); DRECOVER (super_field_acces);}
+;
+
+method_invocation:
+       name OP_TK CP_TK
+               { $$ = build_method_invocation ($1, NULL_TREE); }
+|      name OP_TK argument_list CP_TK
+               { $$ = build_method_invocation ($1, $3); }
+|      primary DOT_TK identifier OP_TK CP_TK
+               { 
+                 tree invok = build_method_invocation ($3, NULL_TREE);
+                 $$ = make_qualified_primary ($1, invok, $2.location);
+               }
+|      primary DOT_TK identifier OP_TK argument_list CP_TK
+               { 
+                 tree invok = build_method_invocation ($3, $5);
+                 $$ = make_qualified_primary ($1, invok, $2.location);
+               }
+|      SUPER_TK DOT_TK identifier OP_TK CP_TK
+               {
+                 tree invok;
+                 tree wfl = build_wfl_node (super_identifier_node, input_filename, 0, 0);
+                 EXPR_WFL_LINECOL (wfl) = $1.location;
+                 invok = build_method_invocation ($3, NULL_TREE);
+                 $$ = make_qualified_primary (wfl, invok, $2.location);
+               }
+|      SUPER_TK DOT_TK identifier OP_TK argument_list CP_TK
+               {
+                 tree invok;
+                 tree wfl = build_wfl_node (super_identifier_node, input_filename, 0, 0);
+                 EXPR_WFL_LINECOL (wfl) = $1.location;
+                 invok = build_method_invocation ($3, $5);
+                 $$ = make_qualified_primary (wfl, invok, $2.location);
+               }
+        /* Screws up thing. I let it here until I'm convinced it can
+           be removed. FIXME
+|      primary DOT_TK error
+               {yyerror ("'(' expected"); DRECOVER(bad);} */
+|      SUPER_TK DOT_TK error CP_TK
+               { yyerror ("'(' expected"); DRECOVER (method_invocation); }
+|      SUPER_TK DOT_TK error DOT_TK
+               { yyerror ("'(' expected"); DRECOVER (method_invocation); }
+;
+
+array_access:
+       name OSB_TK expression CSB_TK
+               { $$ = build_array_ref ($2.location, $1, $3); }
+|      primary_no_new_array OSB_TK expression CSB_TK
+               { $$ = build_array_ref ($2.location, $1, $3); }
+|      name OSB_TK error
+               {
+                 yyerror ("Missing term and ']' expected");
+                 DRECOVER(array_access);
+               }
+|      name OSB_TK expression error
+               {
+                 yyerror ("']' expected");
+                 DRECOVER(array_access);
+               }
+|      primary_no_new_array OSB_TK error
+               {
+                 yyerror ("Missing term and ']' expected");
+                 DRECOVER(array_access);
+               }
+|      primary_no_new_array OSB_TK expression error
+               {
+                 yyerror ("']' expected");
+                 DRECOVER(array_access);
+               }
+;
+
+postfix_expression:
+       primary
+|      name
+|      post_increment_expression
+|      post_decrement_expression
+;
+
+post_increment_expression:
+       postfix_expression INCR_TK
+               { $$ = build_incdec ($2.token, $2.location, $1, 1); }
+;
+
+post_decrement_expression:
+       postfix_expression DECR_TK
+               { $$ = build_incdec ($2.token, $2.location, $1, 1); }
+;
+
+unary_expression:
+       pre_increment_expression
+|      pre_decrement_expression
+|      PLUS_TK unary_expression
+               {$$ = build_unaryop ($1.token, $1.location, $2); }
+|      MINUS_TK unary_expression
+               {$$ = build_unaryop ($1.token, $1.location, $2); }
+|      unary_expression_not_plus_minus
+|      PLUS_TK error
+               {yyerror ("Missing term"); RECOVER}
+|      MINUS_TK error
+               {yyerror ("Missing term"); RECOVER}
+;
+
+pre_increment_expression:
+       INCR_TK unary_expression
+               {$$ = build_incdec ($1.token, $1.location, $2, 0); }
+|      INCR_TK error
+               {yyerror ("Missing term"); RECOVER}
+;
+
+pre_decrement_expression:
+       DECR_TK unary_expression
+               {$$ = build_incdec ($1.token, $1.location, $2, 0); }
+|      DECR_TK error
+               {yyerror ("Missing term"); RECOVER}
+;
+
+unary_expression_not_plus_minus:
+       postfix_expression
+|      NOT_TK unary_expression
+               {$$ = build_unaryop ($1.token, $1.location, $2); }
+|      NEG_TK unary_expression
+               {$$ = build_unaryop ($1.token, $1.location, $2); }
+|      cast_expression
+|       NOT_TK error
+               {yyerror ("Missing term"); RECOVER}
+|       NEG_TK error
+               {yyerror ("Missing term"); RECOVER}
+;
+
+cast_expression:               /* Error handling here is potentially weak */
+       OP_TK primitive_type dims CP_TK unary_expression
+               { 
+                 tree type = $2;
+                 while (ctxp->osb_number--)
+                   type = build_java_array_type (type, -1);
+                 $$ = build_cast ($1.location, type, $5); 
+               }
+|      OP_TK primitive_type CP_TK unary_expression
+               { $$ = build_cast ($1.location, $2, $4); }
+|      OP_TK expression CP_TK unary_expression_not_plus_minus
+               { $$ = build_cast ($1.location, $2, $4); }
+|      OP_TK name dims CP_TK unary_expression_not_plus_minus
+               { 
+                 char *ptr;
+                 while (ctxp->osb_number--)
+                   obstack_1grow (&temporary_obstack, '[');
+                 obstack_grow0 (&temporary_obstack, 
+                                IDENTIFIER_POINTER (EXPR_WFL_NODE ($2)),
+                                IDENTIFIER_LENGTH (EXPR_WFL_NODE ($2)));
+                 ptr = obstack_finish (&temporary_obstack);
+                 EXPR_WFL_NODE ($2) = get_identifier (ptr);
+                 $$ = build_cast ($1.location, $2, $5);
+               }
+|      OP_TK primitive_type OSB_TK error
+               {yyerror ("']' expected, invalid type expression");}
+|       OP_TK error
+               {
+                 if (ctxp->prevent_ese != lineno)
+                   yyerror ("Invalid type expression"); RECOVER;
+                 RECOVER;
+               }
+|      OP_TK primitive_type dims CP_TK error
+               {yyerror ("Missing term"); RECOVER;}
+|      OP_TK primitive_type CP_TK error
+               {yyerror ("Missing term"); RECOVER;}
+|      OP_TK name dims CP_TK error
+               {yyerror ("Missing term"); RECOVER;}
+;
+
+multiplicative_expression:
+       unary_expression
+|      multiplicative_expression MULT_TK unary_expression
+               { 
+                 $$ = build_binop (BINOP_LOOKUP ($2.token), 
+                                   $2.location, $1, $3);
+               }
+|      multiplicative_expression DIV_TK unary_expression
+               {
+                 $$ = build_binop (BINOP_LOOKUP ($2.token), $2.location,
+                                   $1, $3); 
+               }
+|      multiplicative_expression REM_TK unary_expression
+               {
+                 $$ = build_binop (BINOP_LOOKUP ($2.token), $2.location,
+                                   $1, $3); 
+               }
+|      multiplicative_expression MULT_TK error
+               {yyerror ("Missing term"); RECOVER;}
+|      multiplicative_expression DIV_TK error
+               {yyerror ("Missing term"); RECOVER;}
+|      multiplicative_expression REM_TK error
+               {yyerror ("Missing term"); RECOVER;}
+;
+
+additive_expression:
+       multiplicative_expression
+|      additive_expression PLUS_TK multiplicative_expression
+               {
+                 $$ = build_binop (BINOP_LOOKUP ($2.token), $2.location,
+                                   $1, $3); 
+               }
+|      additive_expression MINUS_TK multiplicative_expression
+               {
+                 $$ = build_binop (BINOP_LOOKUP ($2.token), $2.location,
+                                   $1, $3); 
+               }
+|      additive_expression PLUS_TK error
+               {yyerror ("Missing term"); RECOVER;}
+|      additive_expression MINUS_TK error
+               {yyerror ("Missing term"); RECOVER;}
+;
+
+shift_expression:
+       additive_expression
+|      shift_expression LS_TK additive_expression
+               {
+                 $$ = build_binop (BINOP_LOOKUP ($2.token), $2.location,
+                                   $1, $3); 
+               }
+|      shift_expression SRS_TK additive_expression
+               {
+                 $$ = build_binop (BINOP_LOOKUP ($2.token), $2.location,
+                                   $1, $3); 
+               }
+|      shift_expression ZRS_TK additive_expression
+               {
+                 $$ = build_binop (BINOP_LOOKUP ($2.token), $2.location,
+                                   $1, $3); 
+               }
+|      shift_expression LS_TK error
+               {yyerror ("Missing term"); RECOVER;}
+|      shift_expression SRS_TK error
+               {yyerror ("Missing term"); RECOVER;}
+|      shift_expression ZRS_TK error
+               {yyerror ("Missing term"); RECOVER;}
+;
+
+relational_expression:
+       shift_expression
+|      relational_expression LT_TK shift_expression
+               {
+                 $$ = build_binop (BINOP_LOOKUP ($2.token), $2.location,
+                                   $1, $3); 
+               }
+|      relational_expression GT_TK shift_expression
+               {
+                 $$ = build_binop (BINOP_LOOKUP ($2.token), $2.location,
+                                   $1, $3); 
+               }
+|      relational_expression LTE_TK shift_expression
+               {
+                 $$ = build_binop (BINOP_LOOKUP ($2.token), $2.location,
+                                   $1, $3); 
+               }
+|      relational_expression GTE_TK shift_expression
+               {
+                 $$ = build_binop (BINOP_LOOKUP ($2.token), $2.location,
+                                   $1, $3); 
+               }
+|      relational_expression INSTANCEOF_TK reference_type
+|      relational_expression LT_TK error
+               {yyerror ("Missing term"); RECOVER;}
+|      relational_expression GT_TK error
+               {yyerror ("Missing term"); RECOVER;}
+|      relational_expression LTE_TK error
+               {yyerror ("Missing term"); RECOVER;}
+|      relational_expression GTE_TK error
+               {yyerror ("Missing term"); RECOVER;}
+|      relational_expression INSTANCEOF_TK error
+               {yyerror ("Invalid reference type"); RECOVER;}
+;
+
+equality_expression:
+       relational_expression
+|      equality_expression EQ_TK relational_expression
+               {
+                 $$ = build_binop (BINOP_LOOKUP ($2.token), $2.location,
+                                   $1, $3); 
+               }
+|      equality_expression NEQ_TK relational_expression
+               {
+                 $$ = build_binop (BINOP_LOOKUP ($2.token), $2.location,
+                                   $1, $3); 
+               }
+|      equality_expression EQ_TK error
+               {yyerror ("Missing term"); RECOVER;}
+|      equality_expression NEQ_TK error
+               {yyerror ("Missing term"); RECOVER;}
+;
+
+and_expression:
+       equality_expression
+|      and_expression AND_TK equality_expression
+               {
+                 $$ = build_binop (BINOP_LOOKUP ($2.token), $2.location,
+                                   $1, $3); 
+               }
+|      and_expression AND_TK error
+               {yyerror ("Missing term"); RECOVER;}
+;
+
+exclusive_or_expression:
+       and_expression
+|      exclusive_or_expression XOR_TK and_expression
+               {
+                 $$ = build_binop (BINOP_LOOKUP ($2.token), $2.location,
+                                   $1, $3); 
+               }
+|      exclusive_or_expression XOR_TK error
+               {yyerror ("Missing term"); RECOVER;}
+;
+
+inclusive_or_expression:
+       exclusive_or_expression
+|      inclusive_or_expression OR_TK exclusive_or_expression
+               {
+                 $$ = build_binop (BINOP_LOOKUP ($2.token), $2.location,
+                                   $1, $3); 
+               }
+|      inclusive_or_expression OR_TK error
+               {yyerror ("Missing term"); RECOVER;}
+;
+
+conditional_and_expression:
+       inclusive_or_expression
+|      conditional_and_expression BOOL_AND_TK inclusive_or_expression
+               {
+                 $$ = build_binop (BINOP_LOOKUP ($2.token), $2.location,
+                                   $1, $3); 
+               }
+|      conditional_and_expression BOOL_AND_TK error
+               {yyerror ("Missing term"); RECOVER;}
+;
+
+conditional_or_expression:
+       conditional_and_expression
+|      conditional_or_expression BOOL_OR_TK conditional_and_expression
+               {
+                 $$ = build_binop (BINOP_LOOKUP ($2.token), $2.location,
+                                   $1, $3); 
+               }
+|      conditional_or_expression BOOL_OR_TK error
+               {yyerror ("Missing term"); RECOVER;}
+;
+
+conditional_expression:                /* Error handling here is weak */
+       conditional_or_expression
+|      conditional_or_expression REL_QM_TK expression REL_CL_TK conditional_expression
+|      conditional_or_expression REL_QM_TK REL_CL_TK error
+               {
+                 YYERROR_NOW;
+                 yyerror ("Missing term");
+                 DRECOVER (1);
+               }
+|      conditional_or_expression REL_QM_TK error
+               {yyerror ("Missing term"); DRECOVER (2);}
+|      conditional_or_expression REL_QM_TK expression REL_CL_TK error
+               {yyerror ("Missing term"); DRECOVER (3);}
+;
+
+assignment_expression:
+       conditional_expression
+|      assignment
+;
+
+assignment:
+       left_hand_side assignment_operator assignment_expression
+               { $$ = build_assignment ($2.token, $2.location, $1, $3); }
+|      left_hand_side assignment_operator error
+               {
+                 if (ctxp->prevent_ese != lineno)
+                   yyerror ("Missing term");
+                 DRECOVER (assign);
+               }
+;
+
+left_hand_side:
+       name
+|      field_access
+|      array_access
+;
+
+assignment_operator:
+       ASSIGN_ANY_TK
+|      ASSIGN_TK
+;
+
+expression:
+       assignment_expression
+;
+
+constant_expression:
+       expression
+;
+
+%%
+\f
+
+#include "lex.c"
+
+/* Flag for the error report routine to issue the error the first time
+   it's called (overriding the default behavior which is to drop the
+   first invocation and honor the second one, taking advantage of a
+   richer context.  */
+static int force_error = 0;
+
+/* Create a new parser context and make it the current one. */
+
+void
+java_push_parser_context ()
+{
+  struct parser_ctxt *new = 
+    (struct parser_ctxt *)malloc(sizeof (struct parser_ctxt));
+
+  bzero (new, sizeof (struct parser_ctxt));
+  new->next = ctxp;
+  ctxp = new;
+  if (ctxp->next)
+    ctxp->incomplete_class = ctxp->next->incomplete_class;
+}  
+
+void
+java_parser_context_save_global ()
+{
+  ctxp->finput = finput;
+  ctxp->lineno = lineno;
+  ctxp->current_class = current_class;
+  ctxp->filename = input_filename;
+  ctxp->current_function_decl = current_function_decl;
+}
+
+void
+java_parser_context_restore_global ()
+{
+  finput = ctxp->finput;
+  lineno = ctxp->lineno;
+  current_class = ctxp->current_class;
+  input_filename = ctxp->filename;
+  current_function_decl = ctxp->current_function_decl;
+}
+
+void 
+java_pop_parser_context ()
+{
+  tree current;
+  struct parser_ctxt *toFree = ctxp;
+  struct parser_ctxt *next = ctxp->next;
+
+  if (next)
+    {
+      next->incomplete_class = ctxp->incomplete_class;
+      lineno = ctxp->lineno;
+      finput = ctxp->finput;
+      current_class = ctxp->current_class;
+    }
+
+  /* Set the single import class file flag to 0 for the current list
+     of imported things */
+  for (current = ctxp->import_list; current; current = TREE_CHAIN (current))
+    IS_A_SINGLE_IMPORT_CLASSFILE_NAME_P (TREE_PURPOSE (current)) = 0;
+
+  /* And restore those of the previous context */
+  if (ctxp = next)
+    for (current = ctxp->import_list; current; current = TREE_CHAIN (current))
+      IS_A_SINGLE_IMPORT_CLASSFILE_NAME_P (TREE_PURPOSE (current)) = 1;
+
+  free (toFree);
+}
+
+static int do_warning = 0;
+
+void
+yyerror (msg)
+     char *msg;
+{
+  static java_lc elc;
+  static int  prev_lineno;
+  static char *prev_msg;
+
+  int i, save_lineno;
+  char *remainder, *code_from_source;
+  extern struct obstack temporary_obstack;
+  
+  if (!force_error && prev_lineno == lineno)
+    return;
+
+  /* Save current error location but report latter, when the context is
+     richer.  */
+  if (ctxp->java_error_flag == 0)
+    {
+      ctxp->java_error_flag = 1;
+      elc = ctxp->elc;
+      /* Do something to use the previous line if we're reaching the
+        end of the file... */
+#ifdef VERBOSE_SKELETON
+      printf ("* Error detected (%s)\n", (msg ? msg : "(null)"));
+#endif
+      return;
+    }
+
+  /* Ignore duplicate message on the same line. BTW, this is dubious. FIXME */
+  if (!force_error && msg == prev_msg && prev_lineno == elc.line)
+    return;
+
+  ctxp->java_error_flag = 0;
+  if (do_warning)
+    java_warning_count++;
+  else
+    java_error_count++;
+  
+  if (elc.col == 0 && msg[1] == ';')
+    {
+      elc.col  = ctxp->p_line->char_col-1;
+      elc.line = ctxp->p_line->lineno;
+    }
+
+  save_lineno = lineno;
+  prev_lineno = lineno = elc.line;
+  prev_msg = msg;
+
+  code_from_source = java_get_line_col (ctxp->filename, elc.line, elc.col);
+  obstack_grow0 (&temporary_obstack, 
+                code_from_source, strlen (code_from_source));
+  remainder = obstack_finish (&temporary_obstack);
+  if (do_warning)
+    warning ("%s.\n%s", msg, remainder);
+  else
+    error ("%s.\n%s", msg, remainder);
+
+  /* This allow us to cheaply avoid an extra 'Invalid expression
+     statement' error report when errors have been already reported on
+     the same line. This occurs when we report an error but don't have
+     a synchronization point other than ';', which
+     expression_statement is the only one to take care of.  */
+  ctxp->prevent_ese = lineno = save_lineno;
+}
+
+static void
+parse_error (msg)
+     char *msg;
+{
+  java_error (NULL);
+  java_error (msg);
+}
+
+/* Issue an error message at a current source line CL */
+
+static void
+parse_error_context VPROTO ((tree cl, char *msg, ...))
+{
+#ifndef __STDC__
+  tree cl;
+  char *msg;
+#endif
+  char buffer [4096];
+  va_list ap;
+
+  VA_START (ap, msg);
+#ifndef __STDC__
+  cl = va_arg (ap, tree);
+  msg = va_arg (ap, char *);
+#endif
+  vsprintf (buffer, msg, ap);
+
+  force_error = 1;
+  ctxp->elc.line = EXPR_WFL_LINENO (cl);
+  ctxp->elc.col  = (EXPR_WFL_COLNO (cl) == 0xfff ? -1 : EXPR_WFL_COLNO (cl));
+
+  parse_error (buffer);
+  force_error = 0;
+}
+
+/* Issue a warning at a current source line CL */
+
+static void
+parse_warning_context VPROTO ((tree cl, char *msg, ...))
+{
+#ifndef __STDC__
+  tree cl;
+  char *msg;
+#endif
+  char buffer [4096];
+  va_list ap;
+
+  VA_START (ap, msg);
+#ifndef __STDC__
+  cl = va_arg (ap, tree);
+  msg = va_arg (ap, char *);
+#endif
+  vsprintf (buffer, msg, ap);
+
+  force_error = do_warning = 1;
+  ctxp->elc.line = EXPR_WFL_LINENO (cl);
+  ctxp->elc.col  = (EXPR_WFL_COLNO (cl) == 0xfff ? -1 : EXPR_WFL_COLNO (cl));
+
+  parse_error (buffer);
+  do_warning = force_error = 0;
+}
+
+void
+java_report_errors ()
+{
+  if (java_error_count)
+    fprintf (stderr, "%d error%s", 
+            java_error_count, (java_error_count == 1 ? "" : "s"));
+  if (java_warning_count)
+    fprintf (stderr, "%s%d warning%s", (java_error_count ? ", " : ""),
+            java_warning_count, (java_warning_count == 1 ? "" : "s"));
+  if (java_error_count || java_warning_count)
+    putc ('\n', stderr);
+}
+
+static char *
+java_accstring_lookup (flags)
+     int flags;
+{
+  static char buffer [80];
+#define COPY_RETURN(S) {strcpy (buffer, S); return buffer;}
+
+  /* Access modifier looked-up first for easier report on forbidden
+     access. */
+  if (flags & ACC_PUBLIC) COPY_RETURN ("public");
+  if (flags & ACC_PRIVATE) COPY_RETURN ("private");
+  if (flags & ACC_PROTECTED) COPY_RETURN ("protected");
+  if (flags & ACC_STATIC) COPY_RETURN ("static");
+  if (flags & ACC_FINAL) COPY_RETURN ("final");
+  if (flags & ACC_SYNCHRONIZED) COPY_RETURN ("synchronized");
+  if (flags & ACC_VOLATILE) COPY_RETURN ("volatile");
+  if (flags & ACC_TRANSIENT) COPY_RETURN ("transient");
+  if (flags & ACC_NATIVE) COPY_RETURN ("native");
+  if (flags & ACC_INTERFACE) COPY_RETURN ("interface");
+  if (flags & ACC_ABSTRACT) COPY_RETURN ("abstract");
+
+  buffer [0] = '\0';
+  return buffer;
+#undef COPY_RETURN
+}
+
+static void
+redefinition_error (context, id, decl, cl)
+     char *context;
+     tree id, decl, cl;
+{
+  parse_error_context (cl, "%s `%s' already defined in %s:%d", 
+                      context, IDENTIFIER_POINTER (id), 
+                      DECL_SOURCE_FILE (decl), DECL_SOURCE_LINE (decl));
+  /* Here we should point out where its redefined. It's a unicode. FIXME */
+}
+
+/* Build something that the type identifier resolver will identify as
+   being an array to an unresolved type. TYPE_WFL is a WFL on a
+   identifier. */
+
+static tree
+build_unresolved_array_type (type_or_wfl)
+     tree type_or_wfl;
+{
+  char *ptr;
+
+  /* TYPE_OR_WFL might be an array on a primitive type. In this case,
+     just create a array type */
+  if (TREE_CODE (type_or_wfl) == RECORD_TYPE)
+    {
+      tree type = build_java_array_type (type_or_wfl, -1);
+      CLASS_LOADED_P (type) = CLASS_LOADED_P (type_or_wfl);
+      return type;
+    }
+
+  obstack_1grow (&temporary_obstack, '[');
+  obstack_grow0 (&temporary_obstack,
+                IDENTIFIER_POINTER (EXPR_WFL_NODE (type_or_wfl)),
+                IDENTIFIER_LENGTH (EXPR_WFL_NODE (type_or_wfl)));
+  ptr = obstack_finish (&temporary_obstack);
+  return build_expr_wfl (get_identifier (ptr),
+                        EXPR_WFL_FILENAME (type_or_wfl),
+                        EXPR_WFL_LINENO (type_or_wfl),
+                        EXPR_WFL_COLNO (type_or_wfl));
+}
+
+/* Check modifiers. If one doesn't fit, retrieve it in its declaration line
+  and point it out.  */
+
+static void
+check_modifiers (message, value, mask)
+     char *message;
+     int value;
+     int mask;
+{
+  /* Should point out the one that don't fit. ASCII/unicode,
+     going backward. FIXME */
+  if (value & ~mask)
+    {
+      int i, remainder = value & ~mask;
+      for (i = 0; i <= 10; i++)
+        if ((1 << i) & remainder)
+         parse_error_context (ctxp->modifier_ctx [i], message, 
+                              java_accstring_lookup (1 << i));
+    }
+}
+
+static void
+parser_add_interface (class_decl, interface_decl, wfl)
+     tree class_decl, interface_decl, wfl;
+{
+  if (maybe_add_interface (TREE_TYPE (class_decl), TREE_TYPE (interface_decl)))
+    parse_error_context (wfl, "Interface `%s' repeated",
+                        IDENTIFIER_POINTER (DECL_NAME (interface_decl)));
+}
+
+/* Bulk of common class/interface checks. Return 1 if an error was
+   encountered. TAG is 0 for a class, 1 for an interface.  */
+
+static int
+check_class_interface_creation (is_interface, flags, raw_name, qualified_name, decl, cl)
+     int is_interface, flags;
+     tree raw_name, qualified_name, decl, cl;
+{
+  tree node;
+
+  if (!quiet_flag)
+    fprintf (stderr, " %s %s", (is_interface ? "interface" : "class"), 
+            IDENTIFIER_POINTER (qualified_name));
+
+  /* Scope of an interface/class type name:
+       - Can't be imported by a single type import
+       - Can't already exists in the package */
+  if (IS_A_SINGLE_IMPORT_CLASSFILE_NAME_P (raw_name)
+      && (node = find_name_in_single_imports (raw_name)))
+    {
+      parse_error_context 
+       (cl, "%s name `%s' clashes with imported type `%s'",
+        (is_interface ? "Interface" : "Class"),
+        IDENTIFIER_POINTER (raw_name), IDENTIFIER_POINTER (node));
+      return 1;
+    }
+  if (decl && CLASS_COMPLETE_P (decl))
+    {
+      redefinition_error ((is_interface ? "Interface" : "Class"), 
+                         qualified_name, decl, cl);
+      return 1;
+    }
+
+  /* If public, file name should match class/interface name */
+  if (flags & ACC_PUBLIC)
+    {
+      char *f;
+
+      /* Contains OS dependent assumption on path separator. FIXME */
+      for (f = &input_filename [strlen (input_filename)]; 
+          f != input_filename && f[0] != '/'; f--);
+      if (f[0] == '/')
+       f++;
+      if (strncmp (IDENTIFIER_POINTER (raw_name), 
+                  f , IDENTIFIER_LENGTH (raw_name)) ||
+         f [IDENTIFIER_LENGTH (raw_name)] != '.')
+       parse_error_context (cl, "Public %s `%s' must be defined in a file "
+                            "called `%s.java'", 
+                            (is_interface ? "interface" : "class"),
+                            IDENTIFIER_POINTER (qualified_name),
+                            IDENTIFIER_POINTER (raw_name));
+    }
+
+  check_modifiers ((is_interface ? 
+                   "Illegal modifier `%s' for interface declaration" :
+                   "Illegal modifier `%s' for class declaration"), flags,
+                  (is_interface ? INTERFACE_MODIFIERS : CLASS_MODIFIERS));
+  return 0;
+}
+
+/* If DECL is NULL, create and push a new DECL, record the current
+   line CL and do other maintenance things.  */
+
+static tree
+maybe_create_class_interface_decl (decl, qualified_name, cl)
+     tree decl, qualified_name, cl;
+{
+  if (decl)
+    DECL_ARTIFICIAL (decl) = 1; /* FIXME */
+  else
+    decl = push_class (make_class (), qualified_name);
+  
+  /* Take care of the file and line business */
+  DECL_SOURCE_FILE (decl) = EXPR_WFL_FILENAME (cl);
+  DECL_SOURCE_LINE (decl) = EXPR_WFL_LINENO (cl);
+  CLASS_FROM_SOURCE_P (TREE_TYPE (decl)) = 1;
+
+  ctxp->current_parsed_class = decl;
+  
+  /* Link the declaration to the already seen ones */
+  TREE_CHAIN (decl) = ctxp->class_list;
+  ctxp->class_list = decl;
+  /* Install a new dependency list element */
+  create_jdep_list (ctxp);
+
+  SOURCE_FRONTEND_DEBUG (("Defining class/interface %s", 
+                         IDENTIFIER_POINTER (qualified_name)));
+  return decl;
+}
+
+static void
+add_superinterfaces (decl, interface_list)
+     tree decl, interface_list;
+{
+  tree node;
+  /* Superinterface(s): if present and defined, parser_check_super_interface ()
+     takes care of ensuring that:
+       - This is an accessible interface type,
+       - Circularity detection.
+   parser_add_interface is then called. If present but not defined,
+   the check operation is delayed until the super interface gets
+   defined.  */
+  for (node = interface_list; node; node = TREE_CHAIN (node))
+    {
+      tree current = TREE_PURPOSE (node), interface_decl;
+      if ((interface_decl = IDENTIFIER_CLASS_VALUE (EXPR_WFL_NODE (current))))
+       {
+         if (!parser_check_super_interface (interface_decl, decl, current))
+           parser_add_interface (decl, interface_decl, current);
+       }
+      else
+       register_incomplete_type (JDEP_INTERFACE,
+                                 current, decl, NULL_TREE);
+    }
+}
+
+/* Create an interface in pass1 and return its decl. Return the
+   interface's decl in pass 2.  */
+
+static tree
+create_interface (flags, id, super)
+     int flags;
+     tree id, super;
+{
+  int chk;
+  tree raw_name = EXPR_WFL_NODE (id);
+  tree q_name = parser_qualified_classname (id);
+  tree decl = IDENTIFIER_CLASS_VALUE (q_name);
+
+  EXPR_WFL_NODE (id) = q_name; /* Keep source location, even if refined. */
+
+  /* Basic checks: scope, redefinition, modifiers */ 
+  if (check_class_interface_creation (1, flags, raw_name, q_name, decl, id))
+    return NULL_TREE;
+
+  /* Interface modifiers check
+       - public/abstract allowed (already done at that point)
+       - abstract is obsolete (comes first, it's a warning, or should be)
+       - Can't use twice the same (checked in the modifier rule) */
+  if (flags & ACC_ABSTRACT)
+    parse_warning_context 
+      (MODIFIER_WFL (ABSTRACT_TK),
+       "Obsolete use of `abstract' modifier. Interface `%s' is implicitely "
+       "abstract", IDENTIFIER_POINTER (raw_name));
+  if (flags & ACC_PUBLIC && flags & ACC_ABSTRACT)
+    parse_error_context 
+      (MODIFIER_WFL (ABSTRACT_TK),
+       "Can't specify both `public' and `abstract' modifiers in the "
+       "definition of interface `%s'", IDENTIFIER_POINTER (raw_name));
+
+  /* Create a new decl if DECL is NULL, otherwise fix it */
+  decl = maybe_create_class_interface_decl (decl, q_name, id);
+
+  /* Set super info and mark the class a complete */
+  set_super_info (ACC_ABSTRACT | ACC_INTERFACE | flags, TREE_TYPE (decl), 
+                 object_type_node, ctxp->interface_number);
+  ctxp->interface_number = 0;
+  CLASS_COMPLETE_P (decl) = 1;
+  add_superinterfaces (decl, super);
+
+  return decl;
+}
+
+/* Create an class in pass1 and return its decl. Return class
+   interface's decl in pass 2.  */
+
+static tree
+create_class (flags, id, super, interfaces)
+     int flags;
+     tree id, super, interfaces;
+{
+  int chk;
+  tree raw_name = EXPR_WFL_NODE (id);
+  tree class_id, decl;
+  tree super_decl = NULL, super_decl_type;
+
+  class_id = parser_qualified_classname (id);
+  decl = IDENTIFIER_CLASS_VALUE (class_id);
+  EXPR_WFL_NODE (id) = class_id;
+
+  /* Basic check: scope, redefinition, modifiers */
+  if (check_class_interface_creation (0, flags, raw_name, class_id, decl, id))
+    return NULL_TREE;
+
+  /* Class modifier check: 
+       - Allowed modifier (already done at that point)
+       - abstract AND final forbidden 
+       - Public classes defined in the correct file */
+  if ((flags & ACC_ABSTRACT) && (flags & ACC_FINAL))
+    parse_error_context (id, "Class `%s' can't be declared both abstract "
+                        "and final", IDENTIFIER_POINTER (raw_name));
+
+  /* Create a new decl if DECL is NULL, otherwise fix it */
+  decl = maybe_create_class_interface_decl (decl, class_id, id);
+
+  /* If SUPER exists, use it, otherwise use Object */
+  if (super)
+    {
+      /* Can't extend java.lang.Object */
+      if (TREE_TYPE (IDENTIFIER_CLASS_VALUE (class_id)) == object_type_node)
+       {
+         parse_error_context (id, "Can't extend `java.lang.Object'");
+         return NULL_TREE;
+       }
+
+      /* The class is known and exists if there is a decl. Otherwise,
+         postpone the operation and do it later. */
+      super_decl = IDENTIFIER_CLASS_VALUE (EXPR_WFL_NODE (super));
+      if (super_decl)
+       {
+         parser_check_super (super_decl, decl, id);
+         super_decl_type = TREE_TYPE (super_decl);
+       }
+      else
+       super_decl_type = 
+         register_incomplete_type (JDEP_SUPER, super, decl, NULL_TREE);
+    }
+  else if (TREE_TYPE (decl) != object_type_node)
+    super_decl_type = object_type_node;
+  /* We're defining java.lang.Object */
+  else
+    super_decl_type = NULL_TREE;
+
+  /* Set super info and mark the class a complete */
+  set_super_info (flags, TREE_TYPE (decl), super_decl_type, 
+                 ctxp->interface_number);
+  ctxp->interface_number = 0;
+  CLASS_COMPLETE_P (decl) = 1;
+  add_superinterfaces (decl, interfaces);
+
+  return decl;
+}
+
+/* Can't use lookup_field () since we don't want to load the class and
+   can't set the CLASS_LOADED_P flag */
+
+static tree
+find_field (class, name)
+     tree class;
+     tree name;
+{
+  tree decl;
+  for (decl = TYPE_FIELDS (class); decl; decl = TREE_CHAIN (decl))
+    {
+      if (DECL_NAME (decl) == name)
+       return decl;
+    }
+  return NULL_TREE;
+}
+
+/* Wrap around lookup_field that doesn't potentially upset the value
+   of CLASS */
+
+static tree
+lookup_field_wrapper (class, name)
+     tree class, name;
+{
+  tree type = class;
+  return lookup_field (&type, name);
+}
+
+/* Find duplicate field within the same class declarations and report
+   the error */
+
+static int
+duplicate_declaration_error (class, new_field_name, new_type, cl)
+     tree class, new_field_name, new_type, cl;
+{
+  /* This might be modified to work with method decl as well */
+  tree decl = find_field (TREE_TYPE (ctxp->current_parsed_class), 
+                         new_field_name);
+  if (decl)
+    {
+      char *t1 = strdup ((char *)lang_printable_name (new_type, 1));
+      char *t2 = 
+       strdup ((TREE_CODE (TREE_TYPE (decl)) == TREE_LIST ?
+                IDENTIFIER_POINTER (TYPE_NAME 
+                                    (TREE_PURPOSE (TREE_TYPE (decl)))) :
+                (char *)lang_printable_name (TREE_TYPE (decl), 1)));
+      parse_error_context 
+       (cl , "Duplicate variable declaration: `%s %s' was `%s %s' (%s:%d)", 
+        t1, IDENTIFIER_POINTER (new_field_name),
+        t2, IDENTIFIER_POINTER (DECL_NAME (decl)),
+        DECL_SOURCE_FILE (decl), DECL_SOURCE_LINE (decl));
+      free (t1);
+      free (t2);
+      return 0;
+    }
+  return 1;
+}
+
+/* Field registration routine. If TYPE doesn't exist, field
+   declarations are linked to the undefined TYPE dependency list, to
+   be later resolved in java_complete_class () */
+
+static void
+register_fields (flags, type, variable_list)
+     int flags;
+     tree type, variable_list;
+{
+  tree current, type_decl, returned_type;
+  tree class_type = TREE_TYPE (ctxp->current_parsed_class);
+  int saved_lineno = lineno;
+  int must_chain = 0;
+  tree wfl = NULL_TREE;
+
+  /* If we're adding fields to interfaces, those fields are public,
+     static, final */
+  if (CLASS_INTERFACE (TYPE_NAME (class_type)))
+    {
+      OBSOLETE_MODIFIER_WARNING (MODIFIER_WFL (PUBLIC_TK),
+                                flags, ACC_PUBLIC, 
+                                "%s", "interface field(s)");
+      OBSOLETE_MODIFIER_WARNING (MODIFIER_WFL (STATIC_TK),
+                                flags, ACC_STATIC, 
+                                "%s", "interface field(s)");
+      OBSOLETE_MODIFIER_WARNING (MODIFIER_WFL (FINAL_TK),
+                                flags, ACC_FINAL, "%s", "interface field(s)");
+      check_modifiers ("Illegal interface member modifier `%s'", flags,
+                      INTERFACE_FIELD_MODIFIERS);
+      flags |= (ACC_PUBLIC | ACC_STATIC | ACC_FINAL);
+    }
+
+  if (unresolved_type_p (type, &returned_type))
+    {
+      if (returned_type)
+       type = returned_type;
+      else
+       {
+         wfl = type;
+         type = obtain_incomplete_type (type);
+         must_chain = 1;
+       }
+    }
+
+  for (current = variable_list; current; current = TREE_CHAIN (current))
+    {
+      tree cl = TREE_PURPOSE (current);
+      tree init = TREE_VALUE (current);
+      tree current_name = EXPR_WFL_NODE (cl);
+
+      if (duplicate_declaration_error (class_type, current_name, type, cl))
+        {
+         tree field_decl;
+         lineno = EXPR_WFL_LINENO (cl);
+         field_decl = add_field (class_type, current_name, type, flags);
+
+         /* Check if we must chain. */
+         if (must_chain)
+           register_incomplete_type (JDEP_FIELD, wfl, field_decl, type);
+         
+         /* Default value of a static field is 0 and it is considered
+             initialized. */
+         if (flags & ACC_STATIC)
+           INITIALIZED_P (field_decl) = 1;
+
+         /* If we have an initialization value tied to the field */
+         if (init)
+           {
+             /* The field is declared static */
+             if (flags & ACC_STATIC)
+               {
+                 /* FIXME */
+                 if (flags & ACC_FINAL)
+                   ;           
+                 /* Otherwise, the field should be initialized in
+                    <clinit>. This field is remembered so we can
+                    generate <clinit> later. */
+                 else
+                   {
+                     INITIALIZED_P (field_decl) = 1;
+                     TREE_CHAIN (init) = ctxp->static_initialized;
+                     ctxp->static_initialized = init;
+                   }
+               }
+             /* A non-static field declared with an immediate
+                initialization is to be initialized in <init>, if
+                any.  This field is remembered to be processed at the
+                time of the generation of <init>. */
+             else
+               {
+                 TREE_CHAIN (init) = ctxp->non_static_initialized;
+                 ctxp->non_static_initialized = init;
+               }
+           }
+       }
+    }
+  lineno = saved_lineno;
+}
+
+/* Check whether it is necessary to generate a <clinit> for the class
+   we just parsed. */
+
+static void
+maybe_generate_clinit ()
+{
+  int saved_lineno;
+  tree meth, mdecl, c;
+  tree cclass, class_wfl;
+
+  if (!ctxp->static_initialized || java_error_count)
+    return;
+
+  cclass = TREE_TYPE (ctxp->current_parsed_class);
+  class_wfl = build_expr_wfl (DECL_NAME (TYPE_NAME (cclass)),
+                             input_filename, 0, 0);
+  
+  saved_lineno = lineno;
+  lineno = 0;
+  meth = make_node (FUNCTION_TYPE);
+  TREE_TYPE (meth) = void_type_node;
+  TYPE_ARG_TYPES (meth) = NULL_TREE;
+  mdecl = add_method (cclass, ACC_STATIC, clinit_identifier_node,
+                    build_java_signature (meth));
+  lineno = saved_lineno;
+
+  DECL_SOURCE_LINE (mdecl) = 1;
+  DECL_SOURCE_LINE_MERGE (mdecl, 1);
+  source_start_java_method (mdecl);
+  enter_block ();
+
+  /* Keep initialization in order to enforce 8.5 */
+  ctxp->static_initialized = nreverse (ctxp->static_initialized);
+
+  /* We process the list of assignment we produced as the result of
+     the declaration of initialized static field and add them as
+     statement to the <clinit> method. */
+  for (c = ctxp->static_initialized; c; c = TREE_CHAIN (c))
+    {
+      /* We build the assignment expression that will initialize the
+        field to its value. There are strict rules on static
+        initializers (8.5). FIXME */
+      java_method_add_stmt (mdecl, c);
+    }
+
+  BLOCK_EXPR_BODY (DECL_FUNCTION_BODY (mdecl)) = exit_block ();
+  exit_block ();
+  ctxp->static_initialized = NULL_TREE;
+}
+
+/* Shared accros method_declarator and method_header to remember the
+   patch stage that was reached during the declaration of the method.
+   A method DECL is built differently is there is no patch
+   (JDEP_NO_PATCH) or a patch (JDEP_METHOD or JDEP_METHOD_RETURN)
+   pending on the currently defined method.  */
+
+static int patch_stage;
+
+/* Check the method declaration and add the method to its current
+   class.  If the argument list is known to contain incomplete types,
+   the method is partially added and the registration will be resume
+   once the method arguments resolved  */
+
+static tree
+method_header (flags, type, mdecl, throws)
+     int flags;
+     tree type, mdecl, throws;
+{
+  tree meth = TREE_VALUE (mdecl);
+  tree id = TREE_PURPOSE (mdecl);
+  tree this_class = TREE_TYPE (ctxp->current_parsed_class);
+  tree handle_class = CLASS_TO_HANDLE_TYPE (this_class);
+  tree meth_name, returned_type;
+  int saved_lineno;
+  
+  check_modifiers_consistency (flags);
+  
+  /* There are some forbidden modifiers for an abstract method and its
+     class must be abstract as well.  */
+  if (flags & ACC_ABSTRACT)
+    {
+      ABSTRACT_CHECK (flags, ACC_PRIVATE, id, "Private");
+      ABSTRACT_CHECK (flags, ACC_STATIC, id, "Static");
+      ABSTRACT_CHECK (flags, ACC_FINAL, id, "Final");
+      ABSTRACT_CHECK (flags, ACC_NATIVE, id, "Native");
+      ABSTRACT_CHECK (flags, ACC_SYNCHRONIZED,id, "Synchronized");
+      if (!CLASS_ABSTRACT (TYPE_NAME (this_class)))
+       parse_error_context 
+         (id, "Class `%s' must be declared abstract to define abstract "
+          "method `%s'", 
+          IDENTIFIER_POINTER (DECL_NAME (ctxp->current_parsed_class)),
+          IDENTIFIER_POINTER (EXPR_WFL_NODE (id)));
+    }
+
+
+  /* Method declared within the scope of an interface are implicitly
+     abstract and public. Conflicts with other erroneously provided
+     modifiers are check right after. */
+
+  if (CLASS_INTERFACE (TYPE_NAME (this_class)))
+    {
+      /* If FLAGS isn't set because of a modifier, turn the
+        corresponding modifier WFL to NULL so we issue a warning on
+        the obsolete use of the modifier */
+      if (!(flags & ACC_PUBLIC))
+        MODIFIER_WFL (PUBLIC_TK) = NULL;
+      if (!(flags & ACC_ABSTRACT))
+        MODIFIER_WFL (ABSTRACT_TK) = NULL;
+      flags |= ACC_PUBLIC;
+      flags |= ACC_ABSTRACT;
+    }
+
+  /* Modifiers context reset moved up, so abstract method declaration
+     modifiers can be later checked.  */
+
+  meth_name = EXPR_WFL_NODE (id);
+
+  if (unresolved_type_p (type, &returned_type))
+    {
+      if (returned_type)
+       TREE_TYPE (meth) = returned_type;
+      else 
+       {
+         patch_stage = JDEP_METHOD_RETURN;
+         TREE_TYPE (meth) = 
+           register_incomplete_type (patch_stage, type, id, NULL_TREE);
+       }
+    }
+  else
+    TREE_TYPE (meth) = type;
+
+  saved_lineno = lineno;
+  /* When defining an abstract or interface method, the curly
+     bracket at level 1 doesn't exist because there is no function
+     body */
+  lineno = (ctxp->first_ccb_indent1 ? ctxp->first_ccb_indent1 : 
+           EXPR_WFL_LINENO (id));
+
+  if (patch_stage)             /* includes ret type and/or all args */
+    {
+      jdep *jdep;
+      meth = add_method_1 (this_class, flags, meth_name, meth);
+      /* Patch for the return type */
+      if (patch_stage == JDEP_METHOD_RETURN)
+       {
+         jdep = CLASSD_LAST (ctxp->classd_list);
+         JDEP_GET_PATCH (jdep) = &TREE_TYPE (TREE_TYPE (meth));
+       }
+      /* This is the stop JDEP. METH allows the function's signature
+        to be computed. */
+      register_incomplete_type (JDEP_METHOD_END, NULL_TREE, meth, NULL_TREE);
+    }
+  else
+    {
+      tree signature = build_java_signature (meth);
+      tree arg, orig_arg;
+      /* Save original argument list, including argument's names */
+      orig_arg = TYPE_ARG_TYPES (meth);
+      /* Add the method to its class */
+      meth = add_method (this_class, flags, meth_name, signature);
+      /* Fix the method argument list so we have the argument name
+         information */
+      arg = TYPE_ARG_TYPES (TREE_TYPE (meth));
+      if (TREE_CODE (TREE_TYPE (meth)) == METHOD_TYPE)
+       {
+         TREE_PURPOSE (arg) = this_identifier_node;
+         arg = TREE_CHAIN (arg);
+       }
+      while (orig_arg)
+        {
+         TREE_PURPOSE (arg) = TREE_PURPOSE (orig_arg);
+         orig_arg = TREE_CHAIN (orig_arg);
+         arg = TREE_CHAIN (arg);
+       }
+    }
+  DECL_MAX_LOCALS (meth) = ctxp->formal_parameter_number+1;
+  lineno = saved_lineno;
+  /* We set the DECL_NAME to ID so we can track the location where
+     the function was declared. This allow us to report
+     redefinition error accurately. When method are verified,
+     DECL_NAME is reinstalled properly (using the content of the
+     WFL node ID) (see check_method_redefinition). We don't do that
+     when Object is being defined. */
+  if (TREE_TYPE (ctxp->current_parsed_class) != object_type_node)
+    DECL_NAME (meth) = id;
+  return meth;
+}
+
+/* Check modifiers that can be declared but exclusively */
+
+static void
+check_modifiers_consistency (flags)
+     int flags;
+{
+  int acc_count = 0;
+  tree cl = NULL_TREE;
+
+  THIS_MODIFIER_ONLY (flags, ACC_PUBLIC, 0, acc_count, cl);
+  THIS_MODIFIER_ONLY (flags, ACC_PRIVATE, 1, acc_count, cl);
+  THIS_MODIFIER_ONLY (flags, ACC_PROTECTED, 2, acc_count, cl);
+  if (acc_count > 1)
+    parse_error_context
+      (cl, "Inconsistent member declaration. At most one of `public', "
+       "`private', or `protected' may be specified");
+}
+
+/* Check the methode header METH for abstract specifics features */
+
+static void
+check_abstract_method_header (meth)
+     tree meth;
+{
+  int flags = get_access_flags_from_decl (meth);
+  /* DECL_NAME might still be a WFL node */
+  tree name = (TREE_CODE (DECL_NAME (meth)) == EXPR_WITH_FILE_LOCATION ?
+              EXPR_WFL_NODE (DECL_NAME (meth)) : DECL_NAME (meth));
+
+  OBSOLETE_MODIFIER_WARNING (MODIFIER_WFL (ABSTRACT_TK), flags,
+                            ACC_ABSTRACT, "abstract method `%s'",
+                            IDENTIFIER_POINTER (name));
+  OBSOLETE_MODIFIER_WARNING (MODIFIER_WFL (PUBLIC_TK), flags, 
+                            ACC_PUBLIC, "abstract method `%s'",
+                            IDENTIFIER_POINTER (name));
+
+  check_modifiers ("Illegal modifier `%s' for interface method",
+                 flags, INTERFACE_METHOD_MODIFIERS);
+}
+
+/* Create a FUNCTION_TYPE node and start augmenting it with the
+   declared function arguments. Arguments type that can't be resolved
+   are left as they are, but the returned node is marked as containing
+   incomplete types.  */
+
+static tree
+method_declarator (id, list)
+     tree id, list;
+{
+  tree arg_types = NULL_TREE, current, node;
+  tree meth = make_node (FUNCTION_TYPE);
+  jdep *jdep;
+  int incomplete = 0;
+
+  patch_stage = JDEP_NO_PATCH;
+  
+  for (current = list; current; current = TREE_CHAIN (current))
+    {
+      tree wfl_name = TREE_PURPOSE (current);
+      tree type = TREE_VALUE (current);
+      tree name = EXPR_WFL_NODE (wfl_name);
+      tree patchable_type = NULL_TREE, already;
+      tree arg_node, returned_type;
+
+      /* Check redefinition */
+      for (already = arg_types; already; already = TREE_CHAIN (already))
+       if (TREE_PURPOSE (already) == name)
+         {
+           parse_error_context 
+             (wfl_name, "Variable `%s' is used more than once in the "
+              "argument list of method `%s'", IDENTIFIER_POINTER (name),
+              IDENTIFIER_POINTER (EXPR_WFL_NODE (id)));
+           break;
+         }
+
+      /* If we've an incomplete argument type, we know there is a location
+        to patch when the type get resolved, later.  */
+      jdep = NULL;
+      if (unresolved_type_p (type, &returned_type))
+       {
+         if (returned_type)
+           type = returned_type;
+         else
+           {
+             patch_stage = JDEP_METHOD;
+             type = register_incomplete_type (patch_stage, type, 
+                                              wfl_name, NULL_TREE);
+             jdep = CLASSD_LAST (ctxp->classd_list);
+             JDEP_MISC (jdep) = id;
+           }
+       }
+      /* The argument node: a name and a (possibly) incomplete type */
+      arg_node = build_tree_list (name, type);
+      if (jdep)
+       JDEP_GET_PATCH (jdep) = &TREE_VALUE (arg_node);
+      TREE_CHAIN (arg_node) = arg_types;
+      arg_types = arg_node;
+    }
+  TYPE_ARG_TYPES (meth) = nreverse (arg_types);
+  node = build_tree_list (id, meth);
+  return node;
+}
+
+static int
+unresolved_type_p (wfl, returned)
+     tree wfl;
+     tree *returned;
+     
+{
+  if (TREE_CODE (wfl) == EXPR_WITH_FILE_LOCATION)
+    {
+      tree decl = IDENTIFIER_CLASS_VALUE (EXPR_WFL_NODE (wfl));
+      if (returned)
+       *returned = (decl ? TREE_TYPE (decl) : NULL_TREE);
+      return 1;
+    }
+  if (returned)
+    *returned = wfl;
+  return 0;
+}
+
+/* From NAME, build a qualified identifier node using the
+   qualification from the current package definition. */
+
+static tree
+parser_qualified_classname (name)
+     tree name;
+{
+  if (ctxp->package)
+    return merge_qualified_name (ctxp->package, EXPR_WFL_NODE (name));
+  else 
+    return EXPR_WFL_NODE (name);
+}
+
+/* Called once the type a interface extends is resolved. Returns 0 if
+   everything is OK.  */
+
+static int
+parser_check_super_interface (super_decl, this_decl, this_wfl)
+     tree super_decl, this_decl, this_wfl;
+{
+  tree super_type = TREE_TYPE (super_decl);
+
+  /* Has to be an interface */
+  if (!CLASS_INTERFACE (TYPE_NAME (TREE_TYPE (super_decl))))
+    {
+      parse_error_context 
+       (this_wfl, "Can't use %s `%s' to implement/extend %s `%s'",
+        (TYPE_ARRAY_P (super_type) ? "array" : "class"),
+        IDENTIFIER_POINTER (DECL_NAME (super_decl)),
+        (CLASS_INTERFACE (TYPE_NAME (TREE_TYPE (this_decl))) ? 
+         "interface" : "class"),
+        IDENTIFIER_POINTER (DECL_NAME (this_decl)));
+      return 1;
+    }
+
+  /* Check scope: same package OK, other package: OK if public */
+  if (check_pkg_class_access (DECL_NAME (super_decl), lookup_cl (this_decl)))
+    return 1;
+
+  SOURCE_FRONTEND_DEBUG (("Completing interface %s with %s",
+                         IDENTIFIER_POINTER (DECL_NAME (this_decl)),
+                         IDENTIFIER_POINTER (DECL_NAME (super_decl))));
+  return 0;
+}
+
+/* Makes sure that SUPER_DECL is suitable to extend THIS_DECL. Returns
+   0 if everthing is OK.  */
+
+static int
+parser_check_super (super_decl, this_decl, wfl)
+     tree super_decl, this_decl, wfl;
+{
+  tree this_type  = TREE_TYPE (this_decl);
+  tree super_type = TREE_TYPE (super_decl);
+
+  /* SUPER should be a CLASS (neither an array nor an interface) */
+  if (TYPE_ARRAY_P (super_type) || CLASS_INTERFACE (TYPE_NAME (super_type)))
+    {
+      parse_error_context 
+       (wfl, "Class `%s' can't subclass %s `%s'",
+        IDENTIFIER_POINTER (DECL_NAME (this_decl)),
+        (CLASS_INTERFACE (TYPE_NAME (super_type)) ? "interface" : "array"),
+        IDENTIFIER_POINTER (DECL_NAME (super_decl)));
+      return 1;
+    }
+
+  if (CLASS_FINAL (TYPE_NAME (super_type)))
+    {
+      parse_error_context (wfl, "Can't subclass final classes: %s",
+                          IDENTIFIER_POINTER (DECL_NAME (super_decl)));
+      return 1;
+    }
+
+  /* Check scope: same package OK, other package: OK if public */
+  if (check_pkg_class_access (DECL_NAME (super_decl), wfl))
+    return 1;
+  
+  SOURCE_FRONTEND_DEBUG (("Completing class %s with %s",
+                         IDENTIFIER_POINTER (DECL_NAME (this_decl)),
+                         IDENTIFIER_POINTER (DECL_NAME (super_decl))));
+  return 0;
+}
+
+/* Create a new dependency list and link it (in a LIFO manner) to the
+   CTXP list of type dependency list.  */
+
+static void
+create_jdep_list (ctxp)
+     struct parser_ctxt *ctxp;
+{
+  jdeplist *new = malloc (sizeof (jdeplist));  
+  
+  if (!new)
+    fatal ("Can't alloc jdeplist - create_jdep_list");
+    
+  new->first = new->last = NULL;
+  new->next = ctxp->classd_list;
+  ctxp->classd_list = new;
+}
+
+static jdeplist *
+reverse_jdep_list (ctxp)
+     struct parser_ctxt *ctxp;
+{
+  register jdeplist *prev = NULL, *current, *next;
+  for (current = ctxp->classd_list; current; current = next)
+    {
+      next = current->next;
+      current->next = prev;
+      prev = current;
+    }
+  return prev;
+}
+
+/* Create a fake pointer based on the ID stored in the WFL */
+
+static tree
+obtain_incomplete_type (wfl)
+     tree wfl;
+{
+  tree ptr;
+  tree name = EXPR_WFL_NODE (wfl);
+
+  for (ptr = ctxp->incomplete_class; ptr; ptr = TREE_CHAIN (ptr))
+    if (TYPE_NAME (TREE_PURPOSE (ptr)) == name)
+      break;
+
+  if (!ptr)
+    {
+      tree core;
+      push_obstacks (&permanent_obstack, &permanent_obstack);
+      BUILD_PTR_FROM_NAME (core, name);
+      ptr = build_tree_list (core, NULL_TREE);
+      pop_obstacks ();
+      TREE_CHAIN (ptr) = ctxp->incomplete_class;
+      ctxp->incomplete_class = ptr;
+    }
+
+  return ptr;
+}
+
+/* Register a incomplete type whose name is WFL. Reuse PTR if PTR is
+   non NULL instead of computing a new fake type based on WFL. The new
+   dependency is inserted in the current type dependency list, in FIFO
+   manner.  */
+
+static tree
+register_incomplete_type (kind, wfl, decl, ptr)
+     int kind;
+     tree wfl, decl, ptr;
+{
+  jdep *new = malloc (sizeof (jdep));
+
+  if (!new)
+    fatal ("Can't allocate new jdep - register_incomplete_type");
+  if (!ptr && kind != JDEP_METHOD_END) /* JDEP_METHOD_END is a mere marker */
+    ptr = obtain_incomplete_type (wfl);
+
+  JDEP_KIND (new) = kind;
+  JDEP_DECL (new) = decl;
+  JDEP_SOLV (new) = ptr;
+  JDEP_WFL (new) = wfl;
+  JDEP_CHAIN (new) = NULL;
+  JDEP_MISC (new) = NULL_TREE;
+  JDEP_GET_PATCH (new) = (tree *)NULL;
+
+  JDEP_INSERT (ctxp->classd_list, new);
+
+  return ptr;
+}
+
+void
+java_check_circular_reference ()
+{
+  tree current;
+  for (current = ctxp->class_list; current; current = TREE_CHAIN (current))
+    {
+      tree type = TREE_TYPE (current);
+      if (CLASS_INTERFACE (TYPE_NAME (type)))
+       {
+         /* Check all interfaces this class extends */
+         tree basetype_vec = TYPE_BINFO_BASETYPES (type);
+         int n, i;
+
+         if (!basetype_vec)
+           return;
+         n = TREE_VEC_LENGTH (basetype_vec);
+         for (i = 0; i < n; i++)
+           {
+             tree vec_elt = TREE_VEC_ELT (basetype_vec, i);
+             if (vec_elt && BINFO_TYPE (vec_elt) != object_type_node 
+                 && interface_of_p (type, BINFO_TYPE (vec_elt)))
+               parse_error_context (lookup_cl (current),
+                                    "Cyclic interface inheritance");
+           }
+       }
+      else
+       if (inherits_from_p (CLASSTYPE_SUPER (type), type))
+         parse_error_context (lookup_cl (current), 
+                              "Cyclic class inheritance");
+    }
+}
+
+void
+safe_layout_class (class)
+     tree class;
+{
+  tree save_current_class = current_class;
+  char *save_input_filename = input_filename;
+  int save_lineno = lineno;
+  
+  push_obstacks (&permanent_obstack, &permanent_obstack);
+  layout_class (class);
+  pop_obstacks ();
+  
+  current_class = save_current_class;
+  input_filename = save_input_filename;
+  lineno = save_lineno;
+  CLASS_LOADED_P (class) = 1;
+}
+
+static tree
+jdep_resolve_class (dep)
+     jdep *dep;
+{
+  tree decl;
+
+  if (!JDEP_RESOLVED_P (dep))
+    {
+      decl = 
+       resolve_class (JDEP_TO_RESOLVE (dep), JDEP_DECL (dep), JDEP_WFL (dep));
+      JDEP_RESOLVED (dep, decl);
+    }
+  else
+    decl = JDEP_RESOLVED_DECL (dep);
+
+  if (!decl)
+    {
+      complete_class_report_errors (dep);
+      return NULL_TREE;
+    }
+  return decl;
+}
+
+/* Complete unsatisfied class declaration and their dependencies */
+
+void
+java_complete_class ()
+{
+  tree current;
+  tree cclass;
+  jdeplist *cclassd;
+  int error_found;
+
+  push_obstacks (&permanent_obstack, &permanent_obstack);
+
+  /* Process imports and reverse the import on demand list */
+  process_imports ();
+  if (ctxp->import_demand_list)
+    ctxp->import_demand_list = nreverse (ctxp->import_demand_list);
+
+  /* Rever things so we have the right order */
+  ctxp->class_list = nreverse (ctxp->class_list);
+  ctxp->classd_list = reverse_jdep_list (ctxp);
+    
+  for (cclassd = ctxp->classd_list, cclass = ctxp->class_list; 
+       cclass && cclassd; 
+       cclass = TREE_CHAIN (cclass), cclassd = CLASSD_CHAIN (cclassd))
+    {
+      jdep *dep;
+      for (dep = CLASSD_FIRST (cclassd); dep; dep = JDEP_CHAIN (dep))
+       {
+         tree decl;
+
+         if (!(decl = jdep_resolve_class (dep)))
+           continue;
+
+         /* Now it's time to patch */
+         switch (JDEP_KIND (dep))
+           {
+           case JDEP_SUPER:
+             /* Simply patch super */
+             if (parser_check_super (decl, JDEP_DECL (dep), JDEP_WFL (dep)))
+               continue;
+             BINFO_TYPE (TREE_VEC_ELT (BINFO_BASETYPES (TYPE_BINFO 
+               (TREE_TYPE (JDEP_DECL (dep)))), 0)) = TREE_TYPE (decl);
+             break;
+
+           case JDEP_FIELD:
+             {
+               /* We do part of the job done in add_field */
+               tree field_decl = JDEP_DECL (dep);
+               tree field_type = TREE_TYPE (decl);
+               push_obstacks (&permanent_obstack, &permanent_obstack);
+#if ! JAVA_PROMOTE_TO_INT
+               if (TREE_CODE (field_type) == RECORD_TYPE)
+#endif
+                 field_type = promote_type (field_type);
+               pop_obstacks ();
+               TREE_TYPE (field_decl) = field_type;
+               SOURCE_FRONTEND_DEBUG 
+                 (("Completed field/var decl `%s' with `%s'",
+                   IDENTIFIER_POINTER (DECL_NAME (field_decl)),
+                   IDENTIFIER_POINTER (DECL_NAME (decl))));
+               break;
+             }
+           case JDEP_METHOD:   /* We start patching a method */
+           case JDEP_METHOD_RETURN:
+             error_found = 0;
+             while (1)
+               {
+                 if (decl)
+                   {
+                     tree type = promote_type (TREE_TYPE(decl));
+                     JDEP_APPLY_PATCH (dep, type);
+                     SOURCE_FRONTEND_DEBUG 
+                       (((JDEP_KIND (dep) == JDEP_METHOD_RETURN ?
+                          "Completing fct `%s' with ret type `%s'":
+                          "Completing arg `%s' with type `%s'"),
+                         IDENTIFIER_POINTER (EXPR_WFL_NODE 
+                                             (JDEP_DECL_WFL (dep))),
+                         IDENTIFIER_POINTER (DECL_NAME (decl))));
+                   }
+                 else
+                   error_found = 1;
+                 dep = JDEP_CHAIN (dep);
+                 if (JDEP_KIND (dep) == JDEP_METHOD_END)
+                   break;
+                 else
+                   decl = jdep_resolve_class (dep);
+               }
+             if (!error_found)
+               {
+                 tree mdecl = JDEP_DECL (dep), signature;
+                 push_obstacks (&permanent_obstack, &permanent_obstack);
+                 /* Recompute and reset the signature */
+                 signature = build_java_signature (TREE_TYPE (mdecl));
+                 set_java_signature (TREE_TYPE (mdecl), signature);
+                 pop_obstacks ();
+               }
+             else
+               continue;
+             break;
+
+           case JDEP_INTERFACE:
+             if (parser_check_super_interface (decl, JDEP_DECL (dep),
+                                               JDEP_WFL (dep)))
+               continue;
+             parser_add_interface (JDEP_DECL (dep), decl, JDEP_WFL (dep));
+             break;
+
+           case JDEP_VARIABLE:
+             JDEP_APPLY_PATCH (dep, promote_type (TREE_TYPE (decl)));
+             SOURCE_FRONTEND_DEBUG 
+               (("Completing variable `%s' with type `%s'",
+                 (TREE_CODE (JDEP_DECL_WFL (dep)) == EXPR_WITH_FILE_LOCATION ?
+                  IDENTIFIER_POINTER (EXPR_WFL_NODE (JDEP_DECL_WFL (dep))) :
+                  IDENTIFIER_POINTER (DECL_NAME (JDEP_DECL_WFL (dep)))),
+                 IDENTIFIER_POINTER (DECL_NAME (decl))));
+             break;
+
+           case JDEP_TYPE:
+             JDEP_APPLY_PATCH (dep, TREE_TYPE (decl));
+             SOURCE_FRONTEND_DEBUG 
+               (("Completing a random type dependency on a '%s' node",
+                 tree_code_name [TREE_CODE (JDEP_DECL (dep))]));
+             break;
+
+           case JDEP_PARM:
+             JDEP_APPLY_PATCH (dep, promote_type (TREE_TYPE (decl)));
+             SOURCE_FRONTEND_DEBUG 
+               (("Completing parameter `%s' with type `%s'",
+                 IDENTIFIER_POINTER (JDEP_MISC (dep)),
+                 IDENTIFIER_POINTER (DECL_NAME (decl))));
+             break;
+
+           default:
+             fatal ("incomplete switch - java_complete_class");
+           }
+       }
+    }
+  pop_obstacks ();
+  return;
+}
+
+/* Resolve class CLASS_TYPE. Handle the case of trying to resolve an
+   array.  */
+
+static tree
+resolve_class (class_type, decl, cl)
+     tree class_type, decl, cl;
+{
+  char *name = IDENTIFIER_POINTER (TYPE_NAME (class_type));
+  char *base = name;
+  tree resolved_type, resolved_type_decl;
+  
+  /* 1- Check to see if we have an array. If true, find what we really
+     want to resolve  */
+  while (name[0] == '[')
+    name++;
+  if (base != name)
+    TYPE_NAME (class_type) = get_identifier (name);
+
+  /* 2- Resolve the bare type */
+  if (!(resolved_type_decl = do_resolve_class (class_type, decl, cl)))
+    return NULL_TREE;
+  resolved_type = TREE_TYPE (resolved_type_decl);
+
+  /* 3- If we have and array, reconstruct the array down to its nesting */
+  if (base != name)
+    {
+      while (base != name)
+       {
+         if (TREE_CODE (resolved_type) == RECORD_TYPE)
+           resolved_type  = promote_type (resolved_type);
+         resolved_type = build_java_array_type (resolved_type, -1);
+         name--;
+       }
+      /* Build a fake decl for this, since this is what is expected to
+         be returned.  */
+      resolved_type_decl =
+       build_decl (TYPE_DECL, TYPE_NAME (resolved_type), resolved_type);
+      /* Figure how those two things are important for error report. FIXME */
+      DECL_SOURCE_LINE (resolved_type_decl) = 0;
+      DECL_SOURCE_FILE (resolved_type_decl) = input_filename;
+    }
+  return resolved_type_decl;
+}
+
+/* Effectively perform the resolution of class CLASS_TYPE. DECL or CL
+   are used to report error messages.  */
+
+static tree
+do_resolve_class (class_type, decl, cl)
+     tree class_type;
+     tree decl;
+     tree cl;
+{
+  tree new_class_decl;
+  tree original_name = NULL_TREE;
+
+  /* Do not try to replace TYPE_NAME (class_type) by a variable, since
+     its is changed by find_in_imports{_on_demand} */
+
+  /* 1- Check for the type in single imports */
+  if (find_in_imports (class_type))
+    return NULL_TREE;
+
+  /* 2- And check for the type in the current compilation unit. If it fails,
+     try with a name qualified with the package name if appropriate. */
+
+  if ((new_class_decl = IDENTIFIER_CLASS_VALUE (TYPE_NAME (class_type))))
+    {
+      if (!CLASS_LOADED_P (TREE_TYPE (new_class_decl)) &&
+         !CLASS_FROM_SOURCE_P (TREE_TYPE (new_class_decl)))
+       load_class (TYPE_NAME (class_type), 0);
+      return IDENTIFIER_CLASS_VALUE (TYPE_NAME (class_type));
+    }
+
+  original_name = TYPE_NAME (class_type);
+  if (!QUALIFIED_P (TYPE_NAME (class_type)) && ctxp->package)
+    TYPE_NAME (class_type) = merge_qualified_name (ctxp->package, 
+                                                  TYPE_NAME (class_type));
+  if ((new_class_decl = IDENTIFIER_CLASS_VALUE (TYPE_NAME (class_type))))
+    {
+      if (!CLASS_LOADED_P (TREE_TYPE (new_class_decl)) &&
+         !CLASS_FROM_SOURCE_P (TREE_TYPE (new_class_decl)))
+       load_class (TYPE_NAME (class_type), 0);
+      return IDENTIFIER_CLASS_VALUE (TYPE_NAME (class_type));
+    }
+  TYPE_NAME (class_type) = original_name;
+
+  /* 3- Check an other compilation unit that bears the name of type */
+  load_class (TYPE_NAME (class_type), 0);
+  if (check_pkg_class_access (TYPE_NAME (class_type), 
+                             (cl ? cl : lookup_cl (decl))))
+    return NULL_TREE;
+
+  if ((new_class_decl = IDENTIFIER_CLASS_VALUE (TYPE_NAME (class_type))))
+    return new_class_decl;
+
+  /* 4- Check the import on demands. Don't allow bar.baz to be
+     imported from foo.* */
+  if (!QUALIFIED_P (TYPE_NAME (class_type)))
+    if (find_in_imports_on_demand (class_type))
+      return NULL_TREE;
+
+  /* 5- Last call for a resolution */
+  return IDENTIFIER_CLASS_VALUE (TYPE_NAME (class_type));
+}
+
+/* Resolve NAME and lay it out (if not done and if not the current
+   parsed class). Return a decl node.  */
+
+static tree
+resolve_and_layout (name, cl)
+     tree name;
+     tree cl;
+{
+  tree decl = resolve_no_layout (name, cl);
+  if (decl && TREE_TYPE (decl) != current_class 
+      && !CLASS_LOADED_P (TREE_TYPE (decl)))
+    safe_layout_class (TREE_TYPE (decl));
+  return decl;
+}
+
+/* Resolve a class, returns its decl but doesn't perform any
+   layout. The current parsing context is saved and restored */
+
+static tree
+resolve_no_layout (name, cl)
+     tree name, cl;
+{
+  tree ptr, decl;
+  BUILD_PTR_FROM_NAME (ptr, name);
+  java_parser_context_save_global ();
+  decl = resolve_class (ptr, NULL_TREE, cl);
+  java_parser_context_restore_global ();
+  
+  return decl;
+}
+
+/* Called to report errors. Skip leader '[' in a complex array type
+   description that failed to be resolved. */
+
+static char *
+purify_type_name (name)
+     char *name;
+{
+  while (*name && *name == '[')
+    name++;
+  return name;
+}
+
+/* The type CURRENT refers to can't be found. We print error messages.  */
+
+static void
+complete_class_report_errors (dep)
+     jdep *dep;
+{
+  switch (JDEP_KIND (dep))
+    {
+    case JDEP_SUPER:
+      parse_error_context  
+       (JDEP_WFL (dep), "Superclass `%s' of class `%s' not found",
+        IDENTIFIER_POINTER (EXPR_WFL_NODE (JDEP_WFL (dep))),
+        IDENTIFIER_POINTER (DECL_NAME (JDEP_DECL (dep))));
+      break;
+    case JDEP_FIELD:
+      parse_error_context
+       (JDEP_WFL (dep), "Type `%s' not found in declaration of field `%s'",
+        IDENTIFIER_POINTER (EXPR_WFL_NODE (JDEP_WFL (dep))),
+        IDENTIFIER_POINTER (DECL_NAME (JDEP_DECL (dep))));
+      break;
+    case JDEP_METHOD:          /* Covers arguments */
+      parse_error_context
+       (JDEP_WFL (dep), "Type `%s' not found in the declaration of the "
+        "argument `%s' of method `%s'",
+        IDENTIFIER_POINTER (EXPR_WFL_NODE (JDEP_WFL (dep))),
+        IDENTIFIER_POINTER (EXPR_WFL_NODE (JDEP_DECL_WFL (dep))),
+        IDENTIFIER_POINTER (EXPR_WFL_NODE (JDEP_MISC (dep))));
+      break;
+    case JDEP_METHOD_RETURN:   /* Covers return type */
+      parse_error_context
+       (JDEP_WFL (dep), "Type `%s' not found in the declaration of the "
+        "return type of method `%s'", 
+        IDENTIFIER_POINTER (EXPR_WFL_NODE (JDEP_WFL (dep))),
+        IDENTIFIER_POINTER (EXPR_WFL_NODE (JDEP_DECL_WFL (dep))));
+      break;
+    case JDEP_INTERFACE:
+      parse_error_context
+       (JDEP_WFL (dep), "Superinterface `%s' of %s `%s' not found",
+        IDENTIFIER_POINTER (EXPR_WFL_NODE (JDEP_WFL (dep))),
+        (CLASS_OR_INTERFACE (JDEP_DECL (dep), "class", "interface")),
+        IDENTIFIER_POINTER (DECL_NAME (JDEP_DECL (dep))));
+      break;
+    case JDEP_VARIABLE:
+      parse_error_context
+       (JDEP_WFL (dep), "Type `%s' not found in the declaration of the "
+        "local variable `%s'", 
+        purify_type_name (IDENTIFIER_POINTER (EXPR_WFL_NODE (JDEP_WFL (dep)))),
+        IDENTIFIER_POINTER (DECL_NAME (JDEP_DECL (dep))));
+      break;
+    }
+}
+
+/* Check uninitialized final.  */
+
+void
+java_check_final ()
+{
+}
+
+static int
+check_method_redefinition (class, method)
+     tree class, method;
+{
+  tree redef, name;
+  tree cl = DECL_NAME (method);
+  tree sig = TYPE_LANG_SPECIFIC (TREE_TYPE (method))->signature;
+  /* decl name of generated <clinit> doesn't need to be fixed and
+     checked */
+  if (DECL_NAME (method) != clinit_identifier_node)
+    {
+      /* NAME is just the plain name when Object is being defined */
+      if (class != object_type_node)
+       name = DECL_NAME (method) = EXPR_WFL_NODE (DECL_NAME (method));
+      else
+       name = DECL_NAME (method);
+    }
+  else 
+    return 0;
+  
+  for (redef = TYPE_METHODS (class); redef; redef = TREE_CHAIN (redef))
+    {
+      struct lang_type *t = TYPE_LANG_SPECIFIC (TREE_TYPE (redef));
+      
+      if (! t || (redef == method))
+       break;
+      if (DECL_NAME (redef) == name && sig == t->signature)
+       {
+         parse_error_context (cl, "Duplicate method declaration");
+         return 1;
+       }
+    }
+  return 0;
+}
+
+/* Check all the methods of CLASS. Methods are first completed then
+   checked according to regular method existance rules.
+   If no constructor were encountered, then build its declaration. */
+
+static void
+java_check_regular_methods (class_decl)
+     tree class_decl;
+{
+  tree method;
+  tree class = CLASS_TO_HANDLE_TYPE (TREE_TYPE (class_decl));
+  tree super_class = CLASSTYPE_SUPER (class);
+  int seen_constructor = 0;
+
+  TYPE_METHODS (class) = nreverse (TYPE_METHODS (class));
+
+  /* Should take interfaces into account. FIXME */
+  for (method = TYPE_METHODS (class); method; method = TREE_CHAIN (method))
+    {
+      tree found, sig;
+      tree method_wfl = DECL_NAME (method);
+      int aflags;
+
+      if (DECL_CONSTRUCTOR_P (method))
+       seen_constructor = 1;
+
+      /* Check for redefinitions */
+      if (check_method_redefinition (class, method))
+       continue;
+
+      sig = build_java_argument_signature (TREE_TYPE (method));
+
+      found = lookup_argument_method (super_class, DECL_NAME (method), sig);
+      if (! found)
+        continue;
+      /* Can't override a method with the same name and different return
+        types. */
+      if (TREE_TYPE (TREE_TYPE (found)) != TREE_TYPE (TREE_TYPE (method)))
+       parse_warning_context 
+         (method_wfl,
+          "Method `%s' redefined with different return type in class `%s'",
+          lang_printable_name (found),
+          IDENTIFIER_POINTER (DECL_NAME (TYPE_NAME (DECL_CONTEXT (found)))));
+
+      /* Can't override final. Can't override static. */
+      if (METHOD_FINAL (found) || METHOD_STATIC (found))
+       {
+         /* Static *can* override static */
+         if (METHOD_STATIC (found) && METHOD_STATIC (method))
+           continue;
+         parse_error_context 
+           (method_wfl,
+            "%s methods can't be overriden. Method `%s' is %s in class `%s'",
+            (METHOD_FINAL (found) ? "Final" : "Static"),
+            lang_printable_name (found),
+            (METHOD_FINAL (found) ? "final" : "static"),
+            IDENTIFIER_POINTER
+              (DECL_NAME (TYPE_NAME (DECL_CONTEXT (found)))));
+         continue;
+       }
+      /* Static method can't override instance method. */
+      if (METHOD_STATIC (method))
+       {
+         parse_error_context 
+           (method_wfl,
+            "Instance methods can't be overriden by a static method. Method "
+            "`%s' is an instance method in class `%s'",
+            lang_printable_name (found),
+            IDENTIFIER_POINTER
+              (DECL_NAME (TYPE_NAME (DECL_CONTEXT (found)))));
+         continue;
+       }
+      /* Overriding/hiding public must be public or
+        overriding/hiding protected must be protected or public */
+      if ((METHOD_PUBLIC (found) && !METHOD_PUBLIC (method)) ||
+         (METHOD_PROTECTED (found) 
+          && !(METHOD_PUBLIC (method) || METHOD_PROTECTED (method))))
+       {
+         parse_error_context 
+           (method_wfl,
+            "Methods can't be overridden to be more private. Method `%s' is "
+            "%s in class `%s'", lang_printable_name (found),
+            (METHOD_PUBLIC (found) ? "public" : "protected"),
+            IDENTIFIER_POINTER 
+              (DECL_NAME (TYPE_NAME (DECL_CONTEXT (found)))));
+         continue;
+       }
+
+      /* If the method has default access in an other package, then
+      issue a warning that the current method doesn't override the one
+      that was found elsewhere */
+      aflags = get_access_flags_from_decl (found);
+      if ((!aflags || (aflags > ACC_PROTECTED))
+         && !class_in_current_package (DECL_CONTEXT (found)))
+       parse_warning_context 
+         (method_wfl, "Method `%s' in class `%s' does not "
+          "override the corresponding method in class `%s', which is "
+          "private to a different package",
+          lang_printable_name (found),
+          IDENTIFIER_POINTER (DECL_NAME (class_decl)),
+          IDENTIFIER_POINTER (DECL_NAME (TYPE_NAME (DECL_CONTEXT (found)))));
+
+      /* Check on (default) package access. FIXME.  */
+      /* Inheriting multiple methods with the same signature. FIXME */
+    }
+  
+  TYPE_METHODS (class) = nreverse (TYPE_METHODS (class));
+
+  if (!seen_constructor)
+    {
+      /* No constructor seen, we craft one, at line 0 */
+      int saved_lineno = lineno;
+      tree meth, decl;
+      lineno = 0;
+      meth = make_node (FUNCTION_TYPE);
+      TREE_TYPE (meth) = void_type_node;
+      TYPE_ARG_TYPES (meth) = NULL_TREE; 
+      decl = add_method (class, 0, init_identifier_node,
+                        build_java_signature (meth));
+      DECL_CONSTRUCTOR_P (decl) = 1;
+      lineno = saved_lineno;
+    }
+}
+
+/* Check abstract method of interface INTERFACE */
+
+static void
+java_check_abstract_methods (interface)
+     tree interface;
+{
+  int i, n;
+  tree method, basetype_vec, found;
+
+  for (method = TYPE_METHODS (interface); method; method = TREE_CHAIN (method))
+    {
+      char *csig;
+      tree name = DECL_NAME (method);
+
+      /* 2- Check for double definition inside the defining interface */
+      if (check_method_redefinition (interface, method))
+       continue;
+
+      /* 3- Overriding is OK as far as we preserve the return type and
+        the thrown exceptions */
+      found = lookup_java_interface_method2 (interface, method);
+      if (found)
+       {
+         parse_error_context 
+           (lookup_cl (method),
+            "Method `%s' previously defined in interface `%s' is "
+            "redefined with different return type in interface `%s'",
+            lang_printable_name (found),
+            IDENTIFIER_POINTER (DECL_NAME (TYPE_NAME (DECL_CONTEXT (found)))),
+            IDENTIFIER_POINTER (DECL_NAME (TYPE_NAME (interface))));
+         continue;
+       }
+    }
+
+  /* 4- Inherited methods can't differ by their returned types */
+  if (!(basetype_vec = TYPE_BINFO_BASETYPES (interface)))
+    return;
+  n = TREE_VEC_LENGTH (basetype_vec);
+  for (i = 0; i < n; i++)
+    {
+      tree sub_interface_method, sub_interface;
+      tree vec_elt = TREE_VEC_ELT (basetype_vec, i);
+      if (!vec_elt)
+       continue;
+      sub_interface = BINFO_TYPE (vec_elt);
+      for (sub_interface_method = TYPE_METHODS (sub_interface); 
+          sub_interface_method;
+          sub_interface_method = TREE_CHAIN (sub_interface_method))
+       {
+         found = lookup_java_interface_method2 (interface, 
+                                                sub_interface_method);
+         if (found && (found != sub_interface_method))
+           parse_error_context 
+             (lookup_cl (sub_interface_method),
+              "Interface `%s' inherits method `%s' from interface `%s'. This "
+              "method is redefined with a different return "
+              "type in interface `%s'",
+              IDENTIFIER_POINTER (DECL_NAME (TYPE_NAME (interface))),
+              lang_printable_name (found),
+              IDENTIFIER_POINTER 
+                (DECL_NAME (TYPE_NAME (DECL_CONTEXT (sub_interface_method)))),
+              IDENTIFIER_POINTER 
+                (DECL_NAME (TYPE_NAME (DECL_CONTEXT (found)))));
+       }
+    }
+}
+
+/* Check the method on all the defined classes. Should be done to the
+   classes declared in the compilation unit only. FIXME  */
+
+void
+java_check_methods ()
+{
+
+  tree current;
+  for (current = ctxp->class_list; current; current = TREE_CHAIN (current))
+    if (CLASS_FROM_SOURCE_P (TREE_TYPE (current)))
+      {
+       tree class = CLASS_TO_HANDLE_TYPE (TREE_TYPE (current));
+
+       if (CLASS_INTERFACE (TYPE_NAME (class)))
+         java_check_abstract_methods (class);
+       else
+         java_check_regular_methods (current);
+      }
+}
+
+/* Lookup methods in interfaces using their name and partial
+   signature. Return a matching method only if their types differ.  */
+
+static tree
+lookup_java_interface_method2 (class, method_decl)
+     tree class, method_decl;
+{
+  int i, n;
+  tree basetype_vec = TYPE_BINFO_BASETYPES (class), to_return;
+
+  if (!basetype_vec)
+    return NULL_TREE;
+
+  n = TREE_VEC_LENGTH (basetype_vec);
+  for (i = 0; i < n; i++)
+    {
+      tree vec_elt = TREE_VEC_ELT (basetype_vec, i), to_return;
+      if ((BINFO_TYPE (vec_elt) != object_type_node)
+         && (to_return = 
+             lookup_java_method2 (BINFO_TYPE (vec_elt), method_decl, 1)))
+       return to_return;
+    }
+  for (i = 0; i < n; i++)
+    {
+      to_return = lookup_java_interface_method2 
+       (BINFO_TYPE (TREE_VEC_ELT (basetype_vec, i)), method_decl);
+      if (to_return)
+       return to_return;
+    }
+
+  return NULL_TREE;
+}
+
+/* Lookup method using their name and partial signature. Return a
+   matching method only if their types differ.  */
+
+static tree
+lookup_java_method2 (clas, method_decl, do_interface)
+     tree clas, method_decl;
+     int do_interface;
+{
+  tree method, method_signature, method_name, method_type;
+  method_signature = build_java_argument_signature (TREE_TYPE (method_decl));
+  method_name = DECL_NAME (method_decl);
+  method_type = TREE_TYPE (TREE_TYPE (method_decl));
+
+  while (clas != NULL_TREE)
+    {
+      for (method = TYPE_METHODS (clas);
+          method != NULL_TREE;  method = TREE_CHAIN (method))
+       {
+         tree method_sig = build_java_argument_signature (TREE_TYPE (method));
+         if (DECL_NAME (method) == method_name 
+             && method_sig == method_signature 
+             && TREE_TYPE (TREE_TYPE (method)) != method_type)
+           {
+             return method;
+           }
+       }
+      clas = (do_interface ? NULL_TREE : CLASSTYPE_SUPER (clas));
+    }
+  return NULL_TREE;
+}
+
+/* Return the line that matches DECL line number. Used during error
+   report */
+
+static tree
+lookup_cl (decl)
+     tree decl;
+{
+  static tree cl = NULL_TREE;
+  
+  if (!decl)
+    return NULL_TREE;
+
+  if (cl == NULL_TREE)
+    cl = build_expr_wfl (NULL_TREE, NULL, 0, 0);
+
+  EXPR_WFL_FILENAME_NODE (cl) = get_identifier (DECL_SOURCE_FILE (decl));
+  EXPR_WFL_SET_LINECOL (cl, DECL_SOURCE_LINE_FIRST (decl), -1);
+
+  return cl;
+}
+
+/* Look for a simple name in the single-type import list */
+
+static tree
+find_name_in_single_imports (name)
+     tree name;
+{
+  tree node;
+
+  for (node = ctxp->import_list; node; node = TREE_CHAIN (node))
+    if (TREE_VALUE (node) == name)
+      return (EXPR_WFL_NODE (TREE_PURPOSE (node)));
+
+  return NULL_TREE;
+}
+
+/* Process all single-type import. */
+
+static int
+process_imports ()
+{
+  tree import;
+  int error_found;
+
+  for (import = ctxp->import_list; import; import = TREE_CHAIN (import))
+    {
+      tree to_be_found = EXPR_WFL_NODE (TREE_PURPOSE (import));
+
+      /* Don't load twice something already defined. */
+      if (IDENTIFIER_CLASS_VALUE (to_be_found))
+       continue;
+      QUALIFIED_P (to_be_found) = 1;
+      load_class (to_be_found, 0);
+      error_found =
+       check_pkg_class_access (to_be_found, TREE_PURPOSE (import));
+      if (!IDENTIFIER_CLASS_VALUE (to_be_found))
+       {
+         parse_error_context (TREE_PURPOSE (import),
+                              "Class or interface `%s' not found in import",
+                              IDENTIFIER_POINTER (to_be_found));
+         return 1;
+       }
+      if (error_found)
+       return 1;
+    }
+  return 0;
+}
+
+/* Possibly find a class imported by a single-type import statement. Return
+   1 if an error occured, 0 otherwise. */
+
+static int
+find_in_imports (class_type)
+     tree class_type;
+{
+  tree import;
+
+  for (import = ctxp->import_list; import; import = TREE_CHAIN (import))
+    if (TREE_VALUE (import) == TYPE_NAME (class_type))
+      {
+       TYPE_NAME (class_type) = EXPR_WFL_NODE (TREE_PURPOSE (import));
+       QUALIFIED_P (TYPE_NAME (class_type)) = 1;
+       return check_pkg_class_access (TYPE_NAME (class_type), 
+                                      TREE_PURPOSE (import));
+      }
+  return 0;
+}
+
+/* Process a import on demand statement (lazy) */
+
+static int
+read_import_entry (jcf, dirp, returned_name)
+     JCF *jcf;
+     DIR *dirp;
+     char **returned_name;
+{
+  if (dirp)
+    {
+      struct dirent *direntp = readdir (dirp);
+      if (!direntp)
+       {
+         *returned_name = NULL;
+         return 0;
+       }
+      else
+       {
+         *returned_name = direntp->d_name;
+         return (strlen (direntp->d_name));
+       }
+    }
+  else
+    {
+      int current_dir_len = strlen (jcf->classname);
+      char *current_entry;
+      int current_entry_len;
+
+      /* Here we read a zip directory as a file directory. The files
+        we're selecting must have the same root than the directory
+        we're examining. */
+
+      ZipDirectory *zipd = (ZipDirectory *)jcf->zipd; 
+
+      while (zipd)
+       {
+         current_entry = ZIPDIR_FILENAME (zipd);
+         current_entry_len = zipd->filename_length;
+         while (current_entry_len && current_entry [current_entry_len] != '/')
+           current_entry_len--;
+         /* If the path of the current file doesn't match the directory we're
+            scanning, that the end of the search */
+         current_entry_len++;
+         if (strncmp (jcf->classname, current_entry, current_dir_len))
+           {
+             *returned_name = NULL;
+             return 0;
+           }
+         /* Ok, we have at least the same path. The position of the last '/'
+            of the current file we're examining should match the size of
+            name of the directory we're browsing, otherwise that an entry
+            belonging to a sub directory, we want to skip it. */
+         if (current_entry_len != current_dir_len)
+           zipd = ZIPDIR_NEXT (zipd);
+         else
+           {
+             jcf->zipd = ZIPDIR_NEXT (zipd); /* Prepare next read */
+             *returned_name = &current_entry [current_entry_len];
+             return (zipd->filename_length - current_entry_len);
+           }
+       }
+    }
+}
+
+/* Read a import directory, gathering potential match for further type
+   references. Indifferently reads a filesystem or a ZIP archive
+   directory.  */
+
+static void
+read_import_dir (wfl)
+     tree wfl;
+{
+  char *name = IDENTIFIER_POINTER (EXPR_WFL_NODE (wfl));
+  int name_len = IDENTIFIER_LENGTH (EXPR_WFL_NODE (wfl)), reclen;
+  DIR *dirp = NULL;
+  tree dirname = ident_subst (name, name_len, "", '.', '/', "");
+  JCF jcfr, *jcf, *saved_jcf = current_jcf;
+  char *founddirname, *d_name;
+  struct ZipFileCache zip_cache;
+
+  jcf = &jcfr;
+  if (!classpath)
+    fix_classpath ();
+  if (!(founddirname = find_class (name, name_len, jcf, 0)))
+    fatal ("Can't import `%s'", name);
+  if (jcf->outofsynch)
+    jcf_out_of_synch (jcf);
+  if (jcf->seen_in_zip)
+    jcf->zipd = ZIPDIR_NEXT ((ZipDirectory *)jcf->zipd);
+
+  else if (founddirname && (dirp = opendir (founddirname)))
+    {
+      readdir (dirp); readdir (dirp);
+    }
+
+  if (!founddirname && !dirp)
+    {
+      static int first = 1;
+      if (first)
+       {
+         char buffer [256];
+         sprintf (buffer, "Can't find default package `%s'. Check "
+                  "the CLASSPATH environment variable and the access to the "
+                  "archives.", name);
+         error (buffer);
+         java_error_count++;
+         first = 0;
+       }
+      else
+       parse_error_context (wfl, "Package `%s' not found in import", name);
+      current_jcf = saved_jcf;
+      return;
+    }
+
+  /* Here we should have a unified way of retrieving an entry, to be
+     indexed. */
+  while ((reclen = read_import_entry (jcf, dirp, &d_name)))
+    {
+      int java_or_class = 0;
+      int len; 
+      if ((reclen > 5) 
+         && !strcmp (&d_name [reclen-5], ".java"))
+       {
+         java_or_class = 1;
+         len = reclen - 5;
+       }
+         
+      if (!java_or_class && (reclen > 6) &&
+         !strcmp (&d_name [reclen-6], ".class"))
+       {
+         java_or_class = 2;
+         len = reclen - 6;
+       }
+
+      if (java_or_class)
+       {
+         char *id_name;
+         tree node, old;
+
+         obstack_grow (&temporary_obstack, name, name_len);
+         obstack_1grow (&temporary_obstack, '/');
+         obstack_grow0 (&temporary_obstack, d_name, len);
+         id_name = obstack_finish (&temporary_obstack);
+
+         node = get_identifier (id_name);
+         IS_A_CLASSFILE_NAME (node) = 1; /* Or soon to be */
+         QUALIFIED_P (node) = 1; /* As soon as we turn / into . */
+       }
+    }
+  if (dirp)
+    closedir (dirp);
+  
+  current_jcf = saved_jcf;
+}
+
+/* Possibly find a type in the import on demands specified
+   types. Returns 1 if an error occured, 0 otherwise. Run throught the
+   entire list, to detected potential double definitions.  */
+                
+static int
+find_in_imports_on_demand (class_type)
+     tree class_type;
+{
+  tree node, import, node_to_use;
+  int seen_once = -1;
+  tree cl;
+
+  for (import = ctxp->import_demand_list; import; import = TREE_CHAIN (import))
+    {
+      char *id_name;
+      tree found;
+      obstack_grow (&temporary_obstack, 
+                   IDENTIFIER_POINTER (EXPR_WFL_NODE (TREE_PURPOSE (import))),
+                   IDENTIFIER_LENGTH (EXPR_WFL_NODE (TREE_PURPOSE (import))));
+      obstack_1grow (&temporary_obstack, '/');
+      obstack_grow0 (&temporary_obstack, 
+                    IDENTIFIER_POINTER (TYPE_NAME (class_type)),
+                    IDENTIFIER_LENGTH (TYPE_NAME (class_type)));
+      id_name = obstack_finish (&temporary_obstack);
+             
+      node = maybe_get_identifier (id_name);
+      if (node && IS_A_CLASSFILE_NAME (node))
+       {
+         if (seen_once < 0)
+           {
+             cl = TREE_PURPOSE (import);
+             seen_once = 1;
+             node_to_use = node;
+           }
+         else
+           {
+             seen_once++;
+             parse_error_context 
+               (import, "Type `%s' also potentially defined in package `%s'",
+                IDENTIFIER_POINTER (TYPE_NAME (class_type)),
+                IDENTIFIER_POINTER (EXPR_WFL_NODE (TREE_PURPOSE (import))));
+           }
+       }
+    }
+
+  if (seen_once == 1)
+    {
+      /* Setup lineno so that it refers to the line of the import (in
+        case we parse a class file and encounter errors */
+      tree decl;
+      int saved_lineno = lineno;
+      lineno = EXPR_WFL_LINENO (cl);
+      TYPE_NAME (class_type) = ident_subst (IDENTIFIER_POINTER (node_to_use),
+                                           IDENTIFIER_LENGTH (node_to_use),
+                                           "", '/', '.', "");
+      QUALIFIED_P (TYPE_NAME (class_type)) = 1;
+      decl = IDENTIFIER_CLASS_VALUE (TYPE_NAME (class_type));
+      /* If there is no DECL set for the class or if the class isn't
+        loaded and not seen in source yet, the load */
+      if (!decl || (!CLASS_LOADED_P (TREE_TYPE (decl))
+                   && !CLASS_FROM_SOURCE_P (TREE_TYPE (decl))))
+       load_class (node_to_use, 0);
+      lineno = saved_lineno;
+      return check_pkg_class_access (TYPE_NAME (class_type), cl);
+    }
+  else
+    return (seen_once < 0 ? 0 : seen_once); /* It's ok not to have found */
+}
+
+/* Check that CLASS_NAME refers to a PUBLIC class. Return 0 if no
+   access violations were found, 1 otherwise.  */
+
+static int
+check_pkg_class_access (class_name, cl)
+     tree class_name;
+     tree cl;
+{
+  tree type;
+  int access;
+
+  if (!QUALIFIED_P (class_name) || !IDENTIFIER_CLASS_VALUE (class_name))
+    return 0;
+
+  if (!(type = TREE_TYPE (IDENTIFIER_CLASS_VALUE (class_name))))
+    return 0;
+
+  if (!CLASS_PUBLIC (TYPE_NAME (type)))
+    {
+      parse_error_context 
+       (cl, "Can't access %s `%s'. Only public classes and interfaces in "
+        "other packages can be accessed",
+        (CLASS_INTERFACE (TYPE_NAME (type)) ? "interface" : "class"),
+        IDENTIFIER_POINTER (class_name));
+      return 1;
+    }
+  return 0;
+}
+
+/* Local variable declaration. */
+
+static void
+declare_local_variables (modifier, type, vlist)
+     int modifier;
+     tree type;
+     tree vlist;
+{
+  tree decl, current, returned_type, type_wfl, init_stmt = NULL_TREE;
+  int must_chain = 0;
+
+  /* Push a new block if statement were seen between the last time we
+     pushed a block and now. Keep a cound of block to close */
+  if (BLOCK_EXPR_BODY (DECL_FUNCTION_BODY (current_function_decl)))
+    {
+      tree body = DECL_FUNCTION_BODY (current_function_decl);
+      tree b = enter_block ();
+      BLOCK_EXPR_ORIGIN(b) = body;
+    }
+
+  if (modifier)
+    {
+      int i;
+      for (i = 0; i <= 10; i++) if (1 << i & modifier) break;
+      parse_error_context 
+       (ctxp->modifier_ctx [i],
+        (modifier == ACC_FINAL ?
+         "Unsupported JDK1.1 `final' locals" :
+         "Only `final' is allowed as a local variables modifier"));
+      return;
+    }
+
+  if (unresolved_type_p (type, &returned_type))
+    {
+      if (returned_type)
+        type = returned_type;
+      else 
+       {
+         type_wfl = type;
+         type = obtain_incomplete_type (type);
+         must_chain = 1;
+       }
+    }
+  
+  for (current = vlist; current; current = TREE_CHAIN (current))
+    {
+      tree wfl  = TREE_PURPOSE (current);
+      tree name = EXPR_WFL_NODE (wfl);
+      tree init = TREE_VALUE (current);
+      tree other = lookup_name_in_blocks (name);
+
+      /* Don't try to use an INIT statement when an error was found */
+      if (init && java_error_count)
+       init = NULL_TREE;
+
+      if (other)
+       parse_error_context 
+         (wfl, "Variable `%s' is already defined in this method and was "
+          "declared `%s %s' in line %d", 
+          IDENTIFIER_POINTER (name), lang_printable_name (TREE_TYPE (other)),
+          IDENTIFIER_POINTER (name), DECL_SOURCE_LINE (other));
+      else
+       {
+         if (!must_chain && TREE_CODE (type) == RECORD_TYPE)
+           type = promote_type (type);
+         /* Never layout this decl. This will be done when its scope
+             will be entered */
+         decl = build_decl_no_layout (VAR_DECL, name, type);
+         BLOCK_CHAIN_DECL (decl);
+
+         /* Add the initialization function to the current function's code */
+         if (init)
+           {
+             tree wfl;
+             MODIFY_EXPR_FROM_INITIALIZATION_P (init) = 1;
+             java_method_add_stmt 
+               (current_function_decl,
+                build_debugable_stmt (EXPR_WFL_LINECOL (init), init));
+           }
+
+         if (must_chain)
+           {
+             jdep *dep;
+             register_incomplete_type (JDEP_VARIABLE, type_wfl, decl, type);
+             dep = CLASSD_LAST (ctxp->classd_list);
+             JDEP_GET_PATCH (dep) = &TREE_TYPE (decl);
+           }
+       }
+    }
+  SOURCE_FRONTEND_DEBUG (("Defined locals"));
+}
+
+/* Called during parsing. Build decls from argument list.  */
+
+static void
+source_start_java_method (fndecl)
+     tree fndecl;
+{
+  tree tem;
+  tree parm_decl;
+  int i;
+
+  extern tree current_binding_level;
+  current_function_decl = fndecl;
+
+  /* New scope for the function */
+  enter_block ();
+  for (tem = TYPE_ARG_TYPES (TREE_TYPE (fndecl)), i = 0;
+       tem != NULL_TREE; tem = TREE_CHAIN (tem), i++)
+    {
+      tree type = TREE_VALUE (tem);
+      tree name = TREE_PURPOSE (tem);
+      
+      /* If type is incomplete. Layout can't take place
+        now. Create an incomplete decl and ask for the decl to be
+        patched later */
+      if (INCOMPLETE_TYPE_P (type))
+       {
+         jdep *jdep;
+         parm_decl = build_decl_no_layout (PARM_DECL, name, type);
+         
+         register_incomplete_type (JDEP_PARM, NULL_TREE, NULL_TREE, type);
+         jdep = CLASSD_LAST (ctxp->classd_list);
+         JDEP_MISC (jdep) = name;
+         JDEP_GET_PATCH (jdep) = &TREE_TYPE (parm_decl);
+       }
+      else
+       parm_decl = build_decl (PARM_DECL, name, type);
+
+      BLOCK_CHAIN_DECL (parm_decl);
+    }
+  tem = BLOCK_EXPR_DECLS (DECL_FUNCTION_BODY (current_function_decl));
+  BLOCK_EXPR_DECLS (DECL_FUNCTION_BODY (current_function_decl)) =
+    nreverse (tem);
+  DECL_ARG_SLOT_COUNT (current_function_decl) = i;
+}
+
+/* Called during expansion. Push decls formerly built from argument
+   list so they're usable during expansion. */
+
+static void
+expand_start_java_method (fndecl)
+     tree fndecl;
+{
+  tree tem, *ptr;
+  tree parm_decl;
+
+  extern tree current_binding_level;
+  current_function_decl = fndecl;
+
+  announce_function (fndecl);
+  pushlevel (1);               /* Push parameters */
+  ptr = &DECL_ARGUMENTS (fndecl);
+  tem  = BLOCK_EXPR_DECLS (DECL_FUNCTION_BODY (current_function_decl));
+  while (tem)
+    {
+      tree next = TREE_CHAIN (tem);
+      DECL_ARG_TYPE (tem) = TREE_TYPE (tem);
+      layout_decl (tem, 0);
+      pushdecl (tem);
+      INITIALIZED_P (tem) = 1; /* Parms are initialized */
+      *ptr = tem;
+      ptr = &TREE_CHAIN (tem);
+      tem = next;
+    }
+  *ptr = NULL_TREE;
+  pushdecl_force_head (DECL_ARGUMENTS (fndecl));
+  lineno = DECL_SOURCE_LINE_FIRST (fndecl);
+  complete_start_java_method (fndecl); 
+}
+
+/* Terminate a function and expand its body.  */
+
+static void
+source_end_java_method ()
+{
+  tree fndecl = current_function_decl;
+
+  java_parser_context_save_global ();
+  lineno = ctxp->last_ccb_indent1;
+
+  /* Generate function's code */
+  if (BLOCK_EXPR_BODY (DECL_FUNCTION_BODY (fndecl))
+      && ! flag_emit_class_files)
+    expand_expr_stmt (BLOCK_EXPR_BODY (DECL_FUNCTION_BODY (fndecl)));
+
+  /* pop out of its parameters */
+  pushdecl_force_head (DECL_ARGUMENTS (fndecl));
+  poplevel (1, 0, 1);
+  BLOCK_SUPERCONTEXT (DECL_INITIAL (fndecl)) = fndecl;
+
+  /* Generate rtl for function exit.  */
+  if (! flag_emit_class_files)
+    {
+      lineno = DECL_SOURCE_LINE_LAST (fndecl);
+      expand_function_end (input_filename, lineno, 0);
+
+      /* Run the optimizers and output assembler code for this function. */
+      rest_of_compilation (fndecl);
+    }
+
+  current_function_decl = NULL_TREE;
+  /*  permanent_allocation (1); */
+  java_parser_context_restore_global ();
+}
+
+/* Record EXPR in the current function block. Complements compound
+   expression second operand if necessary.  */
+
+tree
+java_method_add_stmt (fndecl, expr)
+     tree fndecl, expr;
+{
+  tree body = BLOCK_EXPR_BODY (DECL_FUNCTION_BODY (fndecl));
+  tree node;
+
+  if (java_error_count)
+    return body;
+  if ((node = add_stmt_to_compound (body, NULL_TREE, expr)) == body)
+    return body;
+
+  BLOCK_EXPR_BODY (DECL_FUNCTION_BODY (fndecl)) = node;
+  TREE_SIDE_EFFECTS (node) = 1;
+  return node;
+}
+
+/* Add STMT to EXISTING if possible, otherwise create a new
+   COMPOUND_EXPR and add STMT to it. */
+
+static tree
+add_stmt_to_compound (existing, type, stmt)
+     tree existing, type, stmt;
+{
+  tree node;
+
+  if (existing && (TREE_CODE (existing) == COMPOUND_EXPR)
+      && TREE_OPERAND (existing, 1) == size_zero_node)
+    {
+      TREE_OPERAND (existing, 1) = stmt;
+      TREE_TYPE (existing) = type;
+      return existing;
+    }
+  else if (existing)
+    node = build (COMPOUND_EXPR, type, existing, stmt);
+  else
+    node = build (COMPOUND_EXPR, type, stmt, size_zero_node);
+  
+  return node;
+}
+
+/* Hold THIS for the scope of the current public method decl.  */
+static tree current_this;
+
+/* Layout all class found during parsing */
+
+void
+java_layout_classes ()
+{
+  tree current;
+  for (current = ctxp->class_list; current; current = TREE_CHAIN (current))
+    {
+      current_class = TREE_TYPE (current);
+      TYPE_FIELDS (current_class) = nreverse (TYPE_FIELDS (current_class));
+      if (!TYPE_SIZE (current_class))
+       safe_layout_class (current_class);
+    }
+}
+
+/* Expand all methods in all registered classes.  */
+
+void
+java_complete_expand_methods ()
+{
+  tree current;
+  
+  for (current = ctxp->class_list; current; current = TREE_CHAIN (current))
+    {
+      extern tree current_constant_pool_data_ref;
+      tree class_type = CLASS_TO_HANDLE_TYPE (TREE_TYPE (current));
+      tree decl;
+      int saved_lineno;
+
+      current_class = TREE_TYPE (current);
+
+      /* Initialize a new constant pool */
+      init_outgoing_cpool ();
+
+      /* Don't process function bodies in interfaces */
+      if (!CLASS_INTERFACE (TYPE_NAME (current_class)))
+       for (decl = TYPE_METHODS (class_type); decl; decl = TREE_CHAIN (decl))
+         {
+           current_function_decl = decl;
+           /* Don't generate debug info on line zero when expanding a
+              generated constructor. */
+           if (DECL_CONSTRUCTOR_P (decl) && !DECL_FUNCTION_BODY (decl))
+             {
+               /* If we found errors, it's too dangerous to try to generate
+                  and expand a constructor */
+               if (!java_error_count)
+                 {
+                   restore_line_number_status (1);
+                   java_complete_expand_method (decl);
+                   restore_line_number_status (0);
+                 }
+             }
+           else
+             java_complete_expand_method (decl);
+         }
+
+      /* Make the class data, register it and run the rest of decl
+         compilation on it */
+      if (!java_error_count && ! flag_emit_class_files)
+       {
+         make_class_data (current_class);
+         register_class ();
+         rest_of_decl_compilation (TYPE_NAME (current_class), (char*) 0, 1, 0);
+       }
+    }
+}
+
+/* Complete and expand a method.  */
+
+static void
+java_complete_expand_method (mdecl)
+     tree mdecl;
+{
+  tree node;
+  jdep *current;
+  int no_ac_found = 1;
+
+  /* We generate some code for an empty constructor */
+  if (DECL_CONSTRUCTOR_P (mdecl) && !DECL_FUNCTION_BODY (mdecl))
+    {
+      tree arg_list, func, call;
+      tree method_type = TREE_TYPE (mdecl);
+      tree class_type = CLASS_TO_HANDLE_TYPE (current_class);
+      tree self_type = (CLASSTYPE_SUPER (class_type) ? 
+                       CLASSTYPE_SUPER (class_type) : class_type);
+      tree method_signature = 
+       TYPE_LANG_SPECIFIC (method_type)->signature;
+      tree method = 
+       lookup_java_constructor (CLASS_TO_HANDLE_TYPE (self_type),
+                                method_signature);
+      tree block, compound;
+
+      /* Fixe the begining/ending lines of the method so that with
+         no_line_numbers set to 1 it doesn't generate debug info at
+         line 1 for this artificial constructor. */
+      DECL_SOURCE_LINE (mdecl) = 1;
+      DECL_SOURCE_LINE_MERGE (mdecl, 1);
+      source_start_java_method (mdecl);
+      arg_list = BLOCK_EXPR_DECLS (DECL_FUNCTION_BODY (mdecl));
+      enter_block ();
+      func = build_known_method_ref (method, method_type, self_type,
+                                    method_signature, arg_list);
+      
+      if (! flag_emit_class_files)
+       func = build1 (NOP_EXPR, build_pointer_type (method_type), func);
+      call = build (CALL_EXPR, TREE_TYPE (method_type), func, 
+                   build_tree_list (NULL_TREE, arg_list), NULL_TREE);
+      TREE_SIDE_EFFECTS (call) = 1;
+      call = build_class_init (self_type, call);
+      compound = java_method_add_stmt (mdecl, call);
+      block = exit_block ();
+      BLOCK_EXPR_BODY (DECL_FUNCTION_BODY (mdecl)) = block;
+      /* The function decl, its block and the compound statement
+         within this block are all of void type. */
+      TREE_TYPE (block) = TREE_TYPE (compound) = 
+       TREE_TYPE (DECL_FUNCTION_BODY (mdecl)) = void_type_node;
+      exit_block ();
+      no_ac_found = 0;
+    }
+  
+  if (DECL_FUNCTION_BODY (mdecl))
+    {
+      expand_start_java_method (mdecl);
+
+      current_this 
+       = (!METHOD_STATIC (mdecl) ? 
+          BLOCK_EXPR_DECLS (DECL_FUNCTION_BODY (mdecl)) : NULL_TREE);
+
+      if (BLOCK_EXPR_BODY (DECL_FUNCTION_BODY (mdecl)) && no_ac_found)
+       java_complete_tree (BLOCK_EXPR_BODY (DECL_FUNCTION_BODY (mdecl)));
+      /* Don't go any further if we've found error(s) during the
+         expansion */
+      if (!java_error_count)
+       source_end_java_method ();
+    }
+}
+
+/* Expand finals.  */
+
+void
+java_expand_finals ()
+{
+}
+
+/* Wrap non WFL PRIMARY around a WFL and set EXPR_WFL_QUALIFICATION to
+   a tree list node containing RIGHT. Fore coming RIGHTs will be
+   chained to this hook. LOCATION contains the location of the
+   separating `.' operator.  */
+
+static tree
+make_qualified_primary (primary, right, location)
+     tree primary, right;
+     int location;
+{
+  tree wfl;
+
+  /* We want to process THIS . xxx symbolicaly, to keep it consistent
+     with the way we're processing SUPER. A THIS from a primary as a
+     different form than a SUPER. Turn THIS into something symbolic */
+  if (TREE_CODE (primary) == JAVA_THIS_EXPR)
+    {
+      wfl = build_wfl_node (this_identifier_node, input_filename, 0, 0);
+      EXPR_WFL_LINECOL (wfl) = EXPR_WFL_LINECOL (primary);
+      wfl = make_qualified_name (wfl, right, location);
+      PRIMARY_P (wfl) = 1;
+      return wfl;
+    }
+  /* Other non WFL node are wrapped around a WFL */
+  else if (TREE_CODE (primary) != EXPR_WITH_FILE_LOCATION)
+    {
+      wfl = build_expr_wfl (NULL_TREE, ctxp->filename, 0, 0);
+      EXPR_WFL_LINECOL (wfl) = EXPR_WFL_LINECOL (primary);
+      EXPR_WFL_QUALIFICATION (wfl) = build_tree_list (primary, NULL_TREE);
+    }
+  else
+    {
+      wfl = primary;
+      if (!EXPR_WFL_QUALIFICATION (primary))
+       EXPR_WFL_QUALIFICATION (primary) = 
+         build_tree_list (primary, NULL_TREE);
+    }
+
+  EXPR_WFL_LINECOL (right) = location;
+  chainon (EXPR_WFL_QUALIFICATION (wfl), build_tree_list (right, NULL_TREE));
+  PRIMARY_P (wfl) =  1;
+  return wfl;
+}
+
+/* Simple merge of two name separated by a `.' */
+
+static tree
+merge_qualified_name (left, right)
+     tree left, right;
+{
+  tree node;
+  obstack_grow (&temporary_obstack, IDENTIFIER_POINTER (left),
+               IDENTIFIER_LENGTH (left));
+  obstack_1grow (&temporary_obstack, '.');
+  obstack_grow0 (&temporary_obstack, IDENTIFIER_POINTER (right),
+                IDENTIFIER_LENGTH (right));
+  node =  get_identifier (obstack_base (&temporary_obstack));
+  obstack_free (&temporary_obstack, obstack_base (&temporary_obstack));
+  QUALIFIED_P (node) = 1;
+  return node;
+}
+
+/* Merge the two parts of a qualified name into LEFT.  Set the
+   location information of the resulting node to LOCATION, usually
+   inherited from the location information of the `.' operator. */
+
+static tree
+make_qualified_name (left, right, location)
+     tree left, right;
+     int location;
+{
+  int qualified;
+  tree left_id = EXPR_WFL_NODE (left);
+  tree right_id = EXPR_WFL_NODE (right);
+  tree wfl, merge;
+
+  merge = merge_qualified_name (left_id, right_id);
+
+  /* Left wasn't qualified and is now qualified */
+  if (!QUALIFIED_P (left_id))
+    {
+      tree wfl = build_expr_wfl (left_id, ctxp->filename, 0, 0);
+      EXPR_WFL_LINECOL (wfl) = EXPR_WFL_LINECOL (left);
+      EXPR_WFL_QUALIFICATION (left) = build_tree_list (wfl, NULL_TREE);
+    }
+  
+  wfl = build_expr_wfl (right_id, ctxp->filename, 0, 0);
+  EXPR_WFL_LINECOL (wfl) = location;
+  chainon (EXPR_WFL_QUALIFICATION (left), build_tree_list (wfl, NULL_TREE));
+
+  EXPR_WFL_NODE (left) = merge;
+  return left;
+}
+
+/* Extract the last identifier component of the qualified in WFL. The
+   last identifier is removed from the linked list */
+
+static tree
+cut_identifier_in_qualified (wfl)
+     tree wfl;
+{
+  tree q;
+  tree previous = NULL_TREE;
+  for (q = EXPR_WFL_QUALIFICATION (wfl); ; previous = q, q = TREE_CHAIN (q))
+    if (!TREE_CHAIN (q))
+      {
+       if (!previous)
+         fatal ("Operating on a non qualified qualified WFL - "
+                "cut_identifier_in_qualified");
+       TREE_CHAIN (previous) = NULL_TREE;
+       return TREE_PURPOSE (q);
+      }
+}
+
+/* Resolve the expression name NAME. Return its decl.  */
+
+static tree
+resolve_expression_name (id)
+     tree id;
+{
+  tree name = EXPR_WFL_NODE (id);
+  tree decl;
+
+  /* 6.5.5.1: Simple expression names */
+  if (!PRIMARY_P (id) && !QUALIFIED_P (name))
+    {
+      /* 15.13.1: NAME can appear within the scope of a local variable
+         declaration */
+      if ((decl = IDENTIFIER_LOCAL_VALUE (name)))
+        return decl;
+
+      /* 15.13.1: NAME can appear within a class declaration */
+      else 
+        {
+         decl = lookup_field_wrapper (current_class, name);
+         if (decl)
+           {
+             int fs = FIELD_STATIC (decl);
+             /* Instance variable (8.3.1.1) can't appear within
+                static method, static initializer or initializer for
+                a static variable. */
+             if (!fs && METHOD_STATIC (current_function_decl))
+               {
+                 parse_error_context 
+                   (id, "Can't make a static reference to nonstatic variable "
+                    "`%s' in class `%s'",
+                    IDENTIFIER_POINTER (name),
+                    IDENTIFIER_POINTER (DECL_NAME 
+                                        (TYPE_NAME (current_class))));
+                 return error_mark_node;
+               }
+             decl = build_field_ref ((fs ? NULL_TREE : current_this),
+                                     current_class, name);
+             return (fs ? build_class_init (current_class, decl) : decl);
+           }
+         /* Fall down to error report on undefined variable */
+       }
+    }
+  /* 6.5.5.2 Qualified Expression Names */
+  else
+    {
+      qualify_ambiguous_name (id);
+      /* 15.10.1 Field Access Using a Primary and/or Expression Name */
+      /* 15.10.2: Accessing Superclass Members using super */
+      return resolve_field_access (id, NULL, NULL);
+    }
+
+  /* We've got an error here */
+  parse_error_context (id, "Undefined variable `%s'", 
+                      IDENTIFIER_POINTER (name));
+
+  return error_mark_node;
+}
+
+/* 15.10.1 Field Acess Using a Primary and/or Expression Name.
+   We return something suitable to generate the field access. We also
+   return the field decl in FIELD_DECL and its type in FIELD_TYPE.  If
+   recipient's address can be null. */
+
+static tree
+resolve_field_access (qual_wfl, field_decl, field_type)
+     tree qual_wfl;
+     tree *field_decl, *field_type;
+{
+  int is_static = 0;
+  tree field_ref;
+  tree decl, where_found, type_found;
+
+  if (resolve_qualified_expression_name (qual_wfl, &decl,
+                                        &where_found, &type_found))
+    return error_mark_node;
+
+  /* Resolve the LENGTH field of an array here */
+  if (DECL_NAME (decl) == length_identifier_node && TYPE_ARRAY_P (type_found)
+      && ! flag_emit_class_files)
+    {
+      tree length = build_java_array_length_access (where_found);
+      field_ref =
+       build_java_arraynull_check (type_found, length, int_type_node);
+    }
+  /* We might have been trying to resolve field.method(). In which
+     case, the resolution is over and decl is the answer */
+  else if (DECL_P (decl) && IDENTIFIER_LOCAL_VALUE (DECL_NAME (decl)) == decl)
+    field_ref = decl;
+  else if (DECL_P (decl))
+    {
+      is_static = DECL_P (decl) && FIELD_STATIC (decl);
+      field_ref = build_field_ref ((is_static ? NULL_TREE : where_found), 
+                                  type_found, DECL_NAME (decl));
+      if (field_ref == error_mark_node)
+       return error_mark_node;
+      if (is_static)
+       {
+         field_ref = build_class_init (type_found, field_ref);
+         /* If the static field was identified by an expression that
+            needs to be generated, make the field access a compound
+            expression whose first part of the evaluation of the
+            field selector part. */
+         if (where_found && TREE_CODE (where_found) != TYPE_DECL)
+           {
+             tree type = QUAL_DECL_TYPE (field_ref);
+             field_ref = build (COMPOUND_EXPR, type, where_found, field_ref);
+           }
+       }
+    }
+  else
+    field_ref = decl;
+
+  if (field_decl)
+    *field_decl = decl;
+  if (field_type)
+    *field_type = QUAL_DECL_TYPE (decl);
+  return field_ref;
+}
+
+/* 6.5.5.2: Qualified Expression Names */
+
+static int
+resolve_qualified_expression_name (wfl, found_decl, where_found, type_found)
+     tree wfl;
+     tree *found_decl, *type_found, *where_found;
+{
+  int from_type = 0;           /* Field search initiated from a type */
+  int from_super = 0, from_cast = 0;
+  int previous_call_static = 0;
+  int is_static;
+  tree decl = NULL_TREE, type = NULL_TREE, q;
+  *where_found = NULL_TREE;
+
+  for (q = EXPR_WFL_QUALIFICATION (wfl); q; q = TREE_CHAIN (q))
+    {
+      tree qual_wfl = QUAL_WFL (q);
+
+      /* 15.10.1 Field Access Using a Primary */
+      
+      switch (TREE_CODE (qual_wfl))
+       {
+       case CALL_EXPR:
+       case JAVA_NEW_CLASS_EXPR:
+         /* If the access to the function call is a non static field,
+            build the code to access it. */
+         if (DECL_P (decl) && !FIELD_STATIC (decl))
+           {
+             decl = maybe_access_field (decl, *where_found, type);
+             if (decl == error_mark_node)
+               return 1;
+           }
+         /* And code for the function call */
+         if (complete_function_arguments (qual_wfl))
+           return 1;
+         *where_found = 
+           patch_method_invocation_stmt (qual_wfl, decl, type, &is_static);
+         if (*where_found == error_mark_node)
+           return 1;
+         *type_found = type = QUAL_DECL_TYPE (*where_found);
+
+         /* If the previous call was static and this one is too,
+            build a compound expression to hold the two (because in
+            that case, previous function calls aren't transported as
+            forcoming function's argument. */
+         if (previous_call_static && is_static)
+           {
+             decl = build (COMPOUND_EXPR, type, decl, *where_found);
+             TREE_SIDE_EFFECTS (decl) = 1;
+           }
+         else
+           {
+             previous_call_static = is_static;
+             decl = *where_found;
+           }
+         continue;
+
+       case CONVERT_EXPR:
+         *where_found = decl = java_complete_tree (qual_wfl);
+         if (decl == error_mark_node)
+           return 1;
+         *type_found = type = QUAL_DECL_TYPE (decl);
+         from_cast = 1;
+         continue;
+
+       case ARRAY_REF:
+         /* If the access to the function call is a non static field,
+            build the code to access it. */
+         if (DECL_P (decl) && !FIELD_STATIC (decl))
+           {
+             decl = maybe_access_field (decl, *where_found, type);
+             if (decl == error_mark_node)
+               return 1;
+           }
+         /* And code for the array reference expression */
+         decl = java_complete_tree (qual_wfl);
+         if (decl == error_mark_node)
+           return 1;
+         type = QUAL_DECL_TYPE (decl);
+         continue;
+       }
+
+      /* If we fall here, we weren't processing a (static) function call. */
+      previous_call_static = 0;
+
+      /* It can be the keyword THIS */
+      if (EXPR_WFL_NODE (qual_wfl) == this_identifier_node)
+       {
+         if (!current_this)
+           {
+             parse_error_context 
+               (wfl, "Keyword `this' used outside allowed context");
+             return 1;
+           }
+         /* We have to generate code for intermediate acess */
+         *where_found = decl = current_this;
+         type = QUAL_DECL_TYPE (decl);
+         continue;
+       }
+
+      /* 15.10.2 Accessing Superclass Members using SUPER */
+      if (EXPR_WFL_NODE (qual_wfl) == super_identifier_node)
+       {
+         tree node;
+         /* Check on the restricted use of SUPER */
+         if (METHOD_STATIC (current_function_decl)
+             || current_class == object_type_node)
+           {
+             parse_error_context 
+               (wfl, "Keyword `super' used outside allowed context");
+             return 1;
+           }
+         /* Otherwise, treat SUPER as (SUPER_CLASS)THIS */
+         node = build_cast (EXPR_WFL_LINECOL (qual_wfl), 
+                            CLASSTYPE_SUPER (current_class),
+                            build_this (EXPR_WFL_LINECOL (qual_wfl)));
+         *where_found = decl = java_complete_tree (node);
+         *type_found = type = QUAL_DECL_TYPE (decl);
+         from_super = from_type = 1;
+         continue;
+       }
+
+      /* 15.13.1: Can't search for field name in packages, so we
+        assume a variable/class name was meant. */
+      if (RESOLVE_PACKAGE_NAME_P (qual_wfl))
+       {
+         if (from_super || from_cast)
+           parse_error_context 
+             ((from_cast ? qual_wfl : wfl),
+              "No variable `%s' defined in class `%s'",
+              IDENTIFIER_POINTER (EXPR_WFL_NODE (qual_wfl)),
+              lang_printable_name (type));
+         else
+           parse_error_context
+             (qual_wfl, "Undefined variable or class name: `%s'",
+              IDENTIFIER_POINTER (EXPR_WFL_NODE (qual_wfl)));
+         return 1;
+       }
+
+      /* We have a type name. It's been already resolved when the
+        expression was qualified. */
+      else if (RESOLVE_TYPE_NAME_P (qual_wfl))
+       {
+         if (!(decl = QUAL_RESOLUTION (q)))
+           return 1;           /* Error reported already */
+
+         if (not_accessible_p (TREE_TYPE (decl), decl, 0))
+           {
+             parse_error_context 
+               (qual_wfl, "Can't access %s field `%s.%s' from `%s'",
+                java_accstring_lookup (get_access_flags_from_decl (decl)),
+                IDENTIFIER_POINTER (DECL_NAME (TYPE_NAME (type))),
+                IDENTIFIER_POINTER (DECL_NAME (decl)),
+                IDENTIFIER_POINTER (DECL_NAME (TYPE_NAME (current_class))));
+             return 1;
+           }
+         
+         type = TREE_TYPE (decl);
+         from_type = 1;
+       }
+      /* We resolve and expression name */
+      else 
+       {
+         tree field_decl;
+
+         /* If there exists an early resolution, use it. That occurs
+            only once and we know that there are more things to
+            come. Don't do that when processing something after SUPER
+            (we need more thing to be put in place below */
+         if (!from_super && QUAL_RESOLUTION (q))
+           decl = QUAL_RESOLUTION (q);
+
+         /* We have to search for a field, knowing the type of its
+             container. The flag FROM_TYPE indicates that we resolved
+             the last member of the expression as a type name, which
+             means that for the resolution of this field, will check
+             on other errors than if the it was resolved as a member
+             of an other field. */
+         else
+           {
+             int is_static;
+             if (!from_type && !JREFERENCE_TYPE_P (type))
+               {
+                 parse_error_context 
+                   (qual_wfl, "Attempt to reference field `%s' in `%s %s'",
+                    IDENTIFIER_POINTER (EXPR_WFL_NODE (qual_wfl)),
+                    lang_printable_name (type),
+                    IDENTIFIER_POINTER (DECL_NAME (field_decl)));
+                 return 1;
+               }
+             
+             if (!(field_decl = 
+                   lookup_field_wrapper (type, EXPR_WFL_NODE (qual_wfl))))
+               {
+                 parse_error_context 
+                   (qual_wfl, "No variable `%s' defined in class `%s'",
+                    IDENTIFIER_POINTER (EXPR_WFL_NODE (qual_wfl)), 
+                    IDENTIFIER_POINTER (DECL_NAME (TYPE_NAME (type))));
+                 return 1;
+               }
+             
+             /* Check on accessibility here */
+             if (not_accessible_p (type, field_decl, from_super))
+               {
+                 parse_error_context 
+                   (qual_wfl,
+                    "Can't access %s field `%s.%s' from `%s'",
+                    java_accstring_lookup 
+                      (get_access_flags_from_decl (field_decl)),
+                    IDENTIFIER_POINTER (DECL_NAME (TYPE_NAME (type))),
+                    IDENTIFIER_POINTER (DECL_NAME (field_decl)),
+                    IDENTIFIER_POINTER 
+                      (DECL_NAME (TYPE_NAME (current_class))));
+                 return 1;
+               }
+             
+             /* There are things to check when fields are accessed
+                from type. There are no restrictions on a static
+                declaration of the field when it is accessed from an
+                interface */
+             is_static = FIELD_STATIC (field_decl);
+             if (!from_super && from_type 
+                 && !TYPE_INTERFACE_P (type) && !is_static)
+               {
+                 parse_error_context 
+                   (qual_wfl, "Can't make a static reference to nonstatic "
+                    "variable `%s' in class `%s'",
+                    IDENTIFIER_POINTER (EXPR_WFL_NODE (qual_wfl)),
+                    IDENTIFIER_POINTER (DECL_NAME (TYPE_NAME (type))));
+                 return 1;
+               }
+             from_cast = from_super = 0;
+
+             /* If we need to generate something to get a proper handle
+                on what this field is accessed from, do it now. */
+             if (!is_static)
+               {
+                 decl = maybe_access_field (decl, *where_found, type);
+                 if (decl == error_mark_node)
+                   return 1;
+               }
+
+             /* We want to keep the location were found it, and the type
+                we found. */
+             *where_found = decl;
+             *type_found = type;
+
+             /* This is the decl found and eventually the next one to
+                search from */
+             decl = field_decl;
+           }
+
+         from_type = 0;
+         type = QUAL_DECL_TYPE (decl);
+       }
+    }
+  *found_decl = decl;
+  return 0;
+}
+
+/* 6.6 Qualified name and access control. Returns 1 if MEMBER (a decl)
+   can't be accessed from REFERENCE (a record type). */
+
+int not_accessible_p (reference, member, from_super)
+     tree reference, member;
+     int from_super;
+{
+  int access_flag = get_access_flags_from_decl (member);
+
+  /* Access always granted for members declared public */
+  if (access_flag & ACC_PUBLIC)
+    return 0;
+  
+  /* Check access on protected members */
+  if (access_flag & ACC_PROTECTED)
+    {
+      /* Access granted if it occurs from within the package
+         containing the class in which the protected member is
+         declared */
+      if (class_in_current_package (DECL_CONTEXT (member)))
+       return 0;
+
+      if (TREE_CODE (member) == FUNCTION_DECL && DECL_CONSTRUCTOR_P (member))
+       {
+         /* Access from SUPER is granted */
+         if (from_super)
+           return 0;
+         /* Otherwise, access isn't granted */
+         return 1;
+       }
+      else
+       {
+         /* If accessed with the form `super.member', then access is
+             granted */
+         if (from_super)
+           return 0;
+
+         /* Otherwise, access is granted if occuring from the class where
+            member is declared or a subclass of it */
+         if (inherits_from_p (reference, current_class))
+           return 0;
+       }
+      return 1;
+    }
+
+  /* Check access on private members. Access is granted only if it
+     occurs from within the class in witch it is declared*/
+
+  if (access_flag & ACC_PRIVATE)
+    return (current_class == DECL_CONTEXT (member) ? 0 : 1);
+
+  /* Default access are permitted only when occuring within the
+     package in which the type (REFERENCE) is declared. In other words,
+     REFERENCE is defined in the current package */
+  if (ctxp->package)
+    return !class_in_current_package (reference);
+  
+  /* Otherwise, access is granted */
+  return 0;
+}
+
+/* Returns 1 if class was declared in the current package, 0 otherwise */
+
+static int
+class_in_current_package (class)
+     tree class;
+{
+  static tree cache = NULL_TREE;
+  int qualified_flag;
+  tree left;
+
+  if (cache == class)
+    return 1;
+
+  qualified_flag = QUALIFIED_P (DECL_NAME (TYPE_NAME (class)));
+
+  /* If the current package is empty and the name of CLASS is
+     qualified, class isn't in the current package.  If there is a
+     current package and the name of the CLASS is not qualified, class
+     isn't in the current package */
+  if (!ctxp->package && qualified_flag || ctxp->package && !qualified_flag)
+    return 0;
+
+  /* If there is not package and the name of CLASS isn't qualified,
+     they belong to the same unnamed package */
+  if (!ctxp->package && !qualified_flag)
+    return 1;
+
+  /* Compare the left part of the name of CLASS with the package name */
+  breakdown_qualified (&left, NULL, DECL_NAME (TYPE_NAME (class)));
+  if (ctxp->package == left)
+    {
+      cache = class;
+      return 1;
+    }
+  return 0;
+}
+
+/* This function may generate code to access DECL from WHERE. This is
+   done only if certain conditions meet.  */
+
+static tree
+maybe_access_field (decl, where, type)
+  tree decl, where, type;
+{
+  if (DECL_P (decl) && decl != current_this
+      && (!(TREE_CODE (decl) != PARM_DECL
+           && FIELD_STATIC (decl)))
+      && !IDENTIFIER_LOCAL_VALUE (DECL_NAME (decl)))
+    decl = build_field_ref (where ? where : current_this, 
+                           type, DECL_NAME (decl));
+  return decl;
+}
+
+/* Build a method invocation statement, by patching PATCH. If non NULL
+   and according to the situation, PRIMARY and WHERE may be
+   used. IS_STATIC is set to 1 if the invoked function is static. */
+
+static tree
+patch_method_invocation_stmt (patch, primary, where, is_static)
+     tree patch, primary, where;
+     int *is_static;
+{
+  tree wfl = TREE_OPERAND (patch, 0);
+  tree args = TREE_OPERAND (patch, 1);
+  tree name = EXPR_WFL_NODE (wfl);
+  tree list, class_type;
+  
+  /* Should be overriden if everything goes well. Otherwise, if
+     something fails, it should keep this value. It stop the
+     evaluation of a bogus assignment. See java_complete_tree,
+     MODIFY_EXPR: for the reasons why we sometimes want to keep on
+     evaluating an assignment */
+  TREE_TYPE (patch) = error_mark_node;
+
+  /* Since lookup functions are messing with line numbers, save the
+     context now.  */
+  java_parser_context_save_global ();
+
+  /* 15.11.1: Compile-Time Step 1: Determine Class or Interface to Search */
+
+  /* Resolution of qualified name, excluding constructors */
+  if (QUALIFIED_P (name) && !CALL_CONSTRUCTOR_P (patch))
+    {
+      tree class_decl, identifier, identifier_wfl;
+      /* Extract the last IDENTIFIER of the qualified
+        expression. This is a wfl and we will use it's location
+        data during error report. */
+      identifier_wfl = cut_identifier_in_qualified (wfl);
+      identifier = EXPR_WFL_NODE (identifier_wfl);
+      
+      /* Given the context, IDENTIFIER is syntactically qualified
+        as a MethodName. We need to qualify what's before */
+      qualify_ambiguous_name (wfl);
+
+      /* Package resolution are erroneous */
+      if (RESOLVE_PACKAGE_NAME_P (wfl))
+       {
+         tree remainder;
+         breakdown_qualified (&remainder, NULL, EXPR_WFL_NODE (wfl));
+         parse_error_context (wfl, "Can't search method `%s' in package "
+                              "`%s'",IDENTIFIER_POINTER (identifier),
+                              IDENTIFIER_POINTER (remainder));
+         return error_mark_node;
+       }
+      /* We're resolving a call from a type */
+      else if (RESOLVE_TYPE_NAME_P (wfl))
+       {
+         tree decl = QUAL_RESOLUTION (EXPR_WFL_QUALIFICATION (wfl));
+         tree name = DECL_NAME (decl);
+         tree type;
+
+         class_decl = resolve_and_layout (name, wfl);
+         if (CLASS_INTERFACE (decl))
+           {
+             parse_error_context
+               (identifier_wfl, "Can't make static reference to method "
+                "`%s' in interface `%s'", IDENTIFIER_POINTER (identifier), 
+                IDENTIFIER_POINTER (name));
+             return error_mark_node;
+           }
+         /* Look the method up in the type selector. The method ought
+             to be static. */
+         type = TREE_TYPE (class_decl);
+         list = lookup_method_invoke (0, wfl, type, identifier, args);
+         if (list && !METHOD_STATIC (list))
+           {
+             char *fct_name = strdup ((char *)lang_printable_name (list));
+             parse_error_context 
+               (identifier_wfl,
+                "Can't make static reference to method `%s %s' in class `%s'",
+                lang_printable_name (TREE_TYPE (TREE_TYPE (list))), fct_name, 
+                IDENTIFIER_POINTER (DECL_NAME (TYPE_NAME (type))));
+             free (fct_name);
+             return error_mark_node;
+           }
+       }
+      /* We're resolving an expression name */
+      else
+       {
+         tree field, type;
+         
+         /* 1- Find the field to which the call applies */
+         field = resolve_field_access (wfl, NULL, &type);
+         if (field == error_mark_node)
+           return error_mark_node;
+         
+         /* 2- Do the layout of the class where the last field
+            was found, so we can search it. */
+         class_decl = 
+           resolve_and_layout (DECL_NAME (TYPE_NAME (type)), NULL_TREE);
+         
+         /* 3- Retrieve a filtered list of method matches, Refine
+            if necessary. In any cases, point out errors.  */
+         list = lookup_method_invoke (0, identifier_wfl, type, 
+                                      identifier, args);
+
+         /* 4- Add the field as an argument */
+         args = tree_cons (NULL_TREE, field, args);
+       }
+
+      /* CLASS_TYPE is used during the call to not_accessible_p and
+        IDENTIFIER_WFL will be used to report any problem further */
+      class_type = TREE_TYPE (class_decl);
+      wfl = identifier_wfl;
+    }
+  /* Resolution of simple names, names generated after a primary: or
+     constructors */
+  else
+    {
+      tree class_to_search;
+      int lc;          /* Looking for Constructor */
+      
+      /* We search constructor in their target class */
+      if (CALL_CONSTRUCTOR_P (patch))
+       {
+         class_to_search = resolve_no_layout (EXPR_WFL_NODE (wfl), NULL_TREE);
+         if (!class_to_search)
+           {
+             parse_error_context 
+               (wfl, "Class `%s' not found in type declaration",
+                IDENTIFIER_POINTER (EXPR_WFL_NODE (wfl)));
+             return error_mark_node;
+           }
+         
+         /* Can't instantiate an abstract class */
+         if (CLASS_ABSTRACT (class_to_search))
+           {
+             parse_error_context 
+               (wfl, "Class `%s' is an abstract class. It can't be "
+                "instantiated", IDENTIFIER_POINTER (EXPR_WFL_NODE (wfl)));
+             return error_mark_node;
+           }
+         class_to_search = TREE_TYPE (class_to_search);
+         lc = 1;
+       }
+      /* This is a regular search in the local class, unless an
+         alternate class is specified. */
+      else
+       {
+         class_to_search = (where ? where : current_class);
+         lc = 0;
+       }
+      
+      /* NAME is a simple identifier or comes from a primary. Search
+        in the class whose declaration contain the method being
+        invoked. */
+      list = lookup_method_invoke (lc, wfl, class_to_search, name, args);
+
+      /* Don't continue if no method were found, as the next statement
+         can't be executed then. */
+      if (!list) return error_mark_node;
+
+      /* Check for static reference if non static methods */
+      if (check_for_static_method_reference (wfl, patch, list, 
+                                            class_to_search, primary))
+       return error_mark_node;
+
+      /* Non static/constructor methods are called with the current
+        object extra argument. If method is resolved as a primary,
+        use the primary otherwise use the current THIS. */
+      if (!CALL_CONSTRUCTOR_P (patch) && !METHOD_STATIC (list))
+       args = tree_cons (NULL_TREE, primary ? primary : current_this, args);
+
+      class_type = class_to_search;
+    }
+  
+  /* Merge point of all resolution schemes. If we have nothing, this
+     is an error, already signaled */
+  if (!list) return error_mark_node;
+  
+  /* Check accessibility, position the is_static flag, build and
+     return the call */
+  if (not_accessible_p (class_type, list, 0))
+    {
+      char *fct_name = strdup ((char *)lang_printable_name (list));
+      parse_error_context 
+       (wfl, "Can't access %s method `%s %s.%s' from `%s'",
+        java_accstring_lookup (get_access_flags_from_decl (list)),
+        lang_printable_name (TREE_TYPE (TREE_TYPE (list))), 
+        IDENTIFIER_POINTER (DECL_NAME (TYPE_NAME (class_type))), fct_name,
+        IDENTIFIER_POINTER (DECL_NAME (TYPE_NAME (current_class))));
+      free (fct_name);
+      return error_mark_node;
+    }
+  
+  if (is_static) 
+    *is_static = METHOD_STATIC (list);
+  java_parser_context_restore_global ();
+  return patch_invoke (patch, list, args, wfl);
+}
+
+/* Check that we're not trying to do a static reference to a method in
+   non static method. Return 1 if it's the case, 0 otherwise. */
+
+static int
+check_for_static_method_reference (wfl, node, method, where, primary)
+     tree wfl, node, method, where, primary;
+{
+  if (METHOD_STATIC (current_function_decl) 
+      && !METHOD_STATIC (method) && !primary && !CALL_CONSTRUCTOR_P (node))
+    {
+      char *fct_name = strdup ((char *)lang_printable_name (method));
+      parse_error_context 
+       (wfl, "Can't make static reference to method `%s %s' in class `%s'", 
+        lang_printable_name (TREE_TYPE (TREE_TYPE (method))), fct_name,
+        IDENTIFIER_POINTER (DECL_NAME (TYPE_NAME (where))));
+      free (fct_name);
+      return 1;
+    }
+  return 0;
+}
+
+/* Patch an invoke expression METHOD and ARGS, based on its invocation
+   mode.  */
+
+static tree
+patch_invoke (patch, method, args, cl)
+     tree patch, method, args;
+     tree cl;
+{
+  tree dtable, func;
+  tree signature = build_java_signature (TREE_TYPE (method));
+  tree original_call;
+
+  switch (invocation_mode (method, 0))
+    {
+    case INVOKE_VIRTUAL:
+      dtable = invoke_build_dtable (0, args);
+      func = build_invokevirtual (dtable, method);
+      break;
+    case INVOKE_STATIC:
+      func = build_known_method_ref (method, TREE_TYPE (method),
+                                    DECL_CONTEXT (method),
+                                    signature, args);
+      args = nreverse (args);
+      break;
+
+    default:
+      fatal ("Unknown invocation mode - build_invoke");
+      return NULL_TREE;
+    }
+
+
+  /* Ensure self_type is initialized, (invokestatic). FIXME */
+  func = build1 (NOP_EXPR, build_pointer_type (TREE_TYPE (method)), func);
+  TREE_TYPE (patch) = TREE_TYPE (TREE_TYPE (method));
+  TREE_OPERAND (patch, 0) = func;
+  TREE_OPERAND (patch, 1) = args;
+  original_call = patch;
+
+  /* We're calling a constructor. New is called an its returned value
+     is an argument to the constructor. We build a COMPOUND_EXPR and
+     use saved expression so that the overall NEW expression value is
+     a pointer to a newly created and initialized class. */
+  if (CALL_CONSTRUCTOR_P (original_call))
+    {
+      tree class = DECL_CONTEXT (method);
+      tree c1, saved_new, size, new;
+      if (!TYPE_SIZE (class))
+       safe_layout_class (class);
+      size = size_in_bytes (class);
+      new = build (CALL_EXPR, promote_type (class),
+                  build_address_of (alloc_object_node),
+                  tree_cons (NULL_TREE, build_class_ref (class),
+                             build_tree_list (NULL_TREE, 
+                                              size_in_bytes (class))),
+                  NULL_TREE);
+      saved_new = save_expr (new);
+      c1 = build_tree_list (NULL_TREE, saved_new);
+      TREE_CHAIN (c1) = TREE_OPERAND (original_call, 1);
+      TREE_OPERAND (original_call, 1) = c1;
+      TREE_SET_CODE (original_call, CALL_EXPR);
+      patch = build (COMPOUND_EXPR, TREE_TYPE (new), patch, saved_new);
+    }
+  return patch;
+}
+
+static int
+invocation_mode (method, super)
+     tree method;
+     int super;
+{
+  int access = get_access_flags_from_decl (method);
+
+  if (access & ACC_STATIC)
+    return INVOKE_STATIC;
+
+  if (CLASS_FINAL (TYPE_NAME (DECL_CONTEXT (method))))
+    return INVOKE_STATIC;
+  
+  if (super)
+    return INVOKE_SUPER;
+  
+  if (CLASS_INTERFACE (TYPE_NAME (DECL_CONTEXT (method))))
+    return INVOKE_INTERFACE;
+  
+  if (DECL_CONSTRUCTOR_P (method))
+    return INVOKE_STATIC;
+  
+  return INVOKE_VIRTUAL;
+}
+
+/* Retrieve a refined list of matching methods. */
+
+static tree
+lookup_method_invoke (lc, cl, class, name, arg_list)
+     int lc;
+     tree cl;
+     tree class, name, arg_list;
+{
+  tree method = make_node (FUNCTION_TYPE);
+  tree arg_type_list = NULL_TREE;
+  tree signature, list, node, scratch;
+
+  for (node = arg_list; node; node = TREE_CHAIN (node))
+    {
+      tree current_arg;
+      current_arg = 
+       build_tree_list (NULL_TREE,
+                        promote_type (TREE_TYPE (TREE_VALUE (node))));
+      arg_type_list = chainon (current_arg, arg_type_list);
+    }
+  TYPE_ARG_TYPES (method) = arg_type_list;
+
+  if (!lc)
+    {
+      signature = build_java_argument_signature (method);
+      list = match_java_method (class, name, signature);
+      list = refine_accessible_methods_list (lc, list);
+    }
+  else
+    {
+      TREE_TYPE (method) = void_type_node;
+      signature = build_java_signature (method);
+      list = lookup_java_constructor (class, signature);
+    }
+
+  if (!list)
+    {
+      parse_error_context (cl, "Can't find method `%s(%s)' in class `%s'",
+                          IDENTIFIER_POINTER (name),
+                          IDENTIFIER_POINTER (signature),
+                          IDENTIFIER_POINTER (DECL_NAME (TYPE_NAME (class))));
+      return NULL_TREE;
+    }
+
+  if (lc)
+    return list;
+
+  if (TREE_CHAIN (list))
+    {
+      tree most_specific_list = NULL_TREE;
+      tree current;
+      /* 15.11.2.2 Choose the Most Specific Method */
+      for (current = list; current; current = TREE_CHAIN (current))
+       {
+         tree rest;
+         tree method = TREE_VALUE (list);
+         tree class_from = DECL_CONTEXT (method);
+         for (rest = TREE_CHAIN (current); rest; rest = TREE_CHAIN (rest))
+           {
+             tree other = TREE_VALUE (rest);
+
+             /* METHOD can be declared more specific with regard to OTHER iif:
+               
+                - The class METHOD belongs can be converted to the
+                  class OTHER belongs by method invocation conversion
+                  (5.3).  Since we're dealing with classes here, it is
+                  covered by the identity conversion or the windening
+                  primitive conversion.
+               
+                - The types of the arguments of METHOD can be
+                  converted to the types of the arguments of OTHER by
+                  method invocation conversion (5.3). */
+
+             if (valid_ref_assignconv_cast_p (class_from, 
+                                              DECL_CONTEXT (other), 0)
+                 && 1)         /* Test on args non implemented */
+               most_specific_list = tree_cons (NULL_TREE, method, 
+                                               most_specific_list);
+           }
+       }
+      list = most_specific_list;
+    }
+
+  if (!list || TREE_CHAIN (list))
+    {
+      parse_error_context (cl, "Can't find method `%s(%s)' in class `%s'",
+                          IDENTIFIER_POINTER (name),
+                          IDENTIFIER_POINTER (signature),
+                          IDENTIFIER_POINTER (DECL_NAME (TYPE_NAME (class))));
+      return NULL_TREE;
+    }
+
+  /* 15.11.3 Is the Chosen Method Appropriate ? */
+  else
+    return TREE_VALUE (list);
+}
+
+/* Refine accessible methods from the raw matching method list, as
+   specified in 15.11.4.3. Return a (possibly empty) new method
+   list.  */
+
+static tree
+refine_accessible_methods_list (lc, list)
+     int lc;                   /* Looking for Constructor */
+     tree list;
+{
+#define ADD_TO_LIST_AND_CONTINUE                               \
+  {                                                            \
+    refined_list = tree_cons (NULL_TREE, method, refined_list);        \
+    continue;                                                  \
+  }
+  tree node, refined_list = NULL_TREE;
+  tree current_class_name = DECL_NAME (TYPE_NAME (current_class));
+
+  for (node = list; node; node = TREE_CHAIN (node))
+    {
+      int access, identical;
+      tree class_from, method, class_from_name;
+      
+      method = TREE_VALUE (node);
+
+      /* Constructor not retained here, unless were specifically
+       looking for them. */
+      if (lc && DECL_CONSTRUCTOR_P (method))
+       ADD_TO_LIST_AND_CONTINUE;
+
+      access = get_access_flags_from_decl (method);
+      class_from = DECL_CONTEXT (method);
+      class_from_name = DECL_NAME (TYPE_NAME (class_from));
+      
+      identical = identical_subpath_p (current_class_name, class_from_name);
+
+      /* Check accessibility of class_from from the current one: This
+        test has been already carried out when qualify_ambiguous_name
+        tried to resolve a type found in an other package. It is not
+        necessary to retest things here, the error has been already
+        reported. */
+                 
+      /* Public method are always OK */
+      if (access & ACC_PUBLIC)
+       ADD_TO_LIST_AND_CONTINUE;
+      
+      /* Protected method access is OK if classes are from the
+        same package or part of the same inheritance lineage */
+      if ((access & ACC_PROTECTED)
+         && (inherits_from_p (current_class, class_from) || identical))
+       ADD_TO_LIST_AND_CONTINUE;
+
+      /* Methods with default (package) access are OK if classes are
+        from the same default package. */
+      if (identical || 
+         (!QUALIFIED_P (class_from_name) && !QUALIFIED_P (current_class_name)))
+       ADD_TO_LIST_AND_CONTINUE;
+
+      /* Private method accessible iff current class is the node where
+        the method is defined */
+      if ((access & ACC_PRIVATE) && (class_from == current_class))
+       ADD_TO_LIST_AND_CONTINUE;
+    }
+#undef ADD_TO_LIST_AND_CONTINUE
+  return refined_list;
+}
+
+/* Qualification routines */
+
+static void
+qualify_ambiguous_name (id)
+     tree id;
+{
+  tree qual, qual_wfl, name, decl, ptr_type, saved_current_class;
+  int again, super_found = 0, this_found = 0;
+
+  /* We first qualify the first element, then derive qualification of
+     others based on the first one. If the first element is qualified
+     by a resolution (field or type), this resolution is stored in the
+     QUAL_RESOLUTION of the qual element being examined. We need to
+     save the current_class since the use of SUPER might change the
+     its value. */
+  saved_current_class = current_class;
+  qual = EXPR_WFL_QUALIFICATION (id);
+  do {
+
+    /* Simple qualified expression feature a qual_wfl that is a
+       WFL. Expression derived from a primary feature more complicated
+       things like a CALL_EXPR. Expression from primary need to be
+       worked out to extract the part on which the qualification will
+       take place. */
+    qual_wfl = QUAL_WFL (qual);
+    switch (TREE_CODE (qual_wfl))
+      {
+      case CALL_EXPR:
+       qual_wfl = TREE_OPERAND (qual_wfl, 0);
+       if (TREE_CODE (qual_wfl) != EXPR_WITH_FILE_LOCATION)
+         {
+           qual = EXPR_WFL_QUALIFICATION (qual_wfl);
+           qual_wfl = QUAL_WFL (qual);
+         }
+       break;
+      case JAVA_NEW_CLASS_EXPR:
+      case CONVERT_EXPR:
+      case ARRAY_REF:
+       qual_wfl = TREE_OPERAND (qual_wfl, 0);
+       break;
+      }
+    name = EXPR_WFL_NODE (qual_wfl);
+    ptr_type = current_class;
+    again = 0;
+    /* If we have a THIS (from a primary), we set the context accordingly */
+    if (name == this_identifier_node)
+      {
+       qual = TREE_CHAIN (qual);
+       qual_wfl = QUAL_WFL (qual);
+       name = EXPR_WFL_NODE (qual_wfl);
+       this_found = 1;
+      }
+    /* If we have a SUPER, we set the context accordingly */
+    if (name == super_identifier_node)
+      {
+       current_class = CLASSTYPE_SUPER (ptr_type);
+       /* Check that there is such a thing as a super class. If not,
+          return.  The error will be caught later on, during the
+          resolution */
+       if (!current_class)
+         {
+           current_class = saved_current_class;
+           return;
+         }
+       qual = TREE_CHAIN (qual);
+       /* Do one more interation to set things up */
+       super_found = again = 1;
+      }
+  } while (again);
+  
+  /* If name appears within the scope of a location variable
+     declaration or parameter declaration, then it is an expression
+     name. We don't carry this test out if we're in the context of the
+     use of SUPER or THIS */
+
+  if (!this_found && !super_found && (decl = IDENTIFIER_LOCAL_VALUE (name)))
+    {
+      RESOLVE_EXPRESSION_NAME_P (qual_wfl) = 1;
+      QUAL_RESOLUTION (qual) = decl;
+    }
+
+  /* If within the class/interface NAME was found to be used there
+     exists a (possibly inherited) field named NAME, then this is an
+     expression name. */
+  else if ((decl = lookup_field_wrapper (ptr_type, name)))
+    {
+      RESOLVE_EXPRESSION_NAME_P (qual_wfl) = 1;
+      QUAL_RESOLUTION (qual) = decl;
+    }
+
+  /* We reclassify NAME as a type name if:
+     - NAME is a class/interface declared within the compilation
+       unit containing NAME,
+     - NAME is imported via a single-type-import declaration,
+     - NAME is declared in an another compilation unit of the package
+       of the compilation unit containing NAME,
+     - NAME is declared by exactly on type-import-on-demand declaration
+     of the compilation unit containing NAME. */
+  else if ((decl = resolve_and_layout (name, NULL_TREE)))
+    {
+      RESOLVE_TYPE_NAME_P (qual_wfl) = 1;
+      QUAL_RESOLUTION (qual) = decl;
+    }
+
+  /* Method call are expression name */
+  else if (TREE_CODE (QUAL_WFL (qual)) == CALL_EXPR)
+    RESOLVE_EXPRESSION_NAME_P (qual_wfl) = 1;
+
+  /* Check here that NAME isn't declared by more than one
+     type-import-on-demand declaration of the compilation unit
+     containing NAME. FIXME */
+
+  /* Otherwise, NAME is reclassified as a package name */
+  else 
+    RESOLVE_PACKAGE_NAME_P (qual_wfl) = 1;
+
+  /* Propagate the qualification accross other components of the
+     qualified name */
+  for (qual = TREE_CHAIN (qual); qual;
+       qual_wfl = QUAL_WFL (qual), qual = TREE_CHAIN (qual))
+    {
+      if (RESOLVE_PACKAGE_NAME_P (qual_wfl))
+       RESOLVE_PACKAGE_NAME_P (QUAL_WFL (qual)) = 1;
+      else 
+       RESOLVE_EXPRESSION_NAME_P (QUAL_WFL (qual)) = 1;
+    }
+
+  /* Store the global qualification for the ambiguous part of ID back
+     into ID fields */
+  if (RESOLVE_EXPRESSION_NAME_P (qual_wfl))
+    RESOLVE_EXPRESSION_NAME_P (id) = 1;
+  else if (RESOLVE_TYPE_NAME_P (qual_wfl))
+    RESOLVE_TYPE_NAME_P (id) = 1;
+  else if (RESOLVE_PACKAGE_NAME_P (qual_wfl))
+    RESOLVE_PACKAGE_NAME_P (id) = 1;
+
+  /* Restore the current class */
+  current_class = saved_current_class;
+}
+
+static int
+breakdown_qualified (left, right, source)
+    tree *left, *right, source;
+{
+  char *p = IDENTIFIER_POINTER (source), *base;
+  int   l = IDENTIFIER_LENGTH (source);
+
+  /* Breakdown NAME into REMAINDER . IDENTIFIER */
+  base = p;
+  p += (l-1);
+  while (*p != '.' && p != base)
+    p--;
+
+  /* We didn't find a '.'. Return an error */
+  if (p == base)
+    return 1;
+
+  *p = '\0';
+  if (right)
+    *right = get_identifier (p+1);
+  *left = get_identifier (IDENTIFIER_POINTER (source));
+  *p = '.';
+  
+  return 0;
+}
+
+/* Return 1 if N1 and N2 have identical sub-path. */
+
+static int
+identical_subpath_p (n1, n2)
+     tree n1, n2;
+{
+  tree left1, left2;
+
+  if (!QUALIFIED_P (n1) || !QUALIFIED_P (n2))
+    return n1 == n2;
+  
+  breakdown_qualified (&left1, NULL, n1);
+  breakdown_qualified (&left2, NULL, n2);
+
+  return left1 == left2;
+}
+
+static int
+not_initialized_as_it_should_p (decl)
+     tree decl;
+{
+  if (DECL_P (decl))
+    {
+      if (TREE_CODE (decl) == FIELD_DECL
+         && METHOD_STATIC (current_function_decl))
+       return 0;
+      return DECL_P (decl) && !INITIALIZED_P (decl);
+    }
+  return 0;
+}
+
+/* Patch tree nodes in a function body. When a BLOCK is found, push
+   local variable decls if present.  */
+
+static tree
+java_complete_tree (node)
+     tree node;
+{
+  tree nn, cn, wfl_op1, wfl_op2;
+  int flag, location;
+
+  /* CONVERT_EXPR always has its type set, even though it needs to be
+     worked out */
+  if (TREE_TYPE (node) && TREE_CODE (node) != CONVERT_EXPR)
+    return node;
+
+  /* The switch block implements cases processing container nodes
+     first.  Contained nodes are always written back. Leaves come
+     next and return a value. */
+  switch (TREE_CODE (node))
+    {
+    case BLOCK:
+
+      /* 1- Block section.
+        Set the local values on decl names so we can identify them
+        faster when they're referenced. At that stage, identifiers
+        are legal so we don't check for declaration errors. */
+      for (cn = BLOCK_EXPR_DECLS (node); cn; cn = TREE_CHAIN (cn))
+       {
+         DECL_CONTEXT (cn) = current_function_decl;
+         IDENTIFIER_LOCAL_VALUE (DECL_NAME (cn)) = cn;
+         INITIALIZED_P (cn) = 0;
+       }
+      if (BLOCK_EXPR_BODY (node))
+       {
+         BLOCK_EXPR_BODY (node) = java_complete_tree (BLOCK_EXPR_BODY (node));
+         if (BLOCK_EXPR_BODY (node) == error_mark_node)
+           return error_mark_node;
+       }
+      /* Turn local bindings to null */
+      for (cn = BLOCK_EXPR_DECLS (node); cn; cn = TREE_CHAIN (cn))
+       IDENTIFIER_LOCAL_VALUE (DECL_NAME (cn)) = NULL_TREE;
+
+      TREE_TYPE (node) = void_type_node;
+      break;
+
+      /* 2- They are expressions but ultimately deal with statements */
+    case LABELED_BLOCK_EXPR:
+      PUSH_LABELED_BLOCK (node);
+      if (LABELED_BLOCK_BODY (node))
+       COMPLETE_CHECK_OP_1 (node);
+      TREE_TYPE (node) = void_type_node;
+      POP_LABELED_BLOCK ();
+      return node;
+
+    case EXIT_BLOCK_EXPR:
+      /* We don't complete operand 1, because it's the return value of
+         the EXIT_BLOCK_EXPR which doesn't exist it Java */
+      return patch_bc_statement (node);
+
+    case LOOP_EXPR:
+      PUSH_LOOP (node);
+      /* Check whether the loop was enclosed in a labeled
+         statement. If not, create one, insert the loop in it and
+         return the node */
+      nn = patch_loop_statement (node);
+      /* Anyways, walk the body of the loop */
+      TREE_OPERAND (node, 0) = java_complete_tree (TREE_OPERAND (node, 0));
+      if (TREE_OPERAND (node, 0) == error_mark_node)
+       return error_mark_node;
+      TREE_TYPE (nn) = TREE_TYPE (node) = void_type_node;
+      /* If we returned something different, that's because we
+         inserted a label. Pop the label too. */
+      if (nn != node)
+       POP_LABELED_BLOCK ();
+      POP_LOOP ();
+      return nn;
+
+    case EXIT_EXPR:
+      TREE_OPERAND (node, 0) = java_complete_tree (TREE_OPERAND (node, 0));
+      return patch_exit_expr (node);
+
+    case COND_EXPR:
+      /* Condition */
+      TREE_OPERAND (node, 0) = java_complete_tree (TREE_OPERAND (node, 0));
+      if (TREE_OPERAND (node, 0) == error_mark_node)
+       return error_mark_node;
+      /* then-else branches */
+      TREE_OPERAND (node, 1) = java_complete_tree (TREE_OPERAND (node, 1));
+      if (TREE_OPERAND (node, 1) == error_mark_node)
+       return error_mark_node;
+      TREE_OPERAND (node, 2) = java_complete_tree (TREE_OPERAND (node, 2));
+      if (TREE_OPERAND (node, 2) == error_mark_node)
+       return error_mark_node;
+      return patch_if_else_statement (node);
+      break;
+
+      /* 3- Expression section */
+    case COMPOUND_EXPR:
+      TREE_OPERAND (node, 0) = java_complete_tree (TREE_OPERAND (node, 0));
+      TREE_OPERAND (node, 1) = java_complete_tree (TREE_OPERAND (node, 1));
+      if (TREE_OPERAND (node, 1) == error_mark_node)
+       return error_mark_node;
+      TREE_TYPE (node) = TREE_TYPE (TREE_OPERAND (node, 1));
+      break;
+
+    case RETURN_EXPR:
+      return patch_return (node);
+
+    case EXPR_WITH_FILE_LOCATION:
+      if (!EXPR_WFL_NODE (node) /* Or a PRIMARY flag ? */
+         || TREE_CODE (EXPR_WFL_NODE (node)) == IDENTIFIER_NODE)
+        return resolve_expression_name (node);
+      else
+       {
+         EXPR_WFL_NODE (node) = java_complete_tree (EXPR_WFL_NODE (node));
+         TREE_SIDE_EFFECTS (node) = 1;
+         if (EXPR_WFL_NODE (node) == error_mark_node)
+           {
+             /* Its important for the evaluation of assignment that
+                this mark on the TREE_TYPE is propagated. */
+             TREE_TYPE (node) = error_mark_node;
+             return error_mark_node;
+           }
+         else
+           TREE_TYPE (node) = TREE_TYPE (EXPR_WFL_NODE (node));
+       }
+      break;
+
+    case JAVA_NEW_ARRAY_EXPR:
+      /* Patch all the dimensions */
+      flag = 0;
+      for (cn = TREE_OPERAND (node, 1); cn; cn = TREE_CHAIN (cn))
+       {
+         int location = EXPR_WFL_LINECOL (TREE_VALUE (cn));
+         tree dim = java_complete_tree (TREE_VALUE (cn));
+         if (dim == error_mark_node)
+           {
+             flag = 1;
+             continue;
+           }
+         else
+           {
+             TREE_VALUE (cn) = save_expr (dim);
+             /* Setup the location of the current dimension, for
+                later error report. */
+             TREE_PURPOSE (cn) = 
+               build_expr_wfl (NULL_TREE, input_filename, 0, 0);
+             EXPR_WFL_LINECOL (TREE_PURPOSE (cn)) = location;
+           }
+       }
+      /* They complete the array creation expression, if no errors
+         were found. */
+      return (flag ? error_mark_node : patch_newarray (node));
+
+    case JAVA_NEW_CLASS_EXPR:
+    case CALL_EXPR:
+      /* Complete function's argument first */
+      if (complete_function_arguments (node))
+       return error_mark_node;
+      else
+       return patch_method_invocation_stmt (node, NULL_TREE, NULL_TREE, NULL);
+
+    case MODIFY_EXPR:
+      /* Save potential wfls */
+      wfl_op1 = TREE_OPERAND (node, 0);
+      wfl_op2 = TREE_OPERAND (node, 1);
+      TREE_OPERAND (node, 0) = java_complete_tree (wfl_op1);
+      if (TREE_OPERAND (node, 0) == error_mark_node)
+       return error_mark_node;
+
+      if (COMPOUND_ASSIGN_P (wfl_op2))
+       {
+         tree lvalue;
+         tree other = 
+           java_complete_tree (TREE_OPERAND (wfl_op2, 0));
+
+         /* Hand stablize the lhs on both places */
+         lvalue = stabilize_reference (other); 
+         TREE_OPERAND (node, 0) = lvalue;
+         TREE_OPERAND (TREE_OPERAND (node, 1), 0) = lvalue;
+       }
+
+      /* There are cases where the type of RHS is fixed. In those
+        cases, if the evaluation of the RHS fails, we further the
+        evaluation of the assignment to detect more errors. */
+      nn = java_complete_tree (TREE_OPERAND (node, 1));
+      if (nn == error_mark_node)
+       {
+         /* It's hopeless, but we can further things on to discover
+            an error during the assignment. In any cases, the
+            assignment operation fails. */
+         if (TREE_CODE (TREE_OPERAND (node, 1)) != EXPR_WITH_FILE_LOCATION
+             && TREE_TYPE (TREE_OPERAND (node, 1)) != error_mark_node)
+           patch_assignment (node, wfl_op1, wfl_op2);
+
+         /* Now, we still mark the lhs as initialized */
+         if (DECL_P (TREE_OPERAND (node, 0)))
+           INITIALIZED_P (TREE_OPERAND (node, 0)) = 1;
+
+         return error_mark_node;
+       }
+      TREE_OPERAND (node, 1) = nn;
+      return patch_assignment (node, wfl_op1, wfl_op2);
+
+    case MULT_EXPR:
+    case PLUS_EXPR:
+    case MINUS_EXPR:
+    case LSHIFT_EXPR:
+    case RSHIFT_EXPR:
+    case URSHIFT_EXPR:
+    case BIT_AND_EXPR:
+    case BIT_XOR_EXPR:
+    case BIT_IOR_EXPR:
+    case TRUNC_MOD_EXPR:
+    case RDIV_EXPR:
+    case TRUTH_ANDIF_EXPR:
+    case TRUTH_ORIF_EXPR:
+    case EQ_EXPR: 
+    case NE_EXPR:
+    case GT_EXPR:
+    case GE_EXPR:
+    case LT_EXPR:
+    case LE_EXPR:
+      /* Operands 0 and 1 are WFL in certain cases only. patch_binop
+        knows how to handle those cases. */
+      wfl_op1 = TREE_OPERAND (node, 0);
+      wfl_op2 = TREE_OPERAND (node, 1);
+      TREE_OPERAND (node, 0) = java_complete_tree (wfl_op1);
+      if (TREE_OPERAND (node, 0) == error_mark_node)
+        return error_mark_node;
+      TREE_OPERAND (node, 1) = java_complete_tree (wfl_op2);
+      if (TREE_OPERAND (node, 1) == error_mark_node)
+       return error_mark_node;
+      return patch_binop (node, wfl_op1, wfl_op2);
+
+    case JAVA_UNARY_PLUS_EXPR:
+    case NEGATE_EXPR:
+    case TRUTH_NOT_EXPR:
+    case BIT_NOT_EXPR:
+    case PREDECREMENT_EXPR:
+    case PREINCREMENT_EXPR:
+    case POSTDECREMENT_EXPR:
+    case POSTINCREMENT_EXPR:
+    case CONVERT_EXPR:
+      /* There are cases were wfl_op1 is a WFL. patch_unaryop knows
+        how to handle those cases. */
+      wfl_op1 = TREE_OPERAND (node, 0);
+      TREE_OPERAND (node, 0) = java_complete_tree (wfl_op1);
+      if (TREE_OPERAND (node, 0) == error_mark_node)
+       return error_mark_node;
+      return patch_unaryop (node, wfl_op1);
+
+    case ARRAY_REF:
+      /* There are cases were wfl_op1 is a WFL. patch_array_ref knows
+        how to handle those cases. */
+      wfl_op1 = TREE_OPERAND (node, 0);
+      TREE_OPERAND (node, 0) = java_complete_tree (wfl_op1);
+      if (TREE_OPERAND (node, 0) == error_mark_node)
+       return error_mark_node;
+      /* The same applies to wfl_op2 */
+      wfl_op2 = TREE_OPERAND (node, 1);
+      TREE_OPERAND (node, 1) = java_complete_tree (wfl_op2);
+      if (TREE_OPERAND (node, 1) == error_mark_node)
+       return error_mark_node;
+      return patch_array_ref (node, wfl_op1, wfl_op2);
+
+    case JAVA_THIS_EXPR:
+      /* Can't use THIS in a static environment */
+      if (!current_this)
+       {
+         EXPR_WFL_LINECOL (wfl_operator) = EXPR_WFL_LINECOL (node);
+         parse_error_context (wfl_operator, "Keyword `this' used outside "
+                              "allowed context");
+         TREE_TYPE (node) = error_mark_node;
+         return error_mark_node;
+       }
+      return current_this;
+
+    case STRING_CST:
+      /* Build the internal string representation */
+      push_obstacks (&permanent_obstack, &permanent_obstack);
+      node = get_identifier (TREE_STRING_POINTER (node));
+      location = alloc_name_constant (CONSTANT_String, node);
+      node = build_ref_from_constant_pool (location);
+      TREE_TYPE (node) = promote_type (string_type_node);
+      return node;
+
+    default:
+      fatal ("No case for tree code `%s' - java_complete_tree\n",
+            tree_code_name [TREE_CODE (node)]);
+    }
+  return node;
+}
+
+/* Complete function call's argument. Return a non zero value is an
+   error was found.  */
+
+static int
+complete_function_arguments (node)
+     tree node;
+{
+  int flag = 0;
+  tree cn;
+
+  for (cn = TREE_OPERAND (node, 1); cn; cn = TREE_CHAIN (cn))
+    {
+      tree wfl = TREE_VALUE (cn), parm;
+      parm = java_complete_tree (wfl);
+      if (parm == error_mark_node)
+       {
+         flag = 1;
+         continue;
+       }
+      if (TREE_CODE (TREE_TYPE (parm)) == RECORD_TYPE)
+       TREE_VALUE (cn) = convert (promote_type (TREE_TYPE (parm)), parm);
+      else
+       TREE_VALUE (cn) = save_expr (parm);
+      if (not_initialized_as_it_should_p (parm))
+       {
+         ERROR_VARIABLE_NOT_INITIALIZED (wfl, EXPR_WFL_NODE (wfl));
+         INITIALIZED_P (parm) = 1;
+       }
+    }
+  return flag;
+}
+
+/* Sometimes (for loops and variable initialized during their
+   declaration), we want to wrap a statement around a WFL and turn it
+   debugable.  */
+
+static tree
+build_debugable_stmt (location, stmt)
+    int location;
+    tree stmt;
+{
+  if (TREE_CODE (stmt) != EXPR_WITH_FILE_LOCATION)
+    {
+      stmt = build_expr_wfl (stmt, input_filename, 0, 0);
+      EXPR_WFL_LINECOL (stmt) = location;
+    }
+  JAVA_MAYBE_GENERATE_DEBUG_INFO (stmt);
+  return stmt;
+}
+
+static tree
+build_expr_block (body, decls)
+     tree body, decls;
+{
+  tree node = make_node (BLOCK);
+  BLOCK_EXPR_DECLS (node) = decls;
+  BLOCK_EXPR_BODY (body);
+  if (body)
+    TREE_TYPE (node) = TREE_TYPE (body);
+  TREE_SIDE_EFFECTS (node) = 1;
+  return node;
+}
+
+/* Create a new function block and link its supercontext to the
+   previous block. The current function DECL is used as supercontext
+   when enter_block is called for the first time for a given
+   function. The current function body (DECL_FUNCTION_BODY) is set to
+   the newly created block.  */
+
+static block_level = 0;
+
+static tree
+enter_block ()
+{
+  tree b = build_expr_block (NULL_TREE, NULL_TREE);
+  tree fndecl = current_function_decl; 
+
+  if (!DECL_FUNCTION_BODY (fndecl))
+    {
+      BLOCK_SUPERCONTEXT (b) = fndecl;
+      DECL_FUNCTION_BODY (fndecl) = b;
+    }
+  else
+    {
+      BLOCK_SUPERCONTEXT (b) = DECL_FUNCTION_BODY (fndecl);
+      DECL_FUNCTION_BODY (fndecl) = b;
+    }
+  return b;
+}
+
+/* Exit a block by changing the current function body
+   (DECL_FUNCTION_BODY) to the current block super context, only if
+   the block being exited isn't the method's top level one.  */
+
+static tree
+exit_block ()
+{
+  tree b = DECL_FUNCTION_BODY (current_function_decl);
+
+  if (BLOCK_SUPERCONTEXT (b) != current_function_decl)
+    DECL_FUNCTION_BODY (current_function_decl) = BLOCK_SUPERCONTEXT (b);
+
+  return b;
+}
+
+/* Lookup for NAME in the nested function's blocks, all the way up to
+   the current toplevel one. It complies with Java's local variable
+   scoping rules.  */
+
+static tree
+lookup_name_in_blocks (name)
+     tree name;
+{
+  tree b = DECL_FUNCTION_BODY (current_function_decl);
+
+  while (b != current_function_decl)
+    {
+      tree current;
+
+      /* Paranoid sanity check. To be removed */
+      if (TREE_CODE (b) != BLOCK)
+       fatal ("non block expr function body - lookup_name_in_blocks");
+
+      for (current = BLOCK_EXPR_DECLS (b); current; 
+          current = TREE_CHAIN (current))
+       if (DECL_NAME (current) == name)
+         return current;
+      b = BLOCK_SUPERCONTEXT (b);
+    }
+  return NULL_TREE;
+}
+
+static void
+maybe_absorb_scoping_blocks ()
+{
+  while (BLOCK_EXPR_ORIGIN (DECL_FUNCTION_BODY (current_function_decl)))
+    {
+      tree b = exit_block ();
+      java_method_add_stmt (current_function_decl, b);
+      SOURCE_FRONTEND_DEBUG (("Absorbing scoping block at line %d", lineno));
+    }
+}
+
+\f
+/* This section of the source is reserved to build_* functions that
+   are building incomplete tree nodes and the patch_* functions that
+   are completing them.  */
+
+/* Build an incomplete CALL_EXPR node. Encapsulate it within a WFL */
+
+static tree
+build_method_invocation (name, args)
+    tree name;
+    tree args;
+{
+  tree call = build (CALL_EXPR, NULL_TREE, name, args, NULL_TREE);
+  TREE_SIDE_EFFECTS (call) = 1;
+  /* Check on cases where NAME isn't a WFL. FIXME */
+  EXPR_WFL_LINECOL (call) = EXPR_WFL_LINECOL (name);
+  return call;
+}
+
+/* Build an incomplete assignment expression. */
+
+static tree
+build_assignment (op, op_location, lhs, rhs)
+     int op, op_location;
+     tree lhs, rhs;
+{
+  tree assignment;
+  /* Build the corresponding binop if we deal with a Compound
+     Assignment operator. Mark the binop sub-tree as part of a
+     Compound Assignment expression */
+  if (op != ASSIGN_TK)
+    {
+      rhs = build_binop (BINOP_LOOKUP (op), op_location, lhs, rhs);
+      COMPOUND_ASSIGN_P (rhs) = 1;
+    }
+  assignment = build (MODIFY_EXPR, NULL_TREE, lhs, rhs);
+  TREE_SIDE_EFFECTS (assignment) = 1;
+  EXPR_WFL_LINECOL (assignment) = op_location;
+  return assignment;
+}
+
+/* Print an INTEGER_CST node in a static buffer, and return the buffer. */
+
+static char *
+print_int_node (node)
+    tree node;
+{
+  static char buffer [80];
+  if (TREE_CONSTANT_OVERFLOW (node))
+    sprintf (buffer, "<overflow>");
+    
+  if (TREE_INT_CST_HIGH (node) == 0)
+    sprintf (buffer, HOST_WIDE_INT_PRINT_UNSIGNED,
+            TREE_INT_CST_LOW (node));
+  else if (TREE_INT_CST_HIGH (node) == -1
+          && TREE_INT_CST_LOW (node) != 0)
+    {
+      buffer [0] = '-';
+      sprintf (&buffer [1], HOST_WIDE_INT_PRINT_UNSIGNED,
+              -TREE_INT_CST_LOW (node));
+    }
+  else
+    sprintf (buffer, HOST_WIDE_INT_PRINT_DOUBLE_HEX,
+            TREE_INT_CST_HIGH (node), TREE_INT_CST_LOW (node));
+
+  return buffer;
+}
+
+/* 15.25 Assignment operators. */
+
+static tree
+patch_assignment (node, wfl_op1, wfl_op2)
+     tree node;
+     tree wfl_op1;
+     tree wfl_op2;
+{
+  tree rhs = TREE_OPERAND (node, 1);
+  tree lvalue = TREE_OPERAND (node, 0);
+  tree lhs_type, rhs_type, new_rhs = NULL_TREE;
+  int all_primitive;
+  int error_found = 0;
+  int lvalue_from_array = 0;
+
+  /* Can't assign to a final. */
+  if (DECL_P (lvalue) && FIELD_FINAL (lvalue))
+    {
+      parse_error_context 
+        (wfl_op1, "Can't assign a value to the final variable `%s'",
+        IDENTIFIER_POINTER (EXPR_WFL_NODE (wfl_op1)));
+      error_found = 1;
+    }
+
+  EXPR_WFL_LINECOL (wfl_operator) = EXPR_WFL_LINECOL (node);
+
+  /* Lhs can be a named variable */
+  if (DECL_P (lvalue))
+    {
+      INITIALIZED_P (lvalue) = 1;
+      lhs_type = TREE_TYPE (lvalue);
+    }
+  /* Or Lhs can be a array acccess. Should that be lvalue ? FIXME +
+     comment on reason why */
+  else if (TREE_CODE (wfl_op1) == ARRAY_REF)
+    {
+      lhs_type = TREE_TYPE (lvalue);
+      lvalue_from_array = 1;
+    }
+  /* Or a field access */
+  else if (TREE_CODE (lvalue) == COMPONENT_REF)
+    lhs_type = TREE_TYPE (lvalue);
+  /* Or a function return slot */
+  else if (TREE_CODE (lvalue) == RESULT_DECL)
+    lhs_type = TREE_TYPE (lvalue);
+  /* Otherwise, this is an error */
+  else
+    {
+      parse_error_context (wfl_op1, "Invalid left hand side of assignment");
+      error_found = 1;
+    }
+
+  rhs_type = TREE_TYPE (rhs);
+
+  /* 5.2 Begin Assignment conversion */
+
+  /* 5.1.1 Try Identity Conversion */
+  if (lhs_type == rhs_type) 
+    new_rhs = rhs;
+  
+  /* 5.1.2 Try Widening Primitive Conversion */
+  all_primitive = JPRIMITIVE_TYPE_P (lhs_type) && JPRIMITIVE_TYPE_P (rhs_type);
+  if (all_primitive && JINTEGRAL_TYPE_P (rhs_type)
+      && ((TYPE_PRECISION (rhs_type) < TYPE_PRECISION (lhs_type))
+         || (JFLOAT_TYPE_P (lhs_type) &&
+             TYPE_PRECISION (rhs_type) == TYPE_PRECISION (lhs_type))))
+    new_rhs = convert (lhs_type, rhs);
+  else if (all_primitive && JFLOAT_TYPE_P (rhs_type)
+          && (TYPE_PRECISION (rhs_type) < TYPE_PRECISION (lhs_type)))
+    new_rhs = convert (lhs_type, rhs);
+
+  /* Try a narrowing primitive conversion: 
+       - expression is a constant expression of type int AND
+       - variable is byte, short or char AND
+       - The value of the expression is representable in the type of the 
+         variable */
+  else if (rhs_type == int_type_node && TREE_CONSTANT (rhs)
+          && (lhs_type == byte_type_node || lhs_type == char_type_node
+              || lhs_type == short_type_node))
+    {
+      if (int_fits_type_p (rhs, lhs_type))
+        new_rhs = convert (lhs_type, rhs);
+      else
+       parse_warning_context 
+         (wfl_op1, "Constant expression `%s' to wide for narrowing "
+          "primitive conversion to `%s'", 
+          print_int_node (rhs), lang_printable_name (lhs_type));
+      /* Reported a warning that will turn into an error further
+        down, so we don't return */
+    }
+
+  /* 5.2 Try a reference conversion */
+  else if (!JPRIMITIVE_TYPE_P (rhs_type) && JREFERENCE_TYPE_P (lhs_type))
+    {
+      /* `null' may be assigned to any reference type */
+      if (rhs == null_pointer_node)
+        new_rhs = null_pointer_node;
+      /* Try the reference assignment conversion */
+      else if (valid_ref_assignconv_cast_p (rhs_type, lhs_type, 0))
+       new_rhs = rhs;
+      if (new_rhs)
+       lhs_type = promote_type (rhs_type);
+    }
+
+  /* 15.25.2 If we have a compound assignment, convert RHS into the
+     type of the LHS */
+  else if (COMPOUND_ASSIGN_P (TREE_OPERAND (node, 1)))
+    new_rhs = convert (lhs_type, rhs);
+
+  /* Explicit cast required. This is an error */
+  if (!new_rhs)
+    {
+      char *t1 = strdup ((char *)lang_printable_name (TREE_TYPE (rhs)));
+      char *t2 = strdup ((char *)lang_printable_name (lhs_type));
+      tree wfl;
+      char operation [32];     /* Max size known */
+
+      /* If the assignment is part of a declaration, we use the WFL of
+        the declared variable to point out the error and call it a
+        declaration problem. If the assignment is a genuine =
+        operator, we call is a operator `=' problem, otherwise we
+        call it an assignment problem. In both of these last cases,
+        we use the WFL of the operator to indicate the error. */
+
+      if (MODIFY_EXPR_FROM_INITIALIZATION_P (node))
+       {
+         wfl = wfl_op1;
+         strcpy (operation, "declaration");
+       }
+      else
+       {
+         wfl = wfl_operator;
+         if (COMPOUND_ASSIGN_P (TREE_OPERAND (node, 1)))
+           strcpy (operation, "assignment");
+         else if (TREE_CODE (TREE_OPERAND (node, 0)) == RESULT_DECL)
+           strcpy (operation, "`return'");
+         else
+           strcpy (operation, "`='");
+       }
+
+      parse_error_context 
+       (wfl, (!can_cast_to_p (rhs_type, lhs_type) ?
+              "Incompatible type for %s. Can't convert `%s' to `%s'" :
+              "Incompatible type for %s. Explicit cast "
+              "needed to convert `%s' to `%s'"), operation, t1, t2);
+      free (t1); free (t2);
+      error_found = 1;
+    }
+
+  /* Before reporting type incompatibility errors, check that the rhs
+     is initialized, if a variable */
+  if (not_initialized_as_it_should_p (rhs))
+    {
+      ERROR_VARIABLE_NOT_INITIALIZED (wfl_op2, DECL_NAME (rhs));
+      INITIALIZED_P (rhs) = 1;
+    }
+
+  if (error_found)
+    return error_mark_node;
+
+  /* If we built a compound expression as the result of a reference
+     assignment into an array element, return it here. */
+  if (TREE_CODE (node) == COMPOUND_EXPR)
+    return node;
+      
+  TREE_OPERAND (node, 0) = lvalue;
+  TREE_OPERAND (node, 1) = new_rhs;
+  TREE_TYPE (node) = lhs_type;
+  return node;
+}
+
+/* Check that SOURCE can be converted into DEST, at least with a
+   cast. If the convertion can't occur at all, return 0 otherwise
+   1. This function is used to produce accurate error messages on the
+   reasons why an assignment failed. */
+
+static int
+can_cast_to_p (source, dest)
+     tree source;
+     tree dest;
+{
+  if (TREE_CODE (source) == POINTER_TYPE)
+    source = TREE_TYPE (source);
+  if (TREE_CODE (dest) == POINTER_TYPE)
+    dest = TREE_TYPE (dest);
+
+  if (TREE_CODE (source) == RECORD_TYPE && TREE_CODE (dest) == RECORD_TYPE)
+    return valid_ref_assignconv_cast_p (source, dest, 1);
+
+  else if (JNUMERIC_TYPE_P (source) && JNUMERIC_TYPE_P (dest))
+    return 1;
+
+  return 0;
+}
+
+/* Check that something of SOURCE type can be assigned or cast to
+   something of DEST type at runtime. Return 1 if the operation is
+   valid, 0 otherwise. If CAST is set to 1, we're treating the case
+   were SOURCE is cast into DEST, which borrows a lot of the
+   assignment check. */
+
+static int
+valid_ref_assignconv_cast_p (source, dest, cast)
+     tree source;
+     tree dest;
+     int cast;
+{
+  if (TREE_CODE (source) == POINTER_TYPE)
+    source = TREE_TYPE (source);
+  if (TREE_CODE (dest) == POINTER_TYPE)
+    dest = TREE_TYPE (dest);
+  /* Case where SOURCE is a class type */
+  if (TYPE_CLASS_P (source))
+    {
+      if (TYPE_CLASS_P (dest))
+       return  source == dest || inherits_from_p (source, dest)
+         || cast && inherits_from_p (dest, source);
+      if (TYPE_INTERFACE_P (dest))
+       {
+         /* If doing a cast and SOURCE is final, the operation is
+             always correct a compile time (because even if SOURCE
+             does not implement DEST, a subclass of SOURCE might). */
+         if (cast && !CLASS_FINAL (TYPE_NAME (source)))
+           return 1;
+         /* Otherwise, SOURCE must implement DEST */
+         return interface_of_p (dest, source);
+       }
+      /* DEST is an array, cast permited if SOURCE is of Object type */
+      return (cast && source == object_type_node ? 1 : 0);
+    }
+  if (TYPE_INTERFACE_P (source))
+    {
+      if (TYPE_CLASS_P (dest))
+       {
+         /* If not casting, DEST must be the Object type */
+         if (!cast)
+           return dest == object_type_node;
+         /* We're doing a cast. The cast is always valid is class
+            DEST is not final, otherwise, DEST must implement SOURCE */
+         else if (!CLASS_FINAL (TYPE_NAME (source)))
+           return 1;
+         else
+           return interface_of_p (source, dest);
+       }
+      if (TYPE_INTERFACE_P (dest))
+       {
+         /* If doing a cast, then if SOURCE and DEST contain method
+             with the same signature but different return type, then
+             this is a (compile time) error */
+         if (cast)
+           {
+             tree method_source, method_dest;
+             tree source_type;
+             tree source_sig, dest_sig;
+             tree source_name;
+             for (method_source = TYPE_METHODS (source); method_source; 
+                  method_source = TREE_CHAIN (method_source))
+               {
+                 source_sig = 
+                   build_java_argument_signature (TREE_TYPE (method_source));
+                 source_type = TREE_TYPE (TREE_TYPE (method_source));
+                 source_name = DECL_NAME (method_source);
+                 for (method_dest = TYPE_METHODS (dest);
+                      method_dest; method_dest = TREE_CHAIN (method_dest))
+                   if (source_sig == 
+                       build_java_argument_signature (TREE_TYPE (method_dest))
+                       && source_name == DECL_NAME (method_dest)
+                       && source_type != TREE_TYPE (TREE_TYPE (method_dest)))
+                     return 0;
+               }
+             return 1;
+           }
+         else
+           return source == dest || interface_of_p (dest, source);
+       }
+      else                     /* Array */
+       return 0;
+    }
+  if (TYPE_ARRAY_P (source))
+    {
+      if (TYPE_CLASS_P (dest))
+       return dest == object_type_node;
+      if (TYPE_INTERFACE_P (dest))
+       return 0;               /* Install test on Clonable. FIXME */
+      else                     /* Arrays */
+       {
+         tree source_element_type = TYPE_ARRAY_ELEMENT (source);
+         tree dest_element_type = TYPE_ARRAY_ELEMENT (dest);
+         
+         if (source_element_type == dest_element_type)
+           return 1;
+         return valid_ref_assignconv_cast_p (source_element_type,
+                                             dest_element_type, cast);
+       }
+      return 0;
+    }
+  return 0;
+}
+
+/* Build an incomplete binop expression. */
+
+static tree
+build_binop (op, op_location, op1, op2)
+     enum tree_code op;
+     int op_location;
+     tree op1, op2;
+{
+  tree wfl;
+
+  /* URSHIFT_EXPR is not part of what GCC understands, we can't directly build
+     a node with it */
+  tree binop = 
+    build ((op == URSHIFT_EXPR ? RSHIFT_EXPR : op), NULL_TREE, op1, op2);
+  if (op == URSHIFT_EXPR)
+    TREE_SET_CODE (binop, op);
+
+  TREE_SIDE_EFFECTS (binop) = 1;
+  /* Store the location of the operator, for better error report. The
+     string of the operator will be rebuild based on the OP value. */
+  EXPR_WFL_LINECOL (binop) = op_location;
+  return binop;
+}
+
+/* Build the string of the operator retained by NODE. If NODE is part
+   of a compound expression, add an '=' at the end of the string. This
+   function is called when an error needs to be reported on an
+   operator. The string is returned as a pointer to a static character
+   buffer. */
+
+static char *
+operator_string (node)
+     tree node;
+{
+#define BUILD_OPERATOR_STRING(S)                                       \
+  {                                                                    \
+    sprintf (buffer, "%s%s", S, (COMPOUND_ASSIGN_P (node) ? "=" : ""));        \
+    return buffer;                                                     \
+  }
+  
+  static char buffer [10];
+  switch (TREE_CODE (node))
+    {
+    case MULT_EXPR: BUILD_OPERATOR_STRING ("*");
+    case RDIV_EXPR: BUILD_OPERATOR_STRING ("/");
+    case TRUNC_MOD_EXPR: BUILD_OPERATOR_STRING ("%");
+    case PLUS_EXPR: BUILD_OPERATOR_STRING ("+");
+    case MINUS_EXPR: BUILD_OPERATOR_STRING ("-");
+    case LSHIFT_EXPR: BUILD_OPERATOR_STRING ("<<");
+    case RSHIFT_EXPR: BUILD_OPERATOR_STRING (">>");
+    case URSHIFT_EXPR: BUILD_OPERATOR_STRING (">>>");
+    case BIT_AND_EXPR: BUILD_OPERATOR_STRING ("&");
+    case BIT_XOR_EXPR: BUILD_OPERATOR_STRING ("^");
+    case BIT_IOR_EXPR: BUILD_OPERATOR_STRING ("|");
+    case TRUTH_ANDIF_EXPR: BUILD_OPERATOR_STRING ("&&");
+    case TRUTH_ORIF_EXPR: BUILD_OPERATOR_STRING ("||");
+    case EQ_EXPR: BUILD_OPERATOR_STRING ("==");
+    case NE_EXPR: BUILD_OPERATOR_STRING ("!=");
+    case GT_EXPR: BUILD_OPERATOR_STRING (">");
+    case GE_EXPR: BUILD_OPERATOR_STRING (">=");
+    case LT_EXPR: BUILD_OPERATOR_STRING ("<");
+    case LE_EXPR: BUILD_OPERATOR_STRING ("<=");
+    case JAVA_UNARY_PLUS_EXPR: BUILD_OPERATOR_STRING ("+");
+    case NEGATE_EXPR: BUILD_OPERATOR_STRING ("-");
+    case TRUTH_NOT_EXPR: BUILD_OPERATOR_STRING ("!");
+    case BIT_NOT_EXPR: BUILD_OPERATOR_STRING ("~");
+    case PREINCREMENT_EXPR:    /* Fall through */
+    case POSTINCREMENT_EXPR: BUILD_OPERATOR_STRING ("++");
+    case PREDECREMENT_EXPR:    /* Fall through */
+    case POSTDECREMENT_EXPR: BUILD_OPERATOR_STRING ("--");
+    default:
+      fatal ("unregistered operator %s - operator_string",
+            tree_code_name [TREE_CODE (node)]);
+    }
+  return NULL;
+#undef BUILD_OPERATOR_STRING
+}
+
+/* Binary operators (15.16 up to 15.18). We return error_mark_node on
+   errors but we modify NODE so that it contains the type computed
+   according to the expression, when it's fixed. Otherwise, we write
+   error_mark_node as the type. It allows us to further the analysis
+   of remaining nodes and detects more errors in certain cases.  */
+
+static tree
+patch_binop (node, wfl_op1, wfl_op2)
+     tree node;
+     tree wfl_op1;
+     tree wfl_op2;
+{
+  tree op1 = TREE_OPERAND (node, 0);
+  tree op2 = TREE_OPERAND (node, 1);
+  tree op1_type = TREE_TYPE (op1);
+  tree op2_type = TREE_TYPE (op2);
+  tree prom_type;
+  int code = TREE_CODE (node);
+  /* If 1, tell the routine that we have to return error_mark_node
+     after checking for the initialization of the RHS */
+  int error_found = 0;
+
+  /* Figure what is going to be checked first for initialization prior
+     its use. If NODE is part of a compound assignment, we check the
+     second operand first, otherwise the first one first. We also
+     initialize the matching WFL for the error report. `cfi' stands
+     for Check For Initialization */
+  tree cfi = (COMPOUND_ASSIGN_P (node) ? op2 : op1);
+  tree cfi_wfl = (COMPOUND_ASSIGN_P (node) ? wfl_op2 : wfl_op1);
+
+  EXPR_WFL_LINECOL (wfl_operator) = EXPR_WFL_LINECOL (node);
+
+  /* Check initialization of LHS first. We then silence further error
+     message if the variable wasn't initialized */
+  if (not_initialized_as_it_should_p (cfi))
+    {
+      ERROR_VARIABLE_NOT_INITIALIZED (cfi_wfl, DECL_NAME (cfi));
+      INITIALIZED_P (op1) = 1;
+    }
+
+  switch (code)
+    {
+    /* 15.16 Multiplicative operators */
+    case MULT_EXPR:            /* 15.16.1 Multiplication Operator * */
+    case RDIV_EXPR:            /* 15.16.2 Division Operator / */
+    case TRUNC_MOD_EXPR:       /* 15.16.3 Remainder operator % */
+      if (!JPRIMITIVE_TYPE_P (op1_type) || !JPRIMITIVE_TYPE_P (op2_type))
+       {
+         if (!JPRIMITIVE_TYPE_P (op1_type))
+           ERROR_CANT_CONVERT_TO_NUMERIC (wfl_operator, node, op1_type);
+         if (!JPRIMITIVE_TYPE_P (op2_type) && (op1_type != op2_type))
+           ERROR_CANT_CONVERT_TO_NUMERIC (wfl_operator, node, op2_type);
+         TREE_TYPE (node) = error_mark_node;
+         error_found = 1;
+         break;
+       }
+      prom_type = binary_numeric_promotion (op1_type, op2_type, &op1, &op2);
+      /* Change the division operator if necessary */
+      if (code == RDIV_EXPR && TREE_CODE (prom_type) == INTEGER_TYPE)
+       TREE_SET_CODE (node, TRUNC_DIV_EXPR);
+      /* This one is more complicated. FLOATs are processed by a function
+        call to soft_fmod. */
+      if (code == TRUNC_MOD_EXPR)
+       return build_java_binop (TRUNC_MOD_EXPR, prom_type, op1, op2);
+      break;
+
+    /* 15.17 Additive Operators */
+    case PLUS_EXPR:            /* 15.17.1 String Concatenation Operator + */
+      if (JSTRING_TYPE_P (op1_type) || JSTRING_TYPE_P (op2_type))
+       fatal ("operator `+' non implemented on String - patch_binop");
+    case MINUS_EXPR:           /* 15.17.2 Additive Operators (+ and -) for
+                                  Numeric Types */
+      if (!JPRIMITIVE_TYPE_P (op1_type) || !JPRIMITIVE_TYPE_P (op2_type))
+       {
+         if (!JPRIMITIVE_TYPE_P (op1_type))
+           ERROR_CANT_CONVERT_TO_NUMERIC (wfl_operator, node, op1_type);
+         if (!JPRIMITIVE_TYPE_P (op2_type) && (op1_type != op2_type))
+           ERROR_CANT_CONVERT_TO_NUMERIC (wfl_operator, node, op2_type);
+         TREE_TYPE (node) = error_mark_node;
+         error_found = 1;
+         break;
+       }
+      prom_type = binary_numeric_promotion (op1_type, op2_type, &op1, &op2);
+      break;
+
+    /* 15.18 Shift Operators */
+    case LSHIFT_EXPR:
+    case RSHIFT_EXPR:
+    case URSHIFT_EXPR:
+      if (!JINTEGRAL_TYPE_P (op1_type) || !JINTEGRAL_TYPE_P (op2_type))
+       {
+         if (!JINTEGRAL_TYPE_P (op1_type))
+           ERROR_CAST_NEEDED_TO_INTEGRAL (wfl_operator, node, op1_type);
+         else
+           parse_error_context 
+             (wfl_operator, (JPRIMITIVE_TYPE_P (op2_type) ? 
+              "Incompatible type for `%s'. Explicit cast needed to convert "
+              "shift distance from `%s' to integral" : 
+              "Incompatible type for `%s'. Can't convert shift distance from "
+              "`%s' to integral"), 
+              operator_string (node), lang_printable_name (op2_type));
+         TREE_TYPE (node) = error_mark_node;
+         error_found = 1;
+         break;
+       }
+
+      /* Unary numeric promotion (5.6.1) is performed on each operand
+         separatly */
+      op1 = convert (promote_type (op1_type), op1);
+      op2 = convert (promote_type (op2_type), op2);
+
+      /* The type of the shift expression is the type of the promoted
+         type of the left-hand operand */
+      prom_type = TREE_TYPE (op1);
+
+      /* Shift int only up to 0x1f and long up to 0x3f */
+      if (prom_type == int_type_node)
+       op2 = fold (build (BIT_AND_EXPR, int_type_node, op2, 
+                          build_int_2 (0x1f, 0)));
+      else
+       op2 = fold (build (BIT_AND_EXPR, int_type_node, op2, 
+                          build_int_2 (0x3f, 0)));
+
+      /* The >>> operator is a >> operating on unsigned quantities */
+      if (code == URSHIFT_EXPR)
+       {
+         op1 = convert (unsigned_type (prom_type), op1);
+         TREE_SET_CODE (node, RSHIFT_EXPR);
+       }
+      break;
+      
+
+      /* 15.21 Bitwise and Logical Operators */
+    case BIT_AND_EXPR:
+    case BIT_XOR_EXPR:
+    case BIT_IOR_EXPR:
+      if (JINTEGRAL_TYPE_P (op1_type) && JINTEGRAL_TYPE_P (op2_type))
+       /* Binary numeric promotion is performed on both operand and the
+          expression retain that type */
+       prom_type = binary_numeric_promotion (op1_type, op2_type, &op1, &op2);
+
+      else if (TREE_CODE (op1_type) == BOOLEAN_TYPE 
+              && TREE_CODE (op1_type) == BOOLEAN_TYPE)
+       /* The type of the bitwise operator expression is BOOLEAN */
+       prom_type = boolean_type_node;
+      else
+       {
+         if (!JINTEGRAL_TYPE_P (op1_type))
+           ERROR_CAST_NEEDED_TO_INTEGRAL (wfl_operator, node, op1_type);
+         if (!JINTEGRAL_TYPE_P (op2_type) && (op1_type != op2_type))
+           ERROR_CAST_NEEDED_TO_INTEGRAL (wfl_operator, node, op2_type);
+         TREE_TYPE (node) = error_mark_node;
+         error_found = 1;
+         /* Insert a break here if adding thing before the switch's
+             break for this case */
+       }
+      break;
+
+      /* 15.22 Conditional-And Operator */
+    case TRUTH_ANDIF_EXPR:
+      /* 15.23 Conditional-Or Operator */
+    case TRUTH_ORIF_EXPR:
+      /* Operands must be of BOOLEAN type */
+      if (TREE_CODE (op1_type) != BOOLEAN_TYPE || 
+         TREE_CODE (op2_type) != BOOLEAN_TYPE)
+       {
+         if (TREE_CODE (op1_type) != BOOLEAN_TYPE)
+           ERROR_CANT_CONVERT_TO_BOOLEAN (wfl_operator, node, op1_type);
+         if (TREE_CODE (op2_type) != BOOLEAN_TYPE && (op1_type != op2_type))
+           ERROR_CANT_CONVERT_TO_BOOLEAN (wfl_operator, node, op2_type);
+         TREE_TYPE (node) = boolean_type_node;
+         error_found = 1;
+         break;
+       }
+      /* The type of the conditional operators is BOOLEAN */
+      prom_type = boolean_type_node;
+      break;
+
+      /* 15.19.1 Numerical Comparison Operators <, <=, >, >= */
+    case LT_EXPR:
+    case GT_EXPR:
+    case LE_EXPR:
+    case GE_EXPR:
+      /* The type of each of the operands must be a primitive numeric
+         type */
+      if (!JNUMERIC_TYPE_P (op1_type) || ! JNUMERIC_TYPE_P (op2_type))
+       {
+         if (!JNUMERIC_TYPE_P (op1_type))
+           ERROR_CANT_CONVERT_TO_NUMERIC (wfl_operator, node, op1_type);
+         if (!JNUMERIC_TYPE_P (op2_type) && (op1_type != op2_type))
+           ERROR_CANT_CONVERT_TO_NUMERIC (wfl_operator, node, op2_type);
+         TREE_TYPE (node) = boolean_type_node;
+         error_found = 1;
+         break;
+       }
+      /* Binary numeric promotion is performed on the operands */
+      binary_numeric_promotion (op1_type, op2_type, &op1, &op2);
+      /* The type of the relation expression is always BOOLEAN */
+      prom_type = boolean_type_node;
+      break;
+
+      /* 15.20 Equality Operator */
+    case EQ_EXPR:
+    case NE_EXPR:
+      /* 15.20.1 Numerical Equality Operators == and != */
+      /* Binary numeric promotion is performed on the operands */
+      if (JPRIMITIVE_TYPE_P (op1_type) && JPRIMITIVE_TYPE_P (op2_type))
+       binary_numeric_promotion (op1_type, op2_type, &op1, &op2);
+      
+      /* 15.20.2 Boolean Equality Operators == and != */
+      else if (TREE_CODE (op1_type) == BOOLEAN_TYPE &&
+         TREE_CODE (op2_type) == BOOLEAN_TYPE)
+       ;                       /* Nothing to do here */
+      
+      /* 15.20.3 Reference Equality Operators == and != */
+      /* Types have to be either references or the null type */
+      else if ((op1 == null_pointer_node || op2 == null_pointer_node 
+               || JREFERENCE_TYPE_P (op1_type) 
+               || JREFERENCE_TYPE_P (op2_type))
+              && ((op1_type == op2_type)
+                  /* The should use a can_cast_to_p() */
+                  ))
+       ;                       /* Nothing to do here */
+         
+      /* Else we have an error figure what can't be converted into
+        what and report the error */
+      else
+       {
+         char *t1;
+         t1 = strdup ((char *)lang_printable_name (op1_type));
+         parse_error_context 
+           (wfl_operator, "Incompatible type for `%s'. Can't convert `%s' "
+            "to `%s'", operator_string (node), t1, 
+            lang_printable_name (op2_type));
+         free (t1);
+         TREE_TYPE (node) = boolean_type_node;
+         error_found = 1;
+         break;
+       }
+      prom_type = boolean_type_node;
+      break;
+    }
+
+  /* Then check the initialization of the RHS. We don't do that if
+     we're dealing with a node that is part of a compound
+     assignment. We then silence further error message if the variable
+     wasn't initialized */
+  if (not_initialized_as_it_should_p (op2) && !COMPOUND_ASSIGN_P (node))
+    {
+      ERROR_VARIABLE_NOT_INITIALIZED (wfl_op2, DECL_NAME (op2));
+      INITIALIZED_P (op2) = 1;
+    }
+
+  if (error_found)
+    return error_mark_node;
+
+  TREE_OPERAND (node, 0) = op1;
+  TREE_OPERAND (node, 1) = op2;
+  TREE_TYPE (node) = prom_type;
+  return fold (node);
+}
+
+/* Build an incomplete unary operator expression. Unary `+' node is
+   build as a CONV_EXPR, even though its tree code is overridden by a
+   JAVA_UNARY_PLUS_EXPR that isn't a tree code, to differentiate it during
+   the walk. */
+
+static tree
+build_unaryop (op_token, op_location, op1)
+     int op_token, op_location;
+     tree op1;
+{
+  enum tree_code op;
+  tree unaryop;
+  switch (op_token)
+    {
+    case PLUS_TK: op = CONVERT_EXPR; break;
+    case MINUS_TK: op = NEGATE_EXPR; break;
+    case NEG_TK: op = TRUTH_NOT_EXPR; break;
+    case NOT_TK: op = BIT_NOT_EXPR; break;
+    default: fatal ("Unknown token `%d' for unary operator - build_unaryop",
+                   op_token);
+    }
+
+  unaryop = build1 (op, NULL_TREE, op1);
+  if (op_token == PLUS_TK)
+    TREE_SET_CODE (unaryop, JAVA_UNARY_PLUS_EXPR);
+
+  TREE_SIDE_EFFECTS (unaryop) = 1;
+  /* Store the location of the operator, for better error report. The
+     string of the operator will be rebuild based on the OP value. */
+  EXPR_WFL_LINECOL (unaryop) = op_location;
+  return unaryop;
+}
+
+/* Special case for the ++/-- operators, since they require an extra
+   argument to build, which is set to NULL and patched
+   later. IS_POST_P is 1 if the operator, 0 otherwise.  */
+
+static tree
+build_incdec (op_token, op_location, op1, is_post_p)
+     int op_token, op_location;
+     tree op1;
+     int is_post_p;
+{
+  static enum tree_code lookup [2][2] = 
+    {
+      { PREDECREMENT_EXPR, PREINCREMENT_EXPR, },
+      { POSTDECREMENT_EXPR, POSTINCREMENT_EXPR, },
+    };
+  tree node = build (lookup [is_post_p][(op_token - DECR_TK)], 
+                    NULL_TREE, op1, NULL_TREE);
+  TREE_SIDE_EFFECTS (node) = 1;
+  /* Store the location of the operator, for better error report. The
+     string of the operator will be rebuild based on the OP value. */
+  EXPR_WFL_LINECOL (node) = op_location;
+  return node;
+}     
+
+/* Build an incomplete cast operator, based on the use of the
+   CONVERT_EXPR. Note that TREE_TYPE of the constructed node is
+   set. java_complete_tree is trained to walk a CONVERT_EXPR even
+   though its type is already set.  */
+
+static tree
+build_cast (location, type, exp)
+     int location;
+     tree type, exp;
+{
+  tree node = build1 (CONVERT_EXPR, type, exp);
+  EXPR_WFL_LINECOL (node) = location;
+  return node;
+}
+
+/* 15.14 Unary operators. We return error_mark_node in case of error,
+   but preserve the type of NODE if the type is fixed.  */
+
+static tree
+patch_unaryop (node, wfl_op)
+     tree node;
+     tree wfl_op;
+{
+  tree op = TREE_OPERAND (node, 0);
+  tree op_type = TREE_TYPE (op);
+  tree prom_type, value;
+  int code = TREE_CODE (node);
+  int error_found = 0;
+
+  EXPR_WFL_LINECOL (wfl_operator) = EXPR_WFL_LINECOL (node);
+
+  switch (code)
+    {
+      /* 15.13.2 Postfix Increment Operator ++ */
+    case POSTINCREMENT_EXPR:
+      /* 15.13.3 Postfix Increment Operator -- */
+    case POSTDECREMENT_EXPR:
+      /* 15.14.1 Prefix Increment Operator ++ */
+    case PREINCREMENT_EXPR:
+      /* 15.14.2 Prefix Decrement Operator -- */
+    case PREDECREMENT_EXPR:
+      if (!DECL_P (op))
+       {
+         parse_error_context (wfl_operator, "Invalid argument to `%s'",
+                              operator_string (node));
+         TREE_TYPE (node) = error_mark_node;
+         error_found = 1;
+       }
+      else if (FIELD_FINAL (op))
+       {
+         parse_error_context 
+           (wfl_op, "Can't assign a value to the final variable `%s'",
+            IDENTIFIER_POINTER (EXPR_WFL_NODE (wfl_op)));
+         TREE_TYPE (node) = error_mark_node;
+         error_found = 1;
+       }
+      /* From now on, we know that op if a variable and that it has a
+         valid wfl. We use wfl_op to locate errors related to the
+         ++/-- operand. */
+      else if (!JNUMERIC_TYPE_P (op_type))
+       {
+         parse_error_context
+           (wfl_op, "Invalid argument type `%s' to `%s'",
+            lang_printable_name (op_type), operator_string (node));
+         TREE_TYPE (node) = error_mark_node;
+         error_found = 1;
+       }
+      else
+       {
+         /* Before the addition, binary numeric promotion if performed on
+            both operands */
+         value = integer_one_node;
+         prom_type = binary_numeric_promotion (op_type, TREE_TYPE (value), 
+                                               &op, &value);
+         /* And write the promoted increment back */
+         TREE_OPERAND (node, 1) = value;
+       }
+      break;
+
+      /* 15.14.3 Unary Plus Operator + */
+    case JAVA_UNARY_PLUS_EXPR:
+      /* 15.14.4 Unary Minus Operator - */
+    case NEGATE_EXPR:
+      if (!JNUMERIC_TYPE_P (op_type))
+       {
+         ERROR_CANT_CONVERT_TO_NUMERIC (wfl_operator, node, op_type);
+         TREE_TYPE (node) = error_mark_node;
+         error_found = 1;
+       }
+      /* Unary numeric promotion is performed on operand */
+      else
+       {
+         prom_type = promote_type (op_type);
+         op = convert (prom_type, op);
+         if (code == JAVA_UNARY_PLUS_EXPR)
+           node = op;
+       }
+      break;
+
+      /* 15.14.5 Bitwise Complement Operator ~ */
+    case BIT_NOT_EXPR:
+      if (!JINTEGRAL_TYPE_P (op_type))
+       {
+         ERROR_CAST_NEEDED_TO_INTEGRAL (wfl_operator, node, op_type);
+         TREE_TYPE (node) = error_mark_node;
+         error_found = 1;
+       }
+      else
+       {
+         prom_type = promote_type (op_type);
+         op = convert (prom_type, op);
+       }
+      break;
+
+      /* 15.14.6 Logical Complement Operator ! */
+    case TRUTH_NOT_EXPR:
+      if (TREE_CODE (op_type) != BOOLEAN_TYPE)
+       {
+         ERROR_CANT_CONVERT_TO_BOOLEAN (wfl_operator, node, op_type);
+         TREE_TYPE (node) = boolean_type_node;
+         error_found = 1;
+       }
+      else
+       prom_type = boolean_type_node;
+      break;
+
+      /* 15.15 Cast Expression */
+    case CONVERT_EXPR:
+      value = patch_cast (node, wfl_op, wfl_operator);
+      if (value == error_mark_node)
+       {
+         TREE_TYPE (node) = error_mark_node;
+         error_found = 1;
+       }
+      else
+       node = value;
+      break;
+    }
+  
+  /* Check variable initialization */
+  if (not_initialized_as_it_should_p (op))
+    {
+      ERROR_VARIABLE_NOT_INITIALIZED (wfl_op, DECL_NAME (op));
+      INITIALIZED_P (op) = 1;
+    }
+  
+  if (error_found)
+    return error_mark_node;
+  /* In the case of JAVA_UNARY_PLUS_EXPR, we replaced NODE by a new one */
+  else if (code != JAVA_UNARY_PLUS_EXPR && code != CONVERT_EXPR)
+    {
+      TREE_OPERAND (node, 0) = op;
+      TREE_TYPE (node) = prom_type;
+    }
+  return fold (node);
+}
+
+/* Generic type resolution that sometimes takes place during node
+   patching. Returned the resolved type or generate an error
+   message. Return the resolved type or NULL_TREE.  */
+
+static tree
+resolve_type_during_patch (type)
+     tree type;
+{
+  if (unresolved_type_p (type, NULL))
+    {
+      tree type_decl = resolve_no_layout (EXPR_WFL_NODE (type), NULL_TREE);
+      if (!type_decl)
+       {
+         parse_error_context (type, 
+                              "Class `%s' not found in type declaration",
+                              IDENTIFIER_POINTER (EXPR_WFL_NODE (type)));
+         return NULL_TREE;
+       }
+      else
+       return TREE_TYPE (type_decl);
+    }
+  return type;
+}
+/* 5.5 Casting Conversion. error_mark_node is returned if an error is
+   found. Otherwise NODE or something meant to replace it is returned.  */
+
+static tree
+patch_cast (node, wfl_op, wfl_operator)
+     tree node;
+     tree wfl_op;
+     tree wfl_operator;
+{
+  tree op = TREE_OPERAND (node, 0);
+  tree op_type = TREE_TYPE (op);
+  tree cast_type = TREE_TYPE (node);
+  char *t1;
+
+  /* First resolve OP_TYPE if unresolved */
+  if (!(cast_type = resolve_type_during_patch (cast_type)))
+    return error_mark_node;
+
+  /* Check on cast that are proven correct at compile time */
+  if (JNUMERIC_TYPE_P (cast_type) && JNUMERIC_TYPE_P (op_type))
+    {
+      static tree convert_narrow ();
+      /* Same type */
+      if (cast_type == op_type)
+       return node;
+
+      /* Try widening/narowwing convertion. Potentially, things need
+        to be worked out in gcc so we implement the extreme cases
+        correctly. fold_convert() needs to be fixed. */
+      return convert (cast_type, op);
+    }
+
+  /* The remaining legal casts involve conversion between reference
+     types. Check for their compile time correctness. */
+  if (JREFERENCE_TYPE_P (op_type) && JREFERENCE_TYPE_P (cast_type) 
+      && valid_ref_assignconv_cast_p (op_type, cast_type, 1))
+    {
+      TREE_TYPE (node) = promote_type (cast_type);
+      /* Now, the case can be determined correct at compile time if
+         OP_TYPE can be converted into CAST_TYPE by assignment
+         conversion (5.2) */
+
+      if (valid_ref_assignconv_cast_p (op_type, cast_type, 0))
+       return node;
+
+      /* The cast requires a run-time check */
+      return build (CALL_EXPR, promote_type (cast_type),
+                   build_address_of (soft_checkcast_node),
+                   tree_cons (NULL_TREE, build_class_ref (cast_type),
+                              build_tree_list (NULL_TREE, op)),
+                   NULL_TREE);
+    }
+
+  /* Any other casts are proven incorrect at compile time */
+  t1 = strdup ((char *)lang_printable_name (op_type));
+  parse_error_context (wfl_operator, "Invalid cast from `%s' to `%s'",
+                      t1, lang_printable_name (cast_type));
+  free (t1);
+  return error_mark_node;
+}
+
+/* Build an ARRAY_REF incomplete tree node. Note that operand 1 isn't
+   a list of indices. */
+static tree
+build_array_ref (location, array, index)
+     int location;
+     tree array, index;
+{
+  tree node = build (ARRAY_REF, NULL_TREE, array, index);
+  EXPR_WFL_LINECOL (node) = location;
+  return node;
+}
+
+/* 15.12 Array Access Expression */
+
+static tree
+patch_array_ref (node, wfl_array, wfl_index)
+     tree node, wfl_array, wfl_index;
+{
+  tree array = TREE_OPERAND (node, 0);
+  tree array_type  = TREE_TYPE (array);
+  tree index = TREE_OPERAND (node, 1);
+  tree index_type = TREE_TYPE (index);
+  tree promoted_index_type;
+  int error_found = 0;
+
+  EXPR_WFL_LINECOL (wfl_operator) = EXPR_WFL_LINECOL (node);
+
+  if (not_initialized_as_it_should_p (array))
+    {
+      ERROR_VARIABLE_NOT_INITIALIZED (wfl_array, DECL_NAME (array));
+      INITIALIZED_P (array) = 1;
+    }
+  if (! flag_emit_class_files)
+    array = save_expr (array);
+
+  if (TREE_CODE (array_type) == POINTER_TYPE)
+    array_type = TREE_TYPE (array_type);
+
+  /* The array reference must be an array */
+  if (!TYPE_ARRAY_P (array_type))
+    {
+      parse_error_context 
+       (wfl_operator, "`[]' can only be applied to arrays. It can't be "
+        "applied to `%s'", lang_printable_name (array_type));
+      TREE_TYPE (node) = error_mark_node;
+      error_found = 1;
+    }
+
+  /* The array index underdoes unary numeric promotion. The promoted
+     type must be int */
+  promoted_index_type = promote_type (index_type);
+  if (promoted_index_type != int_type_node)
+    {
+      int could_cast = can_cast_to_p (index_type, int_type_node);
+      parse_error_context 
+       (wfl_operator, 
+        (could_cast ? "Incompatible type for `[]'. Explicit cast needed to "
+         "convert `%s' to `int'" : "Incompatible type for `[]'. "
+         "Can't convert `%s' to `int'"),
+        lang_printable_name (index_type));
+      TREE_TYPE (node) = error_mark_node;
+      error_found = 1;
+    }
+
+  /* Now if the index is a var/parm decl, check on its initialization */
+  if (not_initialized_as_it_should_p (index))
+    {
+      ERROR_VARIABLE_NOT_INITIALIZED (wfl_index, DECL_NAME (index));
+      INITIALIZED_P (index) = 1;
+    }
+
+  if (error_found)
+    return error_mark_node;
+  index = convert (promoted_index_type, index);
+
+  if (TREE_CODE (array_type) == RECORD_TYPE)
+    array_type = promote_type (TYPE_ARRAY_ELEMENT (array_type));
+  if (flag_emit_class_files)
+    {
+      TREE_OPERAND (node, 0)= array;
+      TREE_OPERAND (node, 1)= index;
+    }
+  else
+    node = build_java_arrayaccess (array, array_type, index);
+  TREE_TYPE (node) = array_type;
+  return node;
+}
+
+/* 15.9 Array Creation Expressions */
+
+static tree
+build_newarray_node (type, dims, extra_dims)
+     tree type;
+     tree dims;
+     int extra_dims;
+{
+  tree node =
+    build (CALL_EXPR, NULL_TREE, type, nreverse (dims), 
+          build_int_2 (extra_dims, 0));
+  TREE_SET_CODE (node, JAVA_NEW_ARRAY_EXPR);
+  return node;
+}
+
+static tree
+patch_newarray (node)
+     tree node;
+{
+  tree type = TREE_OPERAND (node, 0);
+  tree dims = TREE_OPERAND (node, 1);
+  tree cdim, array_type;
+  int error_found = 0;
+  int ndims = 0;
+  int xdims = TREE_INT_CST_LOW (TREE_OPERAND (node, 2));
+  int total_dims;
+
+  /* Dimension types are verified. It's better for the types to be
+     verified in order. */
+  for (cdim = dims, ndims = 0; cdim; cdim = TREE_CHAIN (cdim), ndims++ )
+    {
+      int dim_error = 0;
+      tree dim = TREE_VALUE (cdim);
+
+      /* Dim might have been saved during its evaluation */
+      dim = (TREE_CODE (dim) == SAVE_EXPR ? dim = TREE_OPERAND (dim, 0) : dim);
+
+      /* The type of each specified dimension must be an integral type. */
+      if (!JINTEGRAL_TYPE_P (TREE_TYPE (dim)))
+       dim_error = 1;
+
+      /* Each expression undergoes an unary numeric promotion (5.6.1) and the
+        promoted type must be int. */
+      else
+       {
+         dim = convert (promote_type (TREE_TYPE (dim)), dim);
+         if (TREE_TYPE (dim) != int_type_node)
+           dim_error = 1;
+       }
+
+      /* Report errors on types here */
+      if (dim_error)
+       {
+         parse_error_context 
+           (TREE_PURPOSE (cdim), 
+            "Incompatible type for dimension in array creation expression. "
+            "%s convert `%s' to `int'", 
+            (can_cast_to_p (TREE_TYPE (dim), int_type_node) ?
+             "Explicit cast needed to" : "Can't"),
+            lang_printable_name (TREE_TYPE (dim)));
+         error_found = 1;
+       }
+
+      /* Check for uninitialized variables */
+      if (not_initialized_as_it_should_p (dim))
+       {
+         ERROR_VARIABLE_NOT_INITIALIZED (TREE_PURPOSE (cdim), 
+                                         DECL_NAME (dim));
+         INITIALIZED_P (dim) = 1;
+         error_found = 1;
+       }
+
+      TREE_PURPOSE (cdim) = NULL_TREE;
+    }
+
+  /* Resolve array base type if unresolved */
+  if (!(type = resolve_type_during_patch (type)))
+    error_found = 1;
+
+  if (error_found)
+    {
+      /* We don't want further evaluation of this bogus array creation
+         operation */
+      TREE_TYPE (node) = error_mark_node;
+      return error_mark_node;
+    }
+
+  /* The node is transformed into a function call. Things are done
+     differently according to the number of dimensions. If the number
+     of dimension is equal to 1, then the nature of the base type
+     (primitive or not) matters. */
+  total_dims = xdims + ndims;
+  if (total_dims == 1)
+    {
+      if (JPRIMITIVE_TYPE_P (type))
+       {
+         int type_code;
+         if (type == boolean_type_node)
+           type_code = 4;
+         else if (type == char_type_node)
+           type_code = 5;
+         else if (type == float_type_node)
+           type_code = 6;
+         else if (type == double_type_node)
+           type_code = 7;
+         else if (type == byte_type_node)
+           type_code = 8;
+         else if (type == short_type_node)
+           type_code = 9;
+         else if (type == int_type_node)
+           type_code = 10;
+         else if (type == long_type_node)
+           type_code = 11;
+         else
+           fatal ("Can't compute type code - patch_newarray");
+         return build_newarray (type_code, TREE_VALUE (dims));
+       }
+      else
+       return build_anewarray (type, TREE_VALUE (dims));
+    }
+  
+  /* Add extra dimensions as unknown dimensions */
+  while (xdims--)
+    dims = 
+      chainon (dims, build_tree_list (NULL_TREE, integer_negative_one_node));
+  dims = chainon (dims, build_tree_list (NULL_TREE, integer_zero_node));
+
+  /* Can't reuse what's already written in expr.c because it uses the
+     JVM stack representation. Provide a build_multianewarray. FIXME */
+  array_type = type;
+  for (cdim = TREE_CHAIN (dims); cdim; cdim = TREE_CHAIN (cdim))
+    array_type = build_java_array_type (promote_type (array_type), 
+                                       TREE_CODE (cdim) == INTEGER_CST ?
+                                       TREE_INT_CST_LOW (cdim) : -1);
+  return build (CALL_EXPR,
+               promote_type (array_type),
+               build_address_of (soft_multianewarray_node),
+               tree_cons (NULL_TREE, build_class_ref (array_type),
+                          tree_cons (NULL_TREE, 
+                                     build_int_2 (total_dims, 0), dims )),
+               NULL_TREE);
+}
+
+static tree
+build_this (location)
+     int location;
+{
+  tree node = build_wfl_node (this_identifier_node, input_filename, 0, 0);
+  TREE_SET_CODE (node, JAVA_THIS_EXPR);
+  EXPR_WFL_LINECOL (node) = location;
+  return node;
+}
+
+/* 14.15 The return statement. It builds a modify expression that
+   assigns the returned value to the RESULT_DECL that hold the value
+   to be returned. */
+
+static tree
+build_return (location, op)
+     int location;
+     tree op;
+{
+  tree node = build1 (RETURN_EXPR, NULL_TREE, op);
+  EXPR_WFL_LINECOL (node) = location;
+  return node;
+}
+
+static tree
+patch_return (node)
+     tree node;
+{
+  tree return_exp = TREE_OPERAND (node, 0);
+  tree meth = current_function_decl;
+  tree mtype = TREE_TYPE (TREE_TYPE (current_function_decl));
+  tree modify;
+  int error_found = 0;
+
+  TREE_TYPE (node) = error_mark_node;
+  EXPR_WFL_LINECOL (wfl_operator) = EXPR_WFL_LINECOL (node);
+
+  /* It's invalid to have a return value within a function that is
+     declared with the keyword void or that is a constructor */
+  if (return_exp && (mtype == void_type_node || DECL_CONSTRUCTOR_P (meth)))
+    error_found = 1;
+
+  /* It's invalid to have a no return value within a function that
+     isn't declared with the keyword `void' */
+  if (!return_exp && (mtype != void_type_node && !DECL_CONSTRUCTOR_P (meth)))
+    error_found = 2;
+
+  if (error_found)
+    {
+      char *t = strdup ((char *)lang_printable_name (mtype));
+      parse_error_context (wfl_operator, "`return' with%s value from `%s %s'",
+                          (error_found == 1 ? "" : "out"), t,
+                          lang_printable_name (meth));
+      free (t);
+      return error_mark_node;
+    }
+
+  /* If we have a return_exp, build a modify expression and expand it */
+  if (return_exp)
+    {
+      modify = build (MODIFY_EXPR, NULL_TREE, DECL_RESULT (meth), return_exp);
+      EXPR_WFL_LINECOL (modify) = EXPR_WFL_LINECOL (node);
+      modify = java_complete_tree (modify);
+      if (modify != error_mark_node)
+       {
+         TREE_SIDE_EFFECTS (modify) = 1;
+         TREE_OPERAND (node, 0) = modify;
+       }
+      else
+       return error_mark_node;
+    }
+  TREE_TYPE (node) = void_type_node;
+  TREE_SIDE_EFFECTS (node) = 1;
+  return node;
+}
+
+/* 14.8 The if Statement */
+
+static tree
+build_if_else_statement (location, expression, if_body, else_body)
+     int location;
+     tree expression, if_body, else_body;
+{
+  tree node;
+  /* FIXME: make else body be a void node, where this function is
+     called */
+  if (!else_body)
+    else_body = build (COMPOUND_EXPR, void_type_node, NULL_TREE, NULL_TREE);
+  node = build (COND_EXPR, NULL_TREE, expression, if_body, else_body);
+  EXPR_WFL_LINECOL (node) = location;
+  return node;
+}
+
+static tree
+patch_if_else_statement (node)
+     tree node;
+{
+  tree expression = TREE_OPERAND (node, 0);
+
+  TREE_TYPE (node) = error_mark_node;
+  EXPR_WFL_LINECOL (wfl_operator) = EXPR_WFL_LINECOL (node);
+
+  /* The type of expression must be boolean */
+  if (TREE_TYPE (expression) != boolean_type_node)
+    {
+      parse_error_context 
+       (wfl_operator, 
+        "Incompatible type for `if'. Can't convert `%s' to `boolean'", 
+        lang_printable_name (TREE_TYPE (expression)));
+      return error_mark_node;
+    }
+  
+  TREE_TYPE (node) = void_type_node;
+  TREE_SIDE_EFFECTS (node) = 1;
+  return node;
+}
+
+/* 14.6 Labeled Statements */
+
+/* Action taken when a lableled statement is parsed. a new
+   LABELED_BLOCK_EXPR is created. No statement is attached to the
+   label, yet.  */
+
+static tree
+build_labeled_block (location, label, wfl)
+     int location;
+     tree label, wfl;
+{
+  tree label_name = merge_qualified_name (label_id, label);
+  tree label_decl, node;
+
+  /* Issue a warning if we try to reuse a label that was previously
+     declared */
+  if (IDENTIFIER_LOCAL_VALUE (label_name))
+    {
+      EXPR_WFL_LINECOL (wfl_operator) = location;
+      parse_warning_context (wfl_operator, "Declaration of `%s' shadows "
+                            "a previous declaration",
+                            IDENTIFIER_POINTER (label));
+      EXPR_WFL_LINECOL (wfl_operator) = 
+        EXPR_WFL_LINECOL (IDENTIFIER_LOCAL_VALUE (label_name));
+      parse_warning_context (wfl_operator, "This is the location of the "
+                            "previous declaration of label `%s'",
+                            IDENTIFIER_POINTER (label));
+      java_warning_count--;
+    }
+
+  label_decl = create_label_decl (label_name);
+  node = build (LABELED_BLOCK_EXPR, NULL_TREE, label_decl, NULL_TREE);
+  EXPR_WFL_LINECOL (node) = location;
+  TREE_SIDE_EFFECTS (node) = 1;
+  return node;
+}
+
+/* Generate a label crafting a unique name for it. This is used to
+   implicitely label loops that aren't the body part of labeled
+   statement.  */
+
+static tree
+generate_labeled_block ()
+{
+  static int l_number = 0;
+  char buf [20];
+  tree label_name;
+  
+  sprintf (buf, "$a%d", l_number++);
+  return build_labeled_block (0, get_identifier (buf), NULL_TREE);
+}
+
+/* A labeled statement LBE is attached a statement. If the statement
+   happens to be a loop, a link from the loop back to the label is
+   installed.  */
+
+static tree
+complete_labeled_statement (lbe, statement)
+     tree lbe;                 /* Labeled block expr */
+     tree statement;
+{
+  /* In anyways, tie the loop to its statement */
+  LABELED_BLOCK_BODY (lbe) = statement;
+
+  /* Ok, if statement is a for loop, we have to attach the labeled
+     statement to the block the for loop belongs to and return the
+     block instead */
+  if (TREE_CODE (statement) == LOOP_EXPR && IS_FOR_LOOP_P (statement))
+    {
+      java_method_add_stmt (current_function_decl, lbe);
+      return exit_block ();
+    }
+
+  return lbe;
+}
+
+/* 14.10, 14.11, 14.12 Loop Statements */
+
+/* Create an empty LOOP_EXPR and make it the last in the nested loop
+   list. */
+
+static tree
+build_new_loop (loop_body)
+     tree loop_body;
+{
+  tree loop =  build (LOOP_EXPR, NULL_TREE, loop_body);
+  TREE_SIDE_EFFECTS (loop) = 1;
+  PUSH_LOOP (loop);
+  return loop;
+}
+
+/* Create a loop body according to the following structure:
+     COMPOUND_EXPR
+       COMPOUND_EXPR           (loop main body)
+         EXIT_EXPR             (this order is for while/for loops.
+         LABELED_BLOCK_EXPR      the order is reversed for do loops)
+           LABEL_DECL           (continue occurding here branche at the 
+           BODY                         end of this labeled block)
+       INCREMENT               (if any)
+
+  REVERSED, if non zero, tells that the loop condition expr comes
+  after the body, like in the do-while loop.  */
+
+static tree
+build_loop_body (location, condition, reversed)
+     int location;
+     tree condition;
+     int reversed;
+{
+  tree first, second, label, body;
+
+  condition = build (EXIT_EXPR, NULL_TREE, condition); /* Force walk */
+  EXPR_WFL_LINECOL (condition) = location; /* For accurate error report */
+  condition = build_debugable_stmt (location, condition);
+  TREE_SIDE_EFFECTS (condition) = 1;
+
+  body = generate_labeled_block ();
+  first = (reversed ? body : condition);
+  second = (reversed ? condition : body);
+  return 
+    build (COMPOUND_EXPR, NULL_TREE, 
+          build (COMPOUND_EXPR, NULL_TREE, first, second), size_zero_node);
+}
+
+/* Install CONDITION (if any) and loop BODY (using REVERSED to tell
+   their order) on the current loop. Unlink the current loop from the
+   loop list.  */
+
+static tree
+complete_loop_body (location, condition, body, reversed)
+     int location;
+     tree condition, body;
+     int reversed;
+{
+  tree to_return = ctxp->current_loop;
+  tree loop_body = LOOP_EXPR_BODY (to_return);
+  if (condition)
+    {
+      tree cnode = LOOP_EXPR_BODY_CONDITION_EXPR (loop_body, reversed);
+      /* We wrapped the EXIT_EXPR around a WFL so we can debug it.
+         The real EXIT_EXPR is one operand further. */
+      EXPR_WFL_LINECOL (cnode) = location;
+      /* This one is for accurate error reports */
+      EXPR_WFL_LINECOL (TREE_OPERAND (cnode, 0)) = location;
+      TREE_OPERAND (TREE_OPERAND (cnode, 0), 0) = condition;
+    }
+  LOOP_EXPR_BODY_BODY_EXPR (loop_body, reversed) = body;
+  POP_LOOP ();
+  return to_return;
+}
+
+/* Tailored version of complete_loop_body for FOR loops, when FOR
+   loops feature the condition part */
+
+static tree
+complete_for_loop (location, condition, update, body)
+    int location;
+    tree condition, update, body;
+{
+  /* Put the condition and the loop body in place */
+  tree loop = complete_loop_body (location, condition, body, 0);
+  /* LOOP is the current loop which has been now popped of the loop
+     stack. Install the update block */
+  LOOP_EXPR_BODY_UPDATE_BLOCK (LOOP_EXPR_BODY (loop)) = update;
+  return loop;
+}
+
+/* If the loop isn't surrounded by a labeled statement, create one and
+   insert LOOP as it's body.  */
+
+static tree
+patch_loop_statement (loop)
+     tree loop;
+{
+  tree cbl, loop_label, to_return_as_loop;
+
+  if (LOOP_HAS_LABEL_P (loop))
+    {
+      loop_label = ctxp->current_labeled_block;
+      to_return_as_loop = loop;
+    }
+  else
+    {
+      loop_label = generate_labeled_block ();
+      LABELED_BLOCK_BODY (loop_label) = loop;
+      PUSH_LABELED_BLOCK (loop_label);
+      to_return_as_loop = loop_label;
+    }
+  TREE_TYPE (to_return_as_loop) = void_type_node;
+  return to_return_as_loop;
+}
+
+/* 14.13, 14.14: break and continue Statements */
+
+/* Build a break or a continue statement. a null NAME indicates an
+   unlabeled break/continue statement.  */
+
+static tree
+build_bc_statement (location, is_break, name)
+     int location, is_break;
+     tree name;
+{
+  tree break_continue, label_block_expr = NULL_TREE;
+
+  if (name)
+    {
+      if (!(label_block_expr = IDENTIFIER_LOCAL_VALUE 
+           (merge_qualified_name (label_id, EXPR_WFL_NODE (name)))))
+       /* Null means that we don't have a target for this named
+          break/continue. In this case, we make the target to be the
+          label name, so that the error can be reported accuratly in
+          patch_bc_statement. */
+       label_block_expr = EXPR_WFL_NODE (name);
+    }
+  /* Unlabeled break/continue will be handled during the
+     break/continue patch operation */
+  break_continue 
+    = build (EXIT_BLOCK_EXPR, NULL_TREE, label_block_expr, NULL_TREE);
+
+  IS_BREAK_STMT_P (break_continue) = is_break;
+  TREE_SIDE_EFFECTS (break_continue) = 1;
+  EXPR_WFL_LINECOL (break_continue) = location;
+  return break_continue;
+}
+
+/* Verification of a break/continue statement. */
+
+static tree
+patch_bc_statement (node)
+     tree node;
+{
+  tree bc_label = EXIT_BLOCK_LABELED_BLOCK (node), target_stmt;
+  int is_unlabeled = 0;
+ EXPR_WFL_LINECOL (wfl_operator) = EXPR_WFL_LINECOL (node);
+  /* Not having a target means that the break/continue statement is
+     unlabeled. We try to find a descent label for it */
+  if (!bc_label)
+    {
+      is_unlabeled = 1;
+      /* There should be a loop to branch to */
+      if (ctxp->current_loop)
+       {
+         /* At that stage, we're in the loop body, which is
+            encapsulated around a LABELED_BLOCK_EXPR. So searching
+            the current loop label requires us to consider the
+            labeled block before the current one. */
+         if (!LOOP_HAS_LABEL_SKIP_P (ctxp->current_loop))
+           fatal ("unlabeled loop has not installed label -- "
+                  "patch_bc_statement");
+         bc_label = TREE_CHAIN (ctxp->current_labeled_block);
+       }
+      /* Not having a loop to break/continue to is an error */
+      else
+       {
+         parse_error_context (wfl_operator, "`%s' must be in loop%s",
+                              (IS_BREAK_STMT_P (node) ? "break" : "continue"),
+                              (IS_BREAK_STMT_P (node) ? " or switch" : ""));
+         return error_mark_node;
+       }
+    }
+  /* Having an identifier here means that the target is unknown. */
+  else if (TREE_CODE (bc_label) == IDENTIFIER_NODE)
+    {
+      parse_error_context (wfl_operator, "No label definition found for `%s'",
+                          IDENTIFIER_POINTER (bc_label));
+      return error_mark_node;
+    }
+
+  /* Find the statement we're targeting. */
+  target_stmt = LABELED_BLOCK_BODY (bc_label);
+
+  /* 14.13 The break Statement */
+  if (IS_BREAK_STMT_P (node))
+    {
+      /* Named break are always fine, as far as they have a target
+         (already verified). Anonymous break need to target
+         while/do/for/switch */
+      if (is_unlabeled &&
+         !(TREE_CODE (target_stmt) == LOOP_EXPR   /* do/while/for */
+           || 0))                                 /* switch FIXME */
+       {
+         parse_error_context (wfl_operator, 
+                              "`break' must be in loop or switch");
+         return error_mark_node;
+       }
+      /* If previously unlabeled, install the new found label */
+      if (is_unlabeled)
+       EXIT_BLOCK_LABELED_BLOCK (node) = bc_label;
+    }
+  /* 14.14 The continue Statement */
+  /* The continue statement must always target a loop */
+  else 
+    {
+      if (TREE_CODE (target_stmt) != LOOP_EXPR) /* do/while/for */
+       {
+         parse_error_context (wfl_operator, "`continue' must be in loop");
+         return error_mark_node;
+       }
+      /* Everything looks good. We can fix the `continue' jump to go
+         at the place in the loop were the continue is. The continue
+        is the current labeled block, by construction. */
+      EXIT_BLOCK_LABELED_BLOCK (node) = ctxp->current_labeled_block;
+    }
+
+  /* Our break/continue don't return values. */
+  TREE_TYPE (node) = void_type_node;
+  /* Encapsulate the break within a compound statement so that it's
+     expanded all the times by expand_expr (and not clobered
+     sometimes, like after a if statement) */
+  node = add_stmt_to_compound (NULL_TREE, void_type_node, node);
+  TREE_SIDE_EFFECTS (node) = 1;
+  return node;
+}
+
+/* Process the exit expression belonging to a loop. Its type must be
+   boolean.  */
+
+static tree
+patch_exit_expr (node)
+     tree node;
+{
+  tree expression = TREE_OPERAND (node, 0);
+  TREE_TYPE (node) = error_mark_node;
+  EXPR_WFL_LINECOL (wfl_operator) = EXPR_WFL_LINECOL (node);
+
+  /* The type of expression must be boolean */
+  if (TREE_TYPE (expression) != boolean_type_node)
+    {
+      parse_error_context 
+       (wfl_operator, 
+        "Incompatible type for loop conditional. Can't convert `%s' to "
+        "`boolean'", 
+        lang_printable_name (TREE_TYPE (expression)));
+      return error_mark_node;
+    }
+  /* Now we know things are allright, invert the condition, fold and
+     return */
+  TREE_OPERAND (node, 0) = 
+    fold (build1 (TRUTH_NOT_EXPR, boolean_type_node, expression));
+  TREE_TYPE (node) = void_type_node;
+  return node;
+}
diff --git a/gcc/java/typeck.c b/gcc/java/typeck.c
new file mode 100644 (file)
index 0000000..211e4b6
--- /dev/null
@@ -0,0 +1,785 @@
+/* Handle types for the GNU compiler for the Java(TM) language.
+   Copyright (C) 1996 Free Software Foundation, Inc.
+
+This file is part of GNU CC.
+
+GNU CC is free software; you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation; either version 2, or (at your option)
+any later version.
+
+GNU CC is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with GNU CC; see the file COPYING.  If not, write to
+the Free Software Foundation, 59 Temple Place - Suite 330,
+Boston, MA 02111-1307, USA.  
+
+Java and all Java-based marks are trademarks or registered trademarks
+of Sun Microsystems, Inc. in the United States and other countries.
+The Free Software Foundation is independent of Sun Microsystems, Inc.  */
+
+/* Written by Per Bothner <bothner@cygnus.com> */
+
+#include "config.h"
+#include "tree.h"
+#include "obstack.h"
+#include "java-tree.h"
+#include "jcf.h"
+#include "convert.h"
+
+tree * type_map;
+extern struct obstack permanent_obstack;
+
+/* Set the type of the local variable with index SLOT to TYPE. */
+
+void
+set_local_type (slot, type)
+     int slot;
+     tree type;
+{
+  int max_locals = DECL_MAX_LOCALS(current_function_decl);
+  int nslots = TYPE_IS_WIDE (type) ? 2 : 1;
+  if (slot < 0 || slot + nslots - 1 >= max_locals)
+    fatal ("invalid local variable index");
+  type_map[slot] = type;
+  while (--nslots > 0)
+    type_map[++slot] = void_type_node;
+}
+
+extern tree convert_to_integer (tree type, tree expr);
+extern tree convert_to_real (tree type, tree expr);
+extern tree convert_to_pointer (tree type, tree expr);
+
+/* Create an expression whose value is that of EXPR,
+   converted to type TYPE.  The TREE_TYPE of the value
+   is always TYPE.  This function implements all reasonable
+   conversions; callers should filter out those that are
+   not permitted by the language being compiled.  */
+
+tree
+convert (type, expr)
+     tree type, expr;
+{
+  register enum tree_code code = TREE_CODE (type);
+
+  if (type == TREE_TYPE (expr)
+      || TREE_CODE (expr) == ERROR_MARK)
+    return expr;
+  if (TREE_CODE (TREE_TYPE (expr)) == ERROR_MARK)
+    return error_mark_node;
+  if (code == BOOLEAN_TYPE)
+    return fold (convert_to_boolean (type, expr));
+  if (code == INTEGER_TYPE)
+    return fold (convert_to_integer (type, expr));
+  if (code == REAL_TYPE)
+    return fold (convert_to_real (type, expr));
+  if (code == CHAR_TYPE)
+    return fold (convert_to_char (type, expr));
+  if (code == POINTER_TYPE)
+    return fold (convert_to_pointer (type, expr));
+  error ("conversion to non-scalar type requested");
+  return error_mark_node;
+}
+
+tree
+convert_to_char (type, expr)
+    tree type, expr;
+{
+  return build1 (NOP_EXPR, type, expr);
+}
+
+tree
+convert_to_boolean (type, expr)
+     tree type, expr;
+{
+  return build1 (NOP_EXPR, type, expr);
+}
+
+/* Print an error message for invalid use of an incomplete type.
+   VALUE is the expression that was used (or 0 if that isn't known)
+   and TYPE is the type that was invalid.  */
+
+void
+incomplete_type_error (value, type)
+     tree value;
+     tree type;
+{
+  error ("internal error - use of undefined type");
+}
+
+/* Return a data type that has machine mode MODE.
+   If the mode is an integer,
+   then UNSIGNEDP selects between signed and unsigned types.  */
+
+tree
+type_for_mode (mode, unsignedp)
+     enum machine_mode mode;
+     int unsignedp;
+{
+  if (mode == TYPE_MODE (int_type_node))
+    return unsignedp ? unsigned_int_type_node : int_type_node;
+  if (mode == TYPE_MODE (long_type_node))
+    return unsignedp ? unsigned_long_type_node : long_type_node;
+  if (mode == TYPE_MODE (short_type_node))
+    return unsignedp ? unsigned_short_type_node : short_type_node;
+  if (mode == TYPE_MODE (byte_type_node))
+    return unsignedp ? unsigned_byte_type_node : byte_type_node;
+  if (mode == TYPE_MODE (float_type_node))
+    return float_type_node;
+  if (mode == TYPE_MODE (double_type_node))
+    return double_type_node;
+
+#if 0
+  if (mode == TYPE_MODE (build_pointer_type (char_type_node)))
+    return build_pointer_type (char_type_node);
+
+  if (mode == TYPE_MODE (build_pointer_type (integer_type_node)))
+    return build_pointer_type (integer_type_node);
+#endif
+
+  return 0;
+}
+
+/* Return an integer type with BITS bits of precision,
+   that is unsigned if UNSIGNEDP is nonzero, otherwise signed.  */
+
+tree
+type_for_size (bits, unsignedp)
+     unsigned bits;
+     int unsignedp;
+{
+  if (bits <= TYPE_PRECISION (byte_type_node))
+    return unsignedp ? unsigned_byte_type_node : byte_type_node;
+  if (bits <= TYPE_PRECISION (short_type_node))
+    return unsignedp ? unsigned_short_type_node : short_type_node;
+  if (bits <= TYPE_PRECISION (int_type_node))
+    return unsignedp ? unsigned_int_type_node : int_type_node;
+  if (bits <= TYPE_PRECISION (long_type_node))
+    return unsignedp ? unsigned_long_type_node : long_type_node;
+  return 0;
+}
+
+/* Return a type the same as TYPE except unsigned or
+   signed according to UNSIGNEDP.  */
+
+tree
+signed_or_unsigned_type (unsignedp, type)
+     int unsignedp;
+     tree type;
+{
+  if (! INTEGRAL_TYPE_P (type))
+    return type;
+  if (TYPE_PRECISION (type) == TYPE_PRECISION (int_type_node))
+    return unsignedp ? unsigned_int_type_node : int_type_node;
+  if (TYPE_PRECISION (type) == TYPE_PRECISION (byte_type_node))
+    return unsignedp ? unsigned_byte_type_node : byte_type_node;
+  if (TYPE_PRECISION (type) == TYPE_PRECISION (short_type_node))
+    return unsignedp ? unsigned_short_type_node : short_type_node;
+  if (TYPE_PRECISION (type) == TYPE_PRECISION (long_type_node))
+    return unsignedp ? unsigned_long_type_node : long_type_node;
+  return type;
+}
+
+/* Return a signed type the same as TYPE in other respects.  */
+
+tree
+signed_type (type)
+     tree type;
+{
+  return signed_or_unsigned_type (0, type);
+}
+
+/* Return an unsigned type the same as TYPE in other respects.  */
+
+tree
+unsigned_type (type)
+     tree type;
+{
+  return signed_or_unsigned_type (1, type);
+
+}
+
+/* Mark EXP saying that we need to be able to take the
+   address of it; it should not be allocated in a register.
+   Value is 1 if successful.  */
+
+int
+mark_addressable (exp)
+     tree exp;
+{
+  register tree x = exp;
+  while (1)
+    switch (TREE_CODE (x))
+      {
+      case ADDR_EXPR:
+      case COMPONENT_REF:
+      case ARRAY_REF:
+      case REALPART_EXPR:
+      case IMAGPART_EXPR:
+       x = TREE_OPERAND (x, 0);
+       break;
+
+      case TRUTH_ANDIF_EXPR:
+      case TRUTH_ORIF_EXPR:
+      case COMPOUND_EXPR:
+       x = TREE_OPERAND (x, 1);
+       break;
+
+      case COND_EXPR:
+       return mark_addressable (TREE_OPERAND (x, 1))
+         & mark_addressable (TREE_OPERAND (x, 2));
+
+      case CONSTRUCTOR:
+       TREE_ADDRESSABLE (x) = 1;
+       return 1;
+
+      case INDIRECT_REF:
+       /* We sometimes add a cast *(TYPE*)&FOO to handle type and mode
+          incompatibility problems.  Handle this case by marking FOO.  */
+       if (TREE_CODE (TREE_OPERAND (x, 0)) == NOP_EXPR
+           && TREE_CODE (TREE_OPERAND (TREE_OPERAND (x, 0), 0)) == ADDR_EXPR)
+         {
+           x = TREE_OPERAND (TREE_OPERAND (x, 0), 0);
+           break;
+         }
+       if (TREE_CODE (TREE_OPERAND (x, 0)) == ADDR_EXPR)
+         {
+           x = TREE_OPERAND (x, 0);
+           break;
+         }
+       return 1;
+
+      case VAR_DECL:
+      case CONST_DECL:
+      case PARM_DECL:
+      case RESULT_DECL:
+      case FUNCTION_DECL:
+       TREE_ADDRESSABLE (x) = 1;
+#if 0  /* poplevel deals with this now.  */
+       if (DECL_CONTEXT (x) == 0)
+         TREE_ADDRESSABLE (DECL_ASSEMBLER_NAME (x)) = 1;
+#endif
+       /* drops through */
+      default:
+       return 1;
+    }
+}
+
+/* Thorough checking of the arrayness of TYPE.  */
+
+int
+is_array_type_p (type)
+     tree type;
+{
+  return TREE_CODE (type) == POINTER_TYPE
+    && TREE_CODE (TREE_TYPE (type)) == RECORD_TYPE
+    && TYPE_ARRAY_P (TREE_TYPE (type));
+}
+
+/* Return the length of a Java array type.
+   Return -1 if the length is unknown or non-constant. */
+
+HOST_WIDE_INT
+java_array_type_length (array_type)
+     tree array_type;
+{
+  tree arfld;
+  if (TREE_CODE (array_type) == POINTER_TYPE)
+    array_type = TREE_TYPE (array_type);
+  arfld = TREE_CHAIN (TREE_CHAIN (TYPE_FIELDS (array_type)));
+  if (arfld != NULL_TREE)
+    {
+      tree index_type = TYPE_DOMAIN (TREE_TYPE (arfld));
+      tree high = TYPE_MAX_VALUE (index_type);
+      if (TREE_CODE (high) == INTEGER_CST)
+       return TREE_INT_CST_LOW (high) + 1;
+    }
+  return -1;
+}
+
+tree
+build_prim_array_type (element_type, length)
+     tree element_type;
+     HOST_WIDE_INT length;
+{
+  tree max_index = build_int_2 (length - 1, 0);
+  TREE_TYPE (max_index) = sizetype;
+  return build_array_type (element_type, build_index_type (max_index));
+}
+
+/* Return a Java array type with a given ELEMENT_TYPE and LENGTH.
+   These are hashed (shared) using IDENTIFIER_SIGNATURE_TYPE.
+   The LENGTH is -1 if the length is unknown. */
+
+tree
+build_java_array_type (element_type, length)
+     tree element_type;
+     HOST_WIDE_INT length;
+{
+  tree sig, t, fld;
+  char buf[12];
+  tree elsig = build_java_signature (element_type);
+  tree el_name = element_type;
+  sprintf (buf, length >= 0 ? "[%d" : "[", length);
+  sig = ident_subst (IDENTIFIER_POINTER (elsig), IDENTIFIER_LENGTH (elsig),
+                    buf, 0, 0, "");
+  t = IDENTIFIER_SIGNATURE_TYPE (sig);
+  if (t != NULL_TREE)
+    return t;
+  t = make_class ();
+  IDENTIFIER_SIGNATURE_TYPE (sig) = t;
+  TYPE_ARRAY_P (t) = 1;
+
+  if (TREE_CODE (el_name) == POINTER_TYPE)
+    el_name = TREE_TYPE (el_name);
+  el_name = TYPE_NAME (el_name);
+  if (TREE_CODE (el_name) == TYPE_DECL)
+    el_name = DECL_NAME (el_name);
+  TYPE_NAME (t) = identifier_subst (el_name, "", '.', '.', "[]");
+
+  set_java_signature (t, sig);
+  set_super_info (0, t, object_type_node, 0);
+  TYPE_ARRAY_ELEMENT (t) = element_type;
+
+  /* Add length pseudo-field. */
+  push_obstacks (&permanent_obstack, &permanent_obstack);
+  fld = build_decl (FIELD_DECL, get_identifier ("length"), int_type_node);
+  TYPE_FIELDS (t) = fld;
+  DECL_CONTEXT (fld) = t;
+  FIELD_PUBLIC (fld) = 1;
+  FIELD_FINAL (fld) = 1;
+
+  if (length >= 0)
+    {
+      tree atype = build_prim_array_type (element_type, length);
+      tree arfld = build_decl (FIELD_DECL, get_identifier ("data"), atype);
+      DECL_CONTEXT (arfld) = t;
+      TREE_CHAIN (fld) = arfld;
+    }
+  else
+    TYPE_ALIGN (t) = TYPE_ALIGN (element_type);
+  pop_obstacks ();
+
+  layout_class (t);
+  return t;
+}
+
+/* Promote TYPE to the type actually used for fields and parameters. */
+
+tree
+promote_type (type)
+     tree type;
+{
+  switch (TREE_CODE (type))
+    {
+    case RECORD_TYPE:
+      return build_pointer_type (CLASS_TO_HANDLE_TYPE (type));
+    case BOOLEAN_TYPE:
+      if (type == boolean_type_node)
+       return promoted_boolean_type_node;
+      goto handle_int;
+    case CHAR_TYPE:
+      if (type == char_type_node)
+       return promoted_char_type_node;
+      goto handle_int;
+    case INTEGER_TYPE:
+    handle_int:
+      if (TYPE_PRECISION (type) < TYPE_PRECISION (int_type_node))
+       {
+         if (type == short_type_node)
+           return promoted_short_type_node;
+         if (type == byte_type_node)
+           return promoted_byte_type_node;
+         return int_type_node;
+       }
+      /* ... else fall through ... */
+    default:
+      return type;
+    }
+}
+
+/* Parse a signature string, starting at *PTR and ending at LIMIT.
+   Return the seen TREE_TYPE, updating *PTR. */
+
+static tree
+parse_signature_type (ptr, limit)
+     const unsigned char **ptr, *limit;
+{
+  if ((*ptr) >= limit)
+    fatal ("bad signature string");
+  switch (*(*ptr))
+    {
+    case 'B':  (*ptr)++;  return byte_type_node;
+    case 'C':  (*ptr)++;  return char_type_node;
+    case 'D':  (*ptr)++;  return double_type_node;
+    case 'F':  (*ptr)++;  return float_type_node;
+    case 'S':  (*ptr)++;  return short_type_node;
+    case 'I':  (*ptr)++;  return int_type_node;
+    case 'J':  (*ptr)++;  return long_type_node;
+    case 'Z':  (*ptr)++;  return boolean_type_node;
+    case 'V':  (*ptr)++;  return void_type_node;
+    case '[':
+      for ((*ptr)++; (*ptr) < limit && isdigit (**ptr); ) (*ptr)++;
+      {
+       tree element_type = parse_signature_type (ptr, limit);
+       if (TREE_CODE (element_type) == RECORD_TYPE)
+         element_type = promote_type (element_type);
+       return build_java_array_type (element_type, -1);
+      }
+    case 'L':
+      {
+       const unsigned char *start = ++(*ptr);
+       register const unsigned char *str = start;
+       for ( ; ; str++)
+         {
+           if (str >= limit)
+             fatal ("bad signature string");
+           if (*str == ';')
+             break;
+         }
+       *ptr = str+1;
+       return lookup_class (unmangle_classname (start, str - start));
+      }
+    default:
+      fatal ("unrecognized signature string");
+    }
+}
+
+/* Parse a Java "mangled" signature string, starting at SIG_STRING,
+   and SIG_LENGTH bytes long.
+   Return a gcc type node. */
+
+tree
+parse_signature_string (sig_string, sig_length)
+     const unsigned char *sig_string;
+     int sig_length;
+{
+  tree result_type;
+  const unsigned char *str = sig_string;
+  const unsigned char *limit = str + sig_length;
+
+  push_obstacks (&permanent_obstack, &permanent_obstack);
+  if (str < limit && str[0] == '(')
+    {
+      tree argtype_list = NULL_TREE;
+      str++;
+      while (str < limit && str[0] != ')')
+       {
+         tree argtype = promote_type (parse_signature_type (&str, limit));
+         argtype_list = tree_cons (NULL_TREE, argtype, argtype_list);
+       }
+      if (str++, str >= limit)
+       fatal ("bad signature string");
+      result_type = promote_type (parse_signature_type (&str, limit));
+      result_type = build_function_type (result_type,
+                                        nreverse (argtype_list));
+    }
+  else
+    result_type = parse_signature_type (&str, limit);
+  if (str != limit)
+    error ("junk at end of signature string");
+  pop_obstacks ();
+  return result_type;
+}
+
+/* Convert a signature to its type.
+ * Uses IDENTIFIER_SIGNATURE_TYPE as a cache (except for primitive types).
+ */
+
+tree
+get_type_from_signature (tree signature)
+{
+  unsigned char *sig = (unsigned char *) IDENTIFIER_POINTER (signature);
+  int len = IDENTIFIER_LENGTH (signature);
+  tree type;
+  /* Primitive types aren't cached. */
+  if (len <= 1)
+    return parse_signature_string (sig, len);
+  type = IDENTIFIER_SIGNATURE_TYPE (signature);
+  if (type == NULL_TREE)
+    {
+      type = parse_signature_string (sig, len);
+      IDENTIFIER_SIGNATURE_TYPE (signature) = type;
+    }
+  return type;
+}
+
+/* Return the signature string for the arguments of method type TYPE. */
+
+tree
+build_java_argument_signature (type)
+     tree type;
+{
+  extern struct obstack temporary_obstack;
+  tree sig = TYPE_ARGUMENT_SIGNATURE (type);
+  if (sig == NULL_TREE)
+    {
+      tree args = TYPE_ARG_TYPES (type);
+      if (TREE_CODE (type) == METHOD_TYPE)
+       args = TREE_CHAIN (args);  /* Skip "this" argument. */
+      for (; args != NULL_TREE; args = TREE_CHAIN (args))
+       {
+         tree t = build_java_signature (TREE_VALUE (args));
+         obstack_grow (&temporary_obstack,
+                       IDENTIFIER_POINTER (t), IDENTIFIER_LENGTH (t));
+       }
+      obstack_1grow (&temporary_obstack, '\0');
+
+      sig = get_identifier (obstack_base (&temporary_obstack));
+      TYPE_ARGUMENT_SIGNATURE (type) = sig;
+      obstack_free (&temporary_obstack, obstack_base (&temporary_obstack));
+    }
+  return sig;
+}
+
+/* Return the signature of the given TYPE. */
+
+tree
+build_java_signature (type)
+     tree type;
+{
+  tree sig, t;
+  push_obstacks (&permanent_obstack, &permanent_obstack);
+  while (TREE_CODE (type) == POINTER_TYPE)
+    type = TREE_TYPE (type);
+  if (TYPE_LANG_SPECIFIC (type) == NULL)
+    {
+      TYPE_LANG_SPECIFIC (type) = (struct lang_type *)
+       perm_calloc (1, sizeof (struct lang_type));
+
+    }
+  sig = TYPE_LANG_SPECIFIC (type)->signature;
+  if (sig == NULL_TREE)
+    {
+      char sg[2];
+      switch (TREE_CODE (type))
+       {
+       case BOOLEAN_TYPE: sg[0] = 'Z';  goto native;
+       case CHAR_TYPE:    sg[0] = 'C';  goto native;
+       case VOID_TYPE:    sg[0] = 'V';  goto native;
+       case INTEGER_TYPE:
+         switch (TYPE_PRECISION (type))
+           {
+           case  8:       sg[0] = 'B';  goto native;
+           case 16:       sg[0] = 'S';  goto native;
+           case 32:       sg[0] = 'I';  goto native;
+           case 64:       sg[0] = 'J';  goto native;
+           default:  goto bad_type;
+           }
+       case REAL_TYPE:
+         switch (TYPE_PRECISION (type))
+           {
+           case 32:       sg[0] = 'F';  goto native;
+           case 64:       sg[0] = 'D';  goto native;
+           default:  goto bad_type;
+           }
+       native:
+         sg[1] = 0;
+         sig = get_identifier (sg);
+         break;
+       case RECORD_TYPE:
+         if (TYPE_ARRAY_P (type))
+           {
+             t = build_java_signature (TYPE_ARRAY_ELEMENT (type));
+             sig = ident_subst (IDENTIFIER_POINTER (t), IDENTIFIER_LENGTH (t),
+                                "[", 0, 0, "");
+           }
+         else
+           {
+             t = DECL_NAME (TYPE_NAME (type));
+             sig = ident_subst (IDENTIFIER_POINTER (t), IDENTIFIER_LENGTH (t),
+                                "L", '.', '/', ";");
+           }
+         break;
+       case METHOD_TYPE:
+       case FUNCTION_TYPE:
+         {
+           extern struct obstack temporary_obstack;
+           sig = build_java_argument_signature (type);
+           obstack_1grow (&temporary_obstack, '(');
+           obstack_grow (&temporary_obstack,
+                         IDENTIFIER_POINTER (sig), IDENTIFIER_LENGTH (sig));
+           obstack_1grow (&temporary_obstack, ')');
+  
+           t = build_java_signature (TREE_TYPE (type));
+           obstack_grow0 (&temporary_obstack,
+                          IDENTIFIER_POINTER (t), IDENTIFIER_LENGTH (t));
+
+           sig = get_identifier (obstack_base (&temporary_obstack));
+           obstack_free (&temporary_obstack,
+                         obstack_base (&temporary_obstack));
+         }
+         break;
+       bad_type:
+       default:
+         fatal ("internal error - build_java_signature passed invalid type");
+       }
+      TYPE_LANG_SPECIFIC (type)->signature = sig;
+    }
+  pop_obstacks ();
+  return sig;
+}
+
+/* Save signature string SIG (an IDENTIFIER_NODE) in TYPE for future use. */
+
+void
+set_java_signature (type, sig)
+     tree type;
+     tree sig;
+{
+  tree old_sig;
+  while (TREE_CODE (type) == POINTER_TYPE)
+    type = TREE_TYPE (type);
+  if (TYPE_LANG_SPECIFIC (type) == NULL)
+    {
+      TYPE_LANG_SPECIFIC (type) = (struct lang_type *)
+       perm_calloc (1, sizeof (struct lang_type));
+      
+    }
+  old_sig = TYPE_LANG_SPECIFIC (type)->signature;
+  if (old_sig != NULL_TREE && old_sig != sig)
+    fatal ("internal error - set_java_signature");
+  TYPE_LANG_SPECIFIC (type)->signature = sig;
+#if 0 /* careful about METHOD_TYPE */
+  if (IDENTIFIER_SIGNATURE_TYPE (sig) == NULL_TREE && TREE_PERMANENT (type))
+    IDENTIFIER_SIGNATURE_TYPE (sig) = type;
+#endif
+}
+
+/* Search in class CLAS (and its superclasses) for a method
+   matching METHOD_NAME and argument signature METHOD_SIGNATURE.
+   Return a FUNCTION_DECL on success, or NULL_TREE if none found.
+   (Contrast lookup_java_method, which takes into account return type.) */
+
+tree
+lookup_argument_method (clas, method_name, method_signature)
+     tree clas, method_name, method_signature;
+{
+  tree method;
+  while (clas != NULL_TREE)
+    {
+      for (method = TYPE_METHODS (clas);
+          method != NULL_TREE;  method = TREE_CHAIN (method))
+       {
+         tree method_sig = build_java_argument_signature (TREE_TYPE (method));
+         if (DECL_NAME (method) == method_name && method_sig == method_signature)
+           return method;
+       }
+      clas = CLASSTYPE_SUPER (clas);
+    }
+  return NULL_TREE;
+}
+
+/* Search in class CLAS (and its superclasses) for a method
+   matching METHOD_NAME and signature METHOD_SIGNATURE.
+   Return a FUNCTION_DECL on success, or NULL_TREE if none found.
+   (Contrast lookup_argument_method, which ignores return type.) */
+
+tree
+lookup_java_method (clas, method_name, method_signature)
+     tree clas, method_name, method_signature;
+{
+  tree method;
+  while (clas != NULL_TREE)
+    {
+      for (method = TYPE_METHODS (clas);
+          method != NULL_TREE;  method = TREE_CHAIN (method))
+       {
+         tree method_sig = build_java_signature (TREE_TYPE (method));
+         if (DECL_NAME (method) == method_name && method_sig == method_signature)
+           return method;
+       }
+      clas = CLASSTYPE_SUPER (clas);
+    }
+  return NULL_TREE;
+}
+
+/* Search in class CLAS (and its superclasses) for methods matching
+   METHOD_NAME and METHOD_SIGNATURE. Return a list of FUNCTION_DECLs.
+   When called from here, build_java_signature doesn't take the
+   returned type into account. */
+
+tree
+match_java_method (clas, method_name, method_signature)
+     tree clas, method_name, method_signature;
+{
+  tree method;
+  tree list = NULL_TREE;
+  while (clas != NULL_TREE)
+    {
+      for (method = TYPE_METHODS (clas);
+          method != NULL_TREE;  method = TREE_CHAIN (method))
+       {
+         tree method_sig = build_java_argument_signature (TREE_TYPE (method));
+         if (DECL_NAME (method) == method_name 
+             && method_sig == method_signature)
+           list = tree_cons (NULL_TREE, method, list);
+       }
+      clas = CLASSTYPE_SUPER (clas);
+    }
+  return list;
+}
+
+/* Search in class CLAS for a constructor matching METHOD_SIGNATURE.
+   Return a FUNCTION_DECL on success, or NULL_TREE if none found. */
+
+tree
+lookup_java_constructor (clas, method_signature)
+     tree clas, method_signature;
+{
+  tree method = TYPE_METHODS (clas);
+  for ( ; method != NULL_TREE;  method = TREE_CHAIN (method))
+    {
+      tree method_sig = build_java_signature (TREE_TYPE (method));
+      if (DECL_CONSTRUCTOR_P (method) && method_sig == method_signature)
+       return method;
+    }
+  return NULL_TREE;
+}
+
+/* Return a type which is the Binary Numeric Promotion of the pair T1,
+   T2 and convert EXP1 and/or EXP2. See 5.6.2 Binary Numeric
+   Promotion. It assumes that both T1 and T2 are elligible to BNP. */
+
+tree
+binary_numeric_promotion (t1, t2, exp1, exp2)
+     tree t1;
+     tree t2;
+     tree *exp1;
+     tree *exp2;
+{
+  if (t1 == double_type_node || t2 == double_type_node)
+    {
+      if (t1 != double_type_node)
+       *exp1 = convert (double_type_node, *exp1);
+      if (t2 != double_type_node)
+       *exp2 = convert (double_type_node, *exp2);
+      return double_type_node;
+    }
+  if (t1 == float_type_node || t2 == float_type_node)
+    {
+      if (t1 != float_type_node)
+       *exp1 = convert (float_type_node, *exp1);
+      if (t2 != float_type_node)
+       *exp2 = convert (float_type_node, *exp2);
+      return float_type_node;
+    }
+  if (t1 == long_type_node || t2 == long_type_node)
+    {
+      if (t1 != long_type_node)
+       *exp1 = convert (long_type_node, *exp1);
+      if (t2 != long_type_node)
+       *exp2 = convert (long_type_node, *exp2);
+      return long_type_node;
+    }
+
+  if (t1 != int_type_node)
+    *exp1 = convert (int_type_node, *exp1);
+  if (t2 != int_type_node)
+    *exp2 = convert (int_type_node, *exp2);
+  return int_type_node;
+}
diff --git a/gcc/java/verify.c b/gcc/java/verify.c
new file mode 100644 (file)
index 0000000..82a081f
--- /dev/null
@@ -0,0 +1,1239 @@
+/* Handle verification of bytecoded methods for the GNU compiler for 
+   the Java(TM) language.
+   Copyright (C) 1997 Free Software Foundation, Inc.
+
+This file is part of GNU CC.
+
+GNU CC is free software; you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation; either version 2, or (at your option)
+any later version.
+
+GNU CC is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with GNU CC; see the file COPYING.  If not, write to
+the Free Software Foundation, 59 Temple Place - Suite 330,
+Boston, MA 02111-1307, USA.
+
+Java and all Java-based marks are trademarks or registered trademarks
+of Sun Microsystems, Inc. in the United States and other countries.
+The Free Software Foundation is independent of Sun Microsystems, Inc.  */
+
+#include <stdio.h>
+#include "config.h"
+#include "tree.h"
+#include "java-tree.h"
+#include "javaop.h"
+#include "java-opcodes.h"
+#include "jcf.h"
+#include "java-except.h"
+
+extern int stack_pointer;
+
+/* During verification, start of the current subroutine (jsr target). */
+tree current_subr;
+
+/* A list of pending blocks, chained using  LABEL_PENDING_CHAIN.
+   A pending block is one that has LABEL_CHANGED set, which means
+   it requires (re-) verification. */
+tree pending_blocks;
+
+/* Append TARGET_LABEL to the pending_block stack unless already in it. */
+
+void
+push_pending_label (target_label) 
+     tree target_label;
+{
+  if (! LABEL_CHANGED (target_label))
+    {
+      LABEL_PENDING_CHAIN (target_label) = pending_blocks;
+      pending_blocks = target_label;
+      LABEL_CHANGED (target_label) = 1;
+    }
+}
+
+/* Note that TARGET_LABEL is a possible successor instruction.
+   Merge the type state etc.
+   Return NULL on sucess, or an error message on failure. */
+
+static char *
+check_pending_block (target_label)
+     tree target_label;
+{
+  int changed = merge_type_state (target_label);
+
+  if (changed)
+    {
+      if (changed < 0)
+       return "types could not be merged";
+      push_pending_label (target_label);
+    }
+
+  if (current_subr == NULL)
+    {
+      if (LABEL_IN_SUBR (target_label))
+       return "might transfer control into subroutine";
+    }
+  else
+    {
+      if (LABEL_IN_SUBR (target_label))
+       {
+         if (LABEL_SUBR_START (target_label) != current_subr)
+           return "transfer out of subroutine";
+       }
+      else if (! LABEL_VERIFIED (target_label))
+       {
+         LABEL_IN_SUBR (target_label) = 1;
+         LABEL_SUBR_START (target_label) = current_subr;
+       }
+      else
+       return "transfer out of subroutine";
+    }
+  return NULL;
+}
+
+/* Return the "merged" types of TYPE1 and TYPE2.
+   If either is primitive, the other must match (after promotion to int).
+   For reference types, return the common super-class.
+   Return TYPE_UNKNOWN if the types cannot be merged. */   
+
+tree
+merge_types (type1, type2)
+     tree type1, type2;
+{
+  if (type1 == type2)
+    return type1;
+  if (type1 == TYPE_UNKNOWN || type2 == TYPE_UNKNOWN
+      || type1 == TYPE_RETURN_ADDR || type2 == TYPE_RETURN_ADDR)
+    return TYPE_UNKNOWN;
+  if (TREE_CODE (type1) == POINTER_TYPE && TREE_CODE (type2) == POINTER_TYPE)
+    {
+      int depth1, depth2;
+      tree tt1, tt2;
+      /* ptr_type_node is only used for a null reference,
+        which is compatible with any reference type. */
+      if (type1 == ptr_type_node || type2 == object_ptr_type_node)
+       return type2;
+      if (type2 == ptr_type_node || type1 == object_ptr_type_node)
+       return type1;
+
+      tt1 = HANDLE_TO_CLASS_TYPE (TREE_TYPE (type1));
+      tt2 = HANDLE_TO_CLASS_TYPE (TREE_TYPE (type2));
+
+      if (TYPE_ARRAY_P (tt1) || TYPE_ARRAY_P (tt2))
+       {
+         if (TYPE_ARRAY_P (tt1) == TYPE_ARRAY_P (tt2))
+           {
+             tree el_type1 = TYPE_ARRAY_ELEMENT (tt1);
+             tree el_type2 = TYPE_ARRAY_ELEMENT (tt2);
+             tree el_type = NULL_TREE;
+             if (el_type1 == el_type2)
+               el_type = el_type1;
+             else if (TREE_CODE (el_type1) == POINTER_TYPE
+                      && TREE_CODE (el_type2) == POINTER_TYPE)
+               el_type = merge_types (el_type1, el_type2);
+             if (el_type != NULL_TREE)
+               {
+                 HOST_WIDE_INT len1 = java_array_type_length (tt1);
+                 HOST_WIDE_INT len2 = java_array_type_length (tt2);
+                 if (len1 != len2)
+                   len1 = -1;
+                 else if (el_type1 == el_type2)
+                   return type1;
+                 return promote_type (build_java_array_type (el_type, len1));
+               }
+           }
+         return object_ptr_type_node;
+       }
+      type1 = tt1;
+      type2 = tt2;
+
+      depth1 = class_depth (type1);
+      depth2 = class_depth (type2);
+      for ( ; depth1 > depth2;  depth1--)
+       type1 = TYPE_BINFO_BASETYPE (type1, 0);
+      for ( ; depth2 > depth1;  depth2--)
+       type2 = TYPE_BINFO_BASETYPE (type2, 0);
+      while (type1 != type2)
+       {
+         type1 = TYPE_BINFO_BASETYPE (type1, 0);
+         type2 = TYPE_BINFO_BASETYPE (type2, 0);
+       }
+      return promote_type (type1);
+    }
+  if (INTEGRAL_TYPE_P (type1) && INTEGRAL_TYPE_P (type2)
+      && TYPE_PRECISION (type1) <= 32 && TYPE_PRECISION (type2) <= 32)
+    return int_type_node;
+  return TYPE_UNKNOWN;
+}
+
+/* Merge the current type state with that at LABEL.
+   Return -1 the the states are incompatible (i.e. on error),
+   0 if there was no change, and 1 if there was a change. */
+
+int
+merge_type_state (label)
+     tree label;
+{
+  int nlocals = DECL_MAX_LOCALS(current_function_decl);
+  int cur_length = stack_pointer + nlocals;
+  tree vec = LABEL_TYPE_STATE (label);
+  tree return_map;
+  if (vec == NULL_TREE)
+    {
+      vec = make_tree_vec (cur_length);
+      LABEL_TYPE_STATE (label) = vec;
+      while (--cur_length >= 0)
+       TREE_VEC_ELT (vec, cur_length) = type_map [cur_length];
+      return 1;
+    }
+  else
+    {
+      int i;
+      int changed = 0;
+      if (LABEL_IS_SUBR_START (label) && LABEL_VERIFIED (label)
+         && current_subr != label)
+       return_map = LABEL_RETURN_TYPE_STATE (label);
+      else
+       return_map = NULL_TREE;
+      if (TREE_VEC_LENGTH (vec) != cur_length)
+       {
+         return -1;
+       }
+      for (i = 0; i < cur_length; i++)
+       {
+         tree old_type = TREE_VEC_ELT (vec, i);
+         tree new_type = merge_types (old_type, type_map [i]);
+         if (TREE_VEC_ELT (vec, i) != new_type)
+           {
+             /* If there has been a change, note that since we must re-verify.
+                However, if the label is the start of a subroutine,
+                we don't care about local variables that are neither
+                set nor used in the sub-routine. */
+             if (return_map == NULL_TREE || i >= nlocals
+                 || TREE_VEC_ELT (return_map, i) != TYPE_UNUSED
+                 || (TYPE_IS_WIDE (new_type)
+                     && TREE_VEC_ELT (return_map, i+1) != TYPE_UNUSED))
+               changed = 1;
+           }
+         TREE_VEC_ELT (vec, i) = new_type;
+         if (new_type == TYPE_UNKNOWN)
+           {
+             if (i >= nlocals)
+               return -1;
+           }
+         else if (TYPE_IS_WIDE (new_type))
+           i++;
+       }
+      return changed;
+    }
+}
+
+/* Handle dup-like operations. */
+
+static void
+type_stack_dup (size, offset)
+     int size, offset;
+{
+  tree type[4];
+  int index;
+  if (size + offset > stack_pointer)
+    error ("stack underflow - dup* operation");
+  for (index = 0;  index < size + offset; index++)
+    {
+      type[index] = stack_type_map[stack_pointer - 1];
+      if (type[index] == void_type_node)
+       {
+         index++;
+         type[index] = stack_type_map[stack_pointer - 2];
+         if (! TYPE_IS_WIDE (type[index]))
+           fatal ("internal error - dup operation");
+         if (index == size || index == size + offset)
+           fatal ("dup operation splits 64-bit number");
+       }
+      pop_type (type[index]);
+    }
+  for (index = size;  --index >= 0; )
+    {
+      if (type[index] != void_type_node)
+       push_type (type[index]);
+    }
+
+  for (index = size + offset;  --index >= 0; )
+    {
+      if (type[index] != void_type_node)
+       push_type (type[index]);
+    }
+}
+
+/* This causes the next iteration to ignore the next instruction
+   and look for some other unhandled instruction. */
+#define INVALIDATE_PC (prevpc = -1, oldpc = PC, PC = INVALID_PC)
+#define INVALID_PC (-1)
+
+#define VERIFICATION_ERROR(MESSAGE) \
+  do { message = MESSAGE;  goto verify_error; } while (0)
+
+#define PUSH_PENDING(LABEL) \
+     do { if ((message = check_pending_block (LABEL)) != NULL) \
+             goto verify_error; } while (0)
+
+#ifdef __GNUC__
+#define CHECK_PC_IN_RANGE(PC) ({if (PC < 0 || PC > length) goto bad_pc; 1;})
+#else
+#define CHECK_PC_IN_RANGE(PC) (PC < 0 || PC > length ? \
+  (fatal("Bad byte codes.\n"), 0) : 1)
+#endif
+
+#define BCODE byte_ops
+
+/* Verify the bytecodes of the current method.
+   Return 1 on sucess, 0 on failure. */
+int
+verify_jvm_instructions (jcf, byte_ops, length)
+     JCF* jcf;
+     unsigned char* byte_ops;
+     long length;
+{
+  tree label;
+  int wide = 0;
+  int op_code;
+  int PC;
+  int oldpc; /* PC of start of instruction. */
+  int prevpc;  /* If >= 0, PC of previous instruction. */
+  char *message;
+  int i;
+  register unsigned char *p;
+  struct eh_range *prev_eh_ranges = NULL_EH_RANGE;
+  struct eh_range *eh_ranges;
+
+  jint int_value = -1;
+
+  pending_blocks = NULL_TREE;
+
+  /* Handle the exception table. */
+  method_init_exceptions ();
+  JCF_SEEK (jcf, DECL_CODE_OFFSET (current_function_decl) + length);
+  i = JCF_readu2 (jcf);
+
+  /* We read the exception backwards. */
+  p = jcf->read_ptr + 8 * i;
+  while (--i >= 0)
+    {
+      int start_pc = GET_u2 (p-8);
+      int end_pc = GET_u2 (p-6);
+      int handler_pc = GET_u2 (p-4);
+      int catch_type = GET_u2 (p-2);
+      p -= 8;
+
+      if (start_pc < 0 || start_pc >= length
+         || end_pc < 0 || end_pc > length || start_pc >= end_pc
+         || handler_pc < 0 || handler_pc >= length
+         || (handler_pc >= start_pc && handler_pc < end_pc)
+         || ! (instruction_bits [start_pc] & BCODE_INSTRUCTION_START)
+         || ! (instruction_bits [end_pc] & BCODE_INSTRUCTION_START)
+         || ! (instruction_bits [handler_pc] & BCODE_INSTRUCTION_START))
+       {
+         error ("bad pc in exception_table");
+         return 0;
+       }
+
+      if (! add_handler (start_pc, end_pc,
+                        lookup_label (handler_pc),
+                        catch_type == 0 ? NULL_TREE
+                        : get_class_constant (jcf, catch_type)))
+       {
+         error ("overlapping exception ranges are not supported");
+         return 0;
+       }
+
+      instruction_bits [handler_pc] |= BCODE_EXCEPTION_TARGET;
+    }
+
+  for (PC = 0;;)
+    {
+      int index;
+      tree type, tmp;
+      if (((PC != INVALID_PC
+          && instruction_bits [PC] & BCODE_TARGET) != 0)
+         || PC == 0)
+       {
+         PUSH_PENDING (lookup_label (PC));
+         INVALIDATE_PC;
+       }
+      if (PC == INVALID_PC)
+       {
+         label = pending_blocks;
+         if (label == NULL_TREE)
+           break;  /* We're done! */
+         pending_blocks = LABEL_PENDING_CHAIN (label);
+         LABEL_CHANGED (label) = 0;
+
+         if (LABEL_IN_SUBR (label))
+           current_subr = LABEL_SUBR_START (label);
+         else
+           current_subr = NULL_TREE;
+
+         /* Restore type_map and stack_pointer from
+            LABEL_TYPE_STATE (label), and continue
+            compiling from there. */
+         load_type_state (label);
+         PC = LABEL_PC (label);
+       }
+      else if (PC >= length)
+       VERIFICATION_ERROR ("falling through end of method");
+
+      oldpc = PC;
+
+      if (!(instruction_bits [PC] & BCODE_INSTRUCTION_START) && ! wide)
+       VERIFICATION_ERROR ("PC not at instruction start");
+
+      instruction_bits[PC] |= BCODE_VERIFIED;
+
+      eh_ranges = find_handler (oldpc);
+
+      op_code = byte_ops[PC++];
+      switch (op_code)
+       {
+         int is_static, is_putting;
+       case OPCODE_nop:
+         break;
+       case OPCODE_iconst_m1:
+       case OPCODE_iconst_0:   case OPCODE_iconst_1:   case OPCODE_iconst_2:
+       case OPCODE_iconst_3:   case OPCODE_iconst_4:   case OPCODE_iconst_5:
+         i = op_code - OPCODE_iconst_0;
+         goto push_int;
+       push_int:
+         if (byte_ops[PC] == OPCODE_newarray
+             || byte_ops[PC] == OPCODE_newarray)
+           int_value = i;
+         push_type (int_type_node);  break;
+       case OPCODE_lconst_0:   case OPCODE_lconst_1:
+         push_type (long_type_node);  break;
+       case OPCODE_fconst_0:   case OPCODE_fconst_1:   case OPCODE_fconst_2:
+         push_type (float_type_node);  break;
+       case OPCODE_dconst_0:   case OPCODE_dconst_1:
+         push_type (double_type_node);  break;
+       case OPCODE_bipush:
+         i = IMMEDIATE_s1;
+         goto push_int;
+       case OPCODE_sipush:
+         i = IMMEDIATE_s2;
+         goto push_int;
+       case OPCODE_iload:  type = int_type_node;  goto general_load;
+       case OPCODE_lload:  type = long_type_node;  goto general_load;
+       case OPCODE_fload:  type = float_type_node;  goto general_load;
+       case OPCODE_dload:  type = double_type_node;  goto general_load;
+       case OPCODE_aload:  type = ptr_type_node;  goto general_load;
+       general_load:
+       index = wide ? IMMEDIATE_u2 : IMMEDIATE_u1;
+       wide = 0;
+       goto load;
+       case OPCODE_iload_0:  type = int_type_node;  index = 0; goto load;
+       case OPCODE_iload_1:  type = int_type_node;  index = 1; goto load;
+       case OPCODE_iload_2:  type = int_type_node;  index = 2; goto load;
+       case OPCODE_iload_3:  type = int_type_node;  index = 3; goto load;
+       case OPCODE_lload_0:  type = long_type_node; index = 0; goto load;
+       case OPCODE_lload_1:  type = long_type_node; index = 1; goto load;
+       case OPCODE_lload_2:  type = long_type_node; index = 2; goto load;
+       case OPCODE_lload_3:  type = long_type_node; index = 3; goto load;
+       case OPCODE_fload_0:  type = float_type_node; index = 0; goto load;
+       case OPCODE_fload_1:  type = float_type_node; index = 1; goto load;
+       case OPCODE_fload_2:  type = float_type_node; index = 2; goto load;
+       case OPCODE_fload_3:  type = float_type_node; index = 3; goto load;
+       case OPCODE_dload_0: type = double_type_node; index = 0; goto load;
+       case OPCODE_dload_1: type = double_type_node; index = 1; goto load;
+       case OPCODE_dload_2: type = double_type_node; index = 2; goto load;
+       case OPCODE_dload_3: type = double_type_node; index = 3; goto load;
+       case OPCODE_aload_0:  type = ptr_type_node;  index = 0;  goto load;
+       case OPCODE_aload_1:  type = ptr_type_node;  index = 1;  goto load;
+       case OPCODE_aload_2:  type = ptr_type_node;  index = 2;  goto load;
+       case OPCODE_aload_3:  type = ptr_type_node;  index = 3;  goto load;
+       load:
+       if (index < 0
+           || (index + TYPE_IS_WIDE (type)
+               >= DECL_MAX_LOCALS (current_function_decl)))
+         VERIFICATION_ERROR ("invalid local variable index in load");
+       tmp = type_map[index];
+       if (tmp == TYPE_UNKNOWN || tmp == TYPE_SECOND
+           || (TYPE_IS_WIDE (type)
+               && type_map[index+1] != void_type_node)
+           || (type == ptr_type_node
+               ? TREE_CODE (tmp) != POINTER_TYPE
+               : type == int_type_node
+               ? (! INTEGRAL_TYPE_P (tmp) || TYPE_PRECISION (tmp) > 32)
+               : type != tmp))
+         VERIFICATION_ERROR("invalid local variable type in load");
+       push_type (tmp);
+       goto note_used;
+       case OPCODE_istore:  type = int_type_node;  goto general_store;
+       case OPCODE_lstore:  type = long_type_node;  goto general_store;
+       case OPCODE_fstore:  type = float_type_node;  goto general_store;
+       case OPCODE_dstore:  type = double_type_node;  goto general_store;
+       case OPCODE_astore:  type = ptr_type_node;  goto general_store;
+       general_store:
+       index = wide ? IMMEDIATE_u2 : IMMEDIATE_u1;
+       wide = 0;
+       goto store;
+       case OPCODE_istore_0:  type = int_type_node; index = 0; goto store;
+       case OPCODE_istore_1:  type = int_type_node; index = 1; goto store;
+       case OPCODE_istore_2:  type = int_type_node; index = 2; goto store;
+       case OPCODE_istore_3:  type = int_type_node; index = 3; goto store;
+       case OPCODE_lstore_0:  type = long_type_node; index=0; goto store;
+       case OPCODE_lstore_1:  type = long_type_node; index=1; goto store;
+       case OPCODE_lstore_2:  type = long_type_node; index=2; goto store;
+       case OPCODE_lstore_3:  type = long_type_node; index=3; goto store;
+       case OPCODE_fstore_0:  type=float_type_node; index=0; goto store;
+       case OPCODE_fstore_1:  type=float_type_node; index=1; goto store;
+       case OPCODE_fstore_2:  type=float_type_node; index=2; goto store;
+       case OPCODE_fstore_3:  type=float_type_node; index=3; goto store;
+       case OPCODE_dstore_0:  type=double_type_node; index=0; goto store;
+       case OPCODE_dstore_1:  type=double_type_node; index=1; goto store;
+       case OPCODE_dstore_2:  type=double_type_node; index=2; goto store;
+       case OPCODE_dstore_3:  type=double_type_node; index=3; goto store;
+       case OPCODE_astore_0:  type = ptr_type_node; index = 0; goto store;
+       case OPCODE_astore_1:  type = ptr_type_node; index = 1; goto store;
+       case OPCODE_astore_2:  type = ptr_type_node; index = 2; goto store;
+       case OPCODE_astore_3:  type = ptr_type_node; index = 3; goto store;
+       store:
+       if (index < 0
+           || (index + TYPE_IS_WIDE (type)
+               >= DECL_MAX_LOCALS (current_function_decl)))
+         {
+           VERIFICATION_ERROR ("invalid local variable index in store");
+           return 0;
+         }
+       type = pop_type (type);
+       type_map[index] = type;
+
+       /* If local variable changed, we need to reconsider eh handlers. */
+       prev_eh_ranges = NULL_EH_RANGE;
+
+       /* Allocate decl and rtx for this variable now, so if we're not
+          optmizing, we get a temporary that survives the whole method. */
+       find_local_variable (index, type, oldpc);
+
+        if (TYPE_IS_WIDE (type))
+          type_map[index+1] = TYPE_SECOND;
+       /* ... fall through to note_used ... */
+       note_used:
+         /* For store or load, note that local variable INDEX is used.
+            This is needed to verify try-finally sub-routines. */
+         if (current_subr)
+           {
+             tree vec = LABEL_RETURN_TYPE_STATE (current_subr);
+             tree subr_vec = LABEL_TYPE_STATE (current_subr);
+             int len = 1 + TYPE_IS_WIDE (type);
+             while (--len >= 0)
+               {
+                 if (TREE_VEC_ELT (vec, index) == TYPE_UNUSED)
+                   TREE_VEC_ELT (vec, index) = TREE_VEC_ELT (subr_vec, index);
+               }
+           }
+       break;
+       case OPCODE_iadd:
+       case OPCODE_iand:
+       case OPCODE_idiv:
+       case OPCODE_imul:
+       case OPCODE_ior:
+       case OPCODE_irem:
+       case OPCODE_ishl:
+       case OPCODE_ishr:
+       case OPCODE_isub:
+       case OPCODE_iushr:
+       case OPCODE_ixor:
+         type = int_type_node;  goto binop;
+       case OPCODE_ineg:
+       case OPCODE_i2c:
+       case OPCODE_i2b:
+       case OPCODE_i2s:
+         type = int_type_node;  goto unop;
+       case OPCODE_ladd:
+       case OPCODE_land:
+       case OPCODE_ldiv:
+       case OPCODE_lsub:
+       case OPCODE_lmul:
+       case OPCODE_lrem:
+       case OPCODE_lor:
+       case OPCODE_lxor:
+         type = long_type_node;  goto binop;
+       case OPCODE_lneg:
+         type = long_type_node;  goto unop;
+       case OPCODE_fadd:       case OPCODE_fsub:
+       case OPCODE_fmul:       case OPCODE_fdiv:       case OPCODE_frem:
+         type = float_type_node;  goto binop;
+       case OPCODE_fneg:
+         type = float_type_node;  goto unop;
+       case OPCODE_dadd:       case OPCODE_dsub:
+       case OPCODE_dmul:       case OPCODE_ddiv:       case OPCODE_drem:
+         type = double_type_node;  goto binop;
+       case OPCODE_dneg:
+         type = double_type_node;  goto unop;
+       unop:
+         pop_type (type);
+         push_type (type);
+         break;
+       binop:
+         pop_type (type);
+         pop_type (type);
+         push_type (type);
+         break;
+       case OPCODE_lshl:
+       case OPCODE_lshr:
+       case OPCODE_lushr:
+         pop_type (int_type_node);
+         pop_type (long_type_node);
+         push_type (long_type_node);
+         break;
+       case OPCODE_iinc:
+         index = wide ? IMMEDIATE_u2 : IMMEDIATE_u1;
+         PC += wide + 1;
+         wide = 0;
+         if (index < 0 || index >= DECL_MAX_LOCALS (current_function_decl))
+           VERIFICATION_ERROR ("invalid local variable index in iinc");
+         tmp = type_map[index];
+         if (! INTEGRAL_TYPE_P (tmp) || TYPE_PRECISION (tmp) > 32)
+           VERIFICATION_ERROR ("invalid local variable type in iinc");
+         break;
+       case OPCODE_i2l:
+         pop_type (int_type_node);    push_type (long_type_node);   break;
+       case OPCODE_i2f:
+         pop_type (int_type_node);    push_type (float_type_node);  break;
+       case OPCODE_i2d:
+         pop_type (int_type_node);    push_type (double_type_node); break;
+       case OPCODE_l2i:
+         pop_type (long_type_node);   push_type (int_type_node);    break;
+       case OPCODE_l2f:
+         pop_type (long_type_node);   push_type (float_type_node);  break;
+       case OPCODE_l2d:
+         pop_type (long_type_node);   push_type (double_type_node); break;
+       case OPCODE_f2i:
+         pop_type (float_type_node);  push_type (int_type_node);    break;
+       case OPCODE_f2l:
+         pop_type (float_type_node);  push_type (long_type_node);   break;
+       case OPCODE_f2d:
+         pop_type (float_type_node);  push_type (double_type_node); break;
+       case OPCODE_d2i:
+         pop_type (double_type_node); push_type (int_type_node);    break;
+       case OPCODE_d2l:
+         pop_type (double_type_node); push_type (long_type_node);   break;
+       case OPCODE_d2f:
+         pop_type (double_type_node); push_type (float_type_node);  break;
+       case OPCODE_lcmp:
+         type = long_type_node;  goto compare;
+       case OPCODE_fcmpl:
+       case OPCODE_fcmpg:
+         type = float_type_node;  goto compare;
+       case OPCODE_dcmpl:
+       case OPCODE_dcmpg:
+         type = double_type_node;  goto compare;
+       compare:
+         pop_type (type);  pop_type (type);
+         push_type (int_type_node);  break;
+       case OPCODE_ifeq:
+       case OPCODE_ifne:
+       case OPCODE_iflt:
+       case OPCODE_ifge:
+       case OPCODE_ifgt:
+       case OPCODE_ifle:
+         pop_type (int_type_node);  goto cond;
+       case OPCODE_ifnull:
+       case OPCODE_ifnonnull:
+         pop_type (ptr_type_node ); goto cond;
+       case OPCODE_if_icmpeq:
+       case OPCODE_if_icmpne:
+       case OPCODE_if_icmplt:
+       case OPCODE_if_icmpge:
+       case OPCODE_if_icmpgt:
+       case OPCODE_if_icmple:
+         pop_type (int_type_node);  pop_type (int_type_node);  goto cond;
+       case OPCODE_if_acmpeq:
+       case OPCODE_if_acmpne:
+         pop_type (object_ptr_type_node);  pop_type (object_ptr_type_node);
+         goto cond;
+       cond:
+         PUSH_PENDING (lookup_label (oldpc + IMMEDIATE_s2));
+         break;
+       case OPCODE_goto:
+         PUSH_PENDING (lookup_label (oldpc + IMMEDIATE_s2));
+         INVALIDATE_PC;
+         break;
+       case OPCODE_wide:
+         switch (byte_ops[PC])
+           {
+           case OPCODE_iload:  case OPCODE_lload:
+           case OPCODE_fload:  case OPCODE_dload:  case OPCODE_aload:
+           case OPCODE_istore:  case OPCODE_lstore:
+           case OPCODE_fstore:  case OPCODE_dstore:  case OPCODE_astore:
+           case OPCODE_iinc:
+           case OPCODE_ret:
+             wide = 1;
+             break;
+           default:
+             VERIFICATION_ERROR ("invalid use of wide instruction");
+           }
+         break;
+       case OPCODE_ireturn:  type = int_type_node;    goto ret;
+       case OPCODE_lreturn:  type = long_type_node;   goto ret;
+       case OPCODE_freturn:  type = float_type_node;  goto ret;
+       case OPCODE_dreturn:  type = double_type_node; goto ret;
+       case OPCODE_areturn:  type = ptr_type_node;    goto ret;
+       ret:
+         pop_type (type);
+         /* ... fall through ... */
+       case OPCODE_return:
+         INVALIDATE_PC;
+         break;
+       case OPCODE_getstatic: is_putting = 0;  is_static = 1;  goto field;
+       case OPCODE_putstatic: is_putting = 1;  is_static = 1;  goto field;
+       case OPCODE_getfield:  is_putting = 0;  is_static = 0;  goto field;
+       case OPCODE_putfield:  is_putting = 1;  is_static = 0;  goto field;
+       field:
+         {
+           int index = IMMEDIATE_u2;
+           tree self_type = get_class_constant
+             (jcf, COMPONENT_REF_CLASS_INDEX (&current_jcf->cpool, index));
+           tree field_name = COMPONENT_REF_NAME (&current_jcf->cpool, index);
+           tree field_signature = COMPONENT_REF_SIGNATURE (&current_jcf->cpool, index);
+           tree field_type = get_type_from_signature (field_signature);
+           if (is_putting)
+             pop_type (field_type);
+           if (! is_static)
+             {
+               /* Defer actual checking until next pass. */
+               pop_type (ptr_type_node);
+             }
+           if (! is_putting)
+             push_type (field_type);
+           break;
+         }
+       case OPCODE_new:
+         push_type (get_class_constant (jcf, IMMEDIATE_u2));
+         break;
+       case OPCODE_dup:     type_stack_dup (1, 0);  break;
+       case OPCODE_dup_x1:  type_stack_dup (1, 1);  break;
+       case OPCODE_dup_x2:  type_stack_dup (1, 2);  break;
+       case OPCODE_dup2:    type_stack_dup (2, 0);  break;
+       case OPCODE_dup2_x1: type_stack_dup (2, 1);  break;
+       case OPCODE_dup2_x2: type_stack_dup (2, 2);  break;
+       case OPCODE_pop:  index = 1;  goto pop;
+       case OPCODE_pop2: index = 2;  goto pop;
+       pop:
+         if (stack_pointer < index)
+           VERIFICATION_ERROR ("stack underflow");
+         stack_pointer -= index;
+         break;
+       case OPCODE_swap:
+         if (stack_pointer < 2)
+           VERIFICATION_ERROR ("stack underflow (in swap)");
+         else
+           {
+             tree type1 = stack_type_map[stack_pointer - 1];
+             tree type2 = stack_type_map[stack_pointer - 2];
+             if (type1 == void_type_node || type2 == void_type_node)
+               VERIFICATION_ERROR ("verifier (swap):  double or long value");
+             stack_type_map[stack_pointer - 2] = type1;
+             stack_type_map[stack_pointer - 1] = type2;
+           }
+         break;
+       case OPCODE_ldc:   index = IMMEDIATE_u1;  goto ldc;
+       case OPCODE_ldc2_w:
+       case OPCODE_ldc_w:
+         index = IMMEDIATE_u2;  goto ldc;
+       ldc:
+         if (index <= 0 || index >= JPOOL_SIZE(current_jcf))
+           VERIFICATION_ERROR ("bad constant pool index in ldc");
+         int_value = -1;
+         switch (JPOOL_TAG (current_jcf, index) & ~CONSTANT_ResolvedFlag)
+           {
+           case CONSTANT_Integer:  type = int_type_node;  goto check_ldc;
+           case CONSTANT_Float:    type = float_type_node;  goto check_ldc;
+           case CONSTANT_String:   type = string_type_node; goto check_ldc;
+           case CONSTANT_Long:    type = long_type_node;    goto check_ldc;
+           case CONSTANT_Double:  type = double_type_node;  goto check_ldc;
+           check_ldc:
+             if (TYPE_IS_WIDE (type) == (op_code == OPCODE_ldc2_w))
+               break;
+             /* ... else fall through ... */
+           default:
+           bad_ldc:
+             VERIFICATION_ERROR ("bad constant pool tag in ldc");
+           }
+         if (type == int_type_node)
+           {
+             i = TREE_INT_CST_LOW (get_constant (current_jcf, index));
+             goto push_int;
+           }
+         push_type (type);
+         break;
+
+       case OPCODE_invokevirtual:
+       case OPCODE_invokespecial:
+       case OPCODE_invokestatic:
+       case OPCODE_invokeinterface:
+         {
+           int index = IMMEDIATE_u2;
+           tree sig = COMPONENT_REF_SIGNATURE (&current_jcf->cpool, index);
+           tree self_type = get_class_constant
+             (current_jcf, COMPONENT_REF_CLASS_INDEX (&current_jcf->cpool,
+                                                      index));
+           tree method_name = COMPONENT_REF_NAME (&current_jcf->cpool, index);
+           tree method_type;
+           method_type = parse_signature_string (IDENTIFIER_POINTER (sig),
+                                                 IDENTIFIER_LENGTH (sig));
+           if (TREE_CODE (method_type) != FUNCTION_TYPE)
+             VERIFICATION_ERROR ("bad method signature");
+           pop_argument_types (TYPE_ARG_TYPES (method_type));
+
+           /* Can't invoke <clinit> */
+           if (method_name == clinit_identifier_node)
+             VERIFICATION_ERROR ("invoke opcode can't invoke <clinit>");
+           /* Apart invokespecial, can't invoke <init> */
+           if (op_code != OPCODE_invokespecial
+               && method_name == init_identifier_node)
+             VERIFICATION_ERROR ("invoke opcode can't invoke <init>");
+
+           if (op_code != OPCODE_invokestatic)
+             pop_type (self_type);
+
+           switch (op_code)
+             {
+             case OPCODE_invokeinterface:
+               {
+                 int nargs    = IMMEDIATE_u1;
+                 int notZero  = IMMEDIATE_u1;
+               
+                 if (!nargs || notZero)
+                     VERIFICATION_ERROR 
+                       ("invalid argument number in invokeinterface");
+                 break;                  
+               }
+             }
+
+           if (TREE_TYPE (method_type) != void_type_node)
+             push_type (TREE_TYPE (method_type));
+           break;
+         }
+
+       case OPCODE_arraylength:
+           /* Type checking actually made during code generation */
+           pop_type( ptr_type_node );
+           push_type( int_type_node );
+           break;
+           
+        /* Q&D verification *or* more checking done during code generation
+          for byte/boolean/char/short, the value popped is a int coerced
+          into the right type before being stored.  */
+       case OPCODE_iastore: type = int_type_node;     goto astore;
+       case OPCODE_lastore: type = long_type_node;    goto astore;
+       case OPCODE_fastore: type = float_type_node;   goto astore;
+       case OPCODE_dastore: type = double_type_node;  goto astore;
+       case OPCODE_aastore: type = ptr_type_node;     goto astore;
+       case OPCODE_bastore: type = int_type_node; goto astore;
+       case OPCODE_castore: type = int_type_node; goto astore;
+       case OPCODE_sastore: type = int_type_node; goto astore;
+       astore:
+         /* FIXME - need better verification here */
+         pop_type (type);           /* new value */
+         pop_type (int_type_node);  /* index */
+         pop_type (ptr_type_node);  /* array */
+         break;
+
+        /* Q&D verification *or* more checking done during code generation
+          for byte/boolean/char/short, the value pushed is a int.  */
+       case OPCODE_iaload: type = int_type_node;     goto aload;
+       case OPCODE_laload: type = long_type_node;    goto aload;
+       case OPCODE_faload: type = float_type_node;   goto aload;
+       case OPCODE_daload: type = double_type_node;  goto aload;
+       case OPCODE_aaload: type = ptr_type_node;     goto aload;
+       case OPCODE_baload: type = promote_type (byte_type_node);  goto aload;
+       case OPCODE_caload: type = promote_type (char_type_node);  goto aload;
+       case OPCODE_saload: type = promote_type (short_type_node); goto aload;
+        aload:
+         pop_type (int_type_node);
+         type = pop_type (ptr_type_node);
+         if (! is_array_type_p (type))
+           VERIFICATION_ERROR ("array load from non-array type");
+         push_type (TYPE_ARRAY_ELEMENT (TREE_TYPE (type)));
+         break;
+
+       case OPCODE_anewarray:
+         type = get_class_constant (current_jcf, IMMEDIATE_u2);
+         type = promote_type (type);
+         goto newarray;
+
+       case OPCODE_newarray:
+         index = IMMEDIATE_u1;
+         type = decode_newarray_type (index);
+         if (type == NULL_TREE)
+           VERIFICATION_ERROR ("invalid type code in newarray opcode");
+         goto newarray;
+
+       newarray:
+         if (int_value >= 0 && prevpc >= 0)
+           {
+             /* If previous instruction pushed int constant,
+                we want to use it. */
+             switch (byte_ops[prevpc])
+               {
+               case OPCODE_iconst_0: case OPCODE_iconst_1:
+               case OPCODE_iconst_2: case OPCODE_iconst_3:
+               case OPCODE_iconst_4: case OPCODE_iconst_5:
+               case OPCODE_bipush:  case OPCODE_sipush:
+               case OPCODE_ldc: case OPCODE_ldc_w:
+                 break;
+               default:
+                 int_value = -1;
+               }
+           }
+         else
+           int_value = -1;
+         type = build_java_array_type (type, int_value);
+         pop_type (int_type_node);
+         push_type (type);
+         break;
+
+       case OPCODE_multianewarray:
+         {
+           int ndim, i;
+           index = IMMEDIATE_u2;
+           ndim  = IMMEDIATE_u1;
+
+            if( ndim < 1 )
+              VERIFICATION_ERROR ("number of dimension lower that 1 in multianewarray" );
+
+           for( i = 0; i < ndim; i++ )
+             pop_type (int_type_node);
+           push_type (get_class_constant (current_jcf, index));
+           break;
+         }
+
+       case OPCODE_aconst_null:
+         push_type (ptr_type_node);
+         break;
+
+       case OPCODE_athrow:
+         pop_type (throwable_type_node);
+         INVALIDATE_PC;
+         break;
+
+       case OPCODE_checkcast:
+         pop_type (ptr_type_node);
+         type = get_class_constant (current_jcf, IMMEDIATE_u2);
+         push_type (type);
+         break;
+       case OPCODE_instanceof:
+         pop_type (ptr_type_node);
+         get_class_constant (current_jcf, IMMEDIATE_u2);
+         push_type (integer_type_node);
+         break;
+
+       case OPCODE_tableswitch:
+         {
+           jint default_val, low, high;
+
+           pop_type (integer_type_node);
+           while (PC%4)
+             {
+               if (byte_ops[PC++])
+                 VERIFICATION_ERROR ("bad alignment in tableswitch pad");
+             }
+           PUSH_PENDING (lookup_label (oldpc+IMMEDIATE_s4));
+           low  = IMMEDIATE_s4;
+           high = IMMEDIATE_s4;
+
+           if (low > high)
+             VERIFICATION_ERROR ("unsorted low/high value in tableswitch");
+
+           while (low++ <= high)
+             PUSH_PENDING (lookup_label (oldpc + IMMEDIATE_s4));
+           break;
+         }
+
+       case OPCODE_lookupswitch:
+         {
+           jint npairs, last, not_registered = 1;
+
+           pop_type (integer_type_node);
+           while (PC%4)
+             {
+               if (byte_ops[PC++])
+                 VERIFICATION_ERROR ("bad alignment in lookupswitch pad");
+             }
+
+           PUSH_PENDING (lookup_label (oldpc+IMMEDIATE_s4));
+           npairs = IMMEDIATE_s4;
+           
+           if (npairs < 0)
+             VERIFICATION_ERROR ("invalid number of targets in lookupswitch");
+
+           while (npairs--)
+             {
+               int match = IMMEDIATE_s4;
+               if (not_registered)
+                 not_registered = 0;
+               else if (last >= match)
+                 VERIFICATION_ERROR ("unsorted match value in lookupswitch");
+
+               last = match;
+               PUSH_PENDING (lookup_label (oldpc + IMMEDIATE_s4));
+             }
+           break;
+         }
+
+       case OPCODE_monitorenter: 
+         /* fall thru */
+       case OPCODE_monitorexit:
+         pop_type (ptr_type_node);
+         break;
+
+       case OPCODE_goto_w:
+         PUSH_PENDING (lookup_label (oldpc + IMMEDIATE_s4));
+         INVALIDATE_PC;
+         break;
+
+       case OPCODE_jsr:
+         {
+           tree target = lookup_label (oldpc + IMMEDIATE_s2);
+           tree return_label = lookup_label (PC);
+           push_type (return_address_type_node);
+           if (! LABEL_VERIFIED (target))
+             {
+               /* first time seen */
+               tree return_type_map;
+               int nlocals = DECL_MAX_LOCALS (current_function_decl);
+               index = nlocals + DECL_MAX_STACK (current_function_decl);
+               return_type_map = make_tree_vec (index);
+               while (--index >= nlocals)
+                 TREE_VEC_ELT (return_type_map, index) = TYPE_UNKNOWN;
+               while (--index >= 0)
+                 TREE_VEC_ELT (return_type_map, index) = TYPE_UNUSED;
+               LABEL_RETURN_LABEL (target)
+                 = build_decl (LABEL_DECL, NULL_TREE, TREE_TYPE (target));
+               LABEL_PC (LABEL_RETURN_LABEL (target)) = -1;
+               LABEL_RETURN_TYPE_STATE (target) = return_type_map;
+               LABEL_IS_SUBR_START (target) = 1;
+               LABEL_IN_SUBR (target) = 1;
+               LABEL_SUBR_START (target) = target;
+               LABEL_SUBR_CONTEXT (target) = current_subr;
+             }
+           else if (! LABEL_IS_SUBR_START (target)
+                    || LABEL_SUBR_CONTEXT (target) != current_subr)
+             VERIFICATION_ERROR ("label part of different subroutines");
+
+           i = merge_type_state (target);
+           if (i != 0)
+             {
+               if (i < 0)
+                 VERIFICATION_ERROR ("types could not be merged at jsr");
+               push_pending_label (target);
+             }
+           current_subr = target;
+
+           /* Chain return_pc onto LABEL_RETURN_LABELS (target) if needed. */
+           if (! value_member (return_label, LABEL_RETURN_LABELS (target)))
+             {
+               LABEL_RETURN_LABELS (target)
+                 = tree_cons (NULL_TREE, return_label,
+                              LABEL_RETURN_LABELS (target));
+             }
+
+           if (LABEL_VERIFIED (target))
+             {
+               tree return_map = LABEL_RETURN_TYPE_STATE (target);
+               int len = TREE_VEC_LENGTH (return_map);
+               stack_pointer = len - DECL_MAX_LOCALS (current_function_decl);
+               while (--len >= 0)
+                 {
+                   if (TREE_VEC_ELT (return_map, len) != TYPE_UNUSED)
+                     type_map[len] = TREE_VEC_ELT (return_map, len);
+                 }
+               current_subr = LABEL_SUBR_CONTEXT (target);
+               PUSH_PENDING (return_label);
+             }
+
+           INVALIDATE_PC;
+         }
+         break;
+       case OPCODE_ret:
+         if (current_subr == NULL)
+           VERIFICATION_ERROR ("ret instruction not in a jsr subroutine");
+         else
+           {
+             tree ret_map = LABEL_RETURN_TYPE_STATE (current_subr);
+             tree caller = LABEL_SUBR_CONTEXT (current_subr);
+             int size = DECL_MAX_LOCALS(current_function_decl)+stack_pointer;
+             index = wide ? IMMEDIATE_u2 : IMMEDIATE_u1;
+             wide = 0;
+             INVALIDATE_PC;
+             if (index < 0 || index >= DECL_MAX_LOCALS (current_function_decl)
+                 || type_map[index] != TYPE_RETURN_ADDR)
+               VERIFICATION_ERROR ("invalid ret index");
+
+             /* The next chunk of code is similar to an inlined version of
+              *     merge_type_state (LABEL_RETURN_LABEL (current_subr)).
+              * The main differences are that LABEL_RETURN_LABEL is
+              * pre-allocated by the jsr (but we don't know the size then);
+              * and that we have to handle TYPE_UNUSED. */
+
+             if (! RETURN_MAP_ADJUSTED (ret_map))
+               { /* First return from this subroutine - fix stack pointer. */
+                 TREE_VEC_LENGTH (ret_map) = size;
+                 for (index = size;  --index >= 0; )
+                   {
+                     if (TREE_VEC_ELT (ret_map, index) != TYPE_UNUSED)
+                       TREE_VEC_ELT (ret_map, index) = type_map[index];
+                   }
+                 RETURN_MAP_ADJUSTED (ret_map) = 1;
+               }
+             else
+               {
+                 if (TREE_VEC_LENGTH (ret_map) != size)
+                   VERIFICATION_ERROR ("inconsistent stack size on ret");
+                 for (index = 0;  index < size;  index++)
+                   {
+                     tree type = TREE_VEC_ELT (ret_map, index);
+                     if (type != TYPE_UNUSED)
+                       {
+                         type = merge_types (type, type_map [index]);
+                         TREE_VEC_ELT (ret_map, index) = type;
+                         if (type == TYPE_UNKNOWN)
+                           {
+                             if (index >= size - stack_pointer)
+                               VERIFICATION_ERROR
+                                 ("inconsistent types on ret from jsr");
+                           }
+                         else if (TYPE_IS_WIDE (type))
+                           index++;
+                       }
+                   }
+               }
+
+             /* Check if there are any more pending blocks in this subroutine.
+                Because we push pending blocks in a last-in-first-out order,
+                and because we don't push anything from our caller until we
+                are done with this subroutine or anything nested in it,
+                then we are done if the top of the pending_blocks stack is
+                not in a subroutine, or it is in our caller. */
+             if (pending_blocks == NULL_TREE
+                 || ! LABEL_IN_SUBR (pending_blocks)
+                 || LABEL_SUBR_START (pending_blocks) == caller)
+               {
+                 /* Since we are done with this subroutine (i.e. this is the
+                    last ret from it), set up the (so far known) return
+                    address as pending - with the merged type state. */
+                 tmp = LABEL_RETURN_LABELS (current_subr);
+                 current_subr = caller;
+                 for ( ; tmp != NULL_TREE;  tmp = TREE_CHAIN (tmp))
+                   {
+                     tree return_label = TREE_VALUE (tmp);
+                     tree return_state = LABEL_TYPE_STATE (return_label);
+                     if (return_state == NULL_TREE)
+                       {
+                         /* This means means we had not verified the
+                            subroutine earlier, so this is the first jsr to
+                            call it.  In this case, the type_map of the return
+                            address is just the current type_map - and that
+                            is handled by the following PUSH_PENDING. */
+                       }
+                     else
+                       {
+                         /* In this case we have to do a merge.  But first
+                            restore the type_map for unused slots to those
+                            that were in effect at the jsr. */
+                         for (index = size;  --index >= 0; )
+                           {
+                             type_map[index] = TREE_VEC_ELT (ret_map, index);
+                             if (type_map[index] == TYPE_UNUSED)
+                               type_map[index]
+                                 = TREE_VEC_ELT (return_state, index);
+                           }
+                       }
+                     PUSH_PENDING (return_label);
+                   }
+               }
+           }
+         break;
+       case OPCODE_jsr_w:        
+       case OPCODE_ret_w:
+        default:
+         error ("unknown opcode %d@pc=%d during verification", op_code, PC-1);
+         return 0;
+       }
+
+      prevpc = oldpc;
+
+      /* The following test is true if we have entered or exited an exception
+        handler range *or* we have done a store to a local variable.
+        In either case we need to consider any exception handlers that
+        might "follow" this instruction. */
+
+      if (eh_ranges != prev_eh_ranges)
+       {
+         int save_stack_pointer = stack_pointer;
+         int index = DECL_MAX_LOCALS (current_function_decl);
+         tree save_type = type_map[index];
+         tree save_current_subr = current_subr;
+         struct eh_range *ranges = find_handler (oldpc);
+         stack_pointer = 1;
+         for (; ranges != NULL_EH_RANGE;  ranges = ranges->outer)
+           {
+             tree chain = ranges->handlers;
+
+             /* We need to determine if the handler is part of current_subr.
+                The are two cases:  (1) The exception catch range
+                is entirely within current_subr.  In that case the handler
+                is also part of current_subr.
+                (2) Some of the catch range is not in current_subr.
+                In that case, the handler is *not* part of current_subr.
+
+                Figuring out which is the case is not necessarily obvious,
+                in the presence of clever code generators (and obfuscators).
+                We make a simplifying assumption that in case (2) we
+                have that the current_subr is entirely within the catch range.
+                In that case we can assume if that if a caller (the jsr) of
+                a subroutine is within the catch range, then the handler is
+                *not* part of the subroutine, and vice versa. */
+
+             current_subr = save_current_subr;
+             for ( ; current_subr != NULL_TREE;
+                   current_subr = LABEL_SUBR_CONTEXT (current_subr))
+               {
+                 tree return_labels = LABEL_RETURN_LABELS (current_subr);
+                 /* There could be multiple return_labels, but
+                    we only need to check one. */
+                 int return_pc = LABEL_PC (TREE_VALUE (return_labels));
+                 if (return_pc <= ranges->start_pc
+                     || return_pc > ranges->end_pc)
+                   break;
+               }
+
+             for ( ;  chain != NULL_TREE;  chain = TREE_CHAIN (chain))
+               {
+                 tree handler = TREE_VALUE (chain);
+                 tree type = TREE_PURPOSE (chain);
+                 if (type == NULL_TREE)  /* a finally handler */
+                   type = throwable_type_node;
+                 type_map[index] = promote_type (type);
+
+                 PUSH_PENDING (handler);
+               }
+           }
+         stack_pointer = save_stack_pointer;
+         current_subr = save_current_subr;
+         type_map[index] = save_type;
+         prev_eh_ranges = eh_ranges;
+       }
+    }
+  return 1;
+ bad_pc:
+  message = "program counter out of range";
+  goto verify_error;
+ verify_error:
+  error ("verification error at PC=%d: %s", oldpc, message);
+  return 0;
+}
diff --git a/gcc/java/zextract.c b/gcc/java/zextract.c
new file mode 100644 (file)
index 0000000..39bb06c
--- /dev/null
@@ -0,0 +1,350 @@
+/* Handle a .class file embedded in a .zip archive.
+   This extracts a member from a .zip file, but does not handle
+   uncompression (since that is not needed for classes.zip).
+
+   Copyright (C) 1996  Free Software Foundation, Inc.
+
+This program is free software; you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation; either version 2, or (at your option)
+any later version.
+
+This program is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with GNU CC; see the file COPYING.  If not, write to
+the Free Software Foundation, 59 Temple Place - Suite 330,
+Boston, MA 02111-1307, USA.  
+
+Java and all Java-based marks are trademarks or registered trademarks
+of Sun Microsystems, Inc. in the United States and other countries.
+The Free Software Foundation is independent of Sun Microsystems, Inc.  */
+
+/* Written by Per Bothner <bothner@cygnus.com>, February 1996. */
+
+#include "zipfile.h"
+
+
+/* This stuff is partly based on the 28 August 1994 public release of the
+Info-ZIP group's portable UnZip zipfile-extraction program (and related
+utilities). */
+
+#include <stdio.h>
+#ifdef __STDC__
+#include <stdlib.h>
+#endif
+#include <errno.h>       /* used in mapname() */
+
+/*************/
+/*  Defines  */
+/*************/
+
+#define UNZIP
+#define UNZIP_VERSION     20   /* compatible with PKUNZIP 2.0 */
+#define VMS_UNZIP_VERSION 42   /* if OS-needed-to-extract is VMS:  can do */
+
+
+#define ZSUFX             ".zip"
+#define CENTRAL_HDR_SIG   "\113\001\002"   /* the infamous "PK" signature */
+#define LOCAL_HDR_SIG     "\113\003\004"   /*  bytes, sans "P" (so unzip */
+#define END_CENTRAL_SIG   "\113\005\006"   /*  executable not mistaken for */
+#define EXTD_LOCAL_SIG    "\113\007\010"   /*  zipfile itself) */
+
+#define STORED            0    /* compression methods */
+#define SHRUNK            1
+#define REDUCED1          2
+#define REDUCED2          3
+#define REDUCED3          4
+#define REDUCED4          5
+#define IMPLODED          6
+#define TOKENIZED         7
+#define DEFLATED          8
+#define NUM_METHODS       9    /* index of last method + 1 */
+/* don't forget to update list_files() appropriately if NUM_METHODS changes */
+
+#define PK_OK             0    /* no error */
+#define PK_COOL           0    /* no error */
+#define PK_GNARLY         0    /* no error */
+#define PK_WARN           1    /* warning error */
+#define PK_ERR            2    /* error in zipfile */
+#define PK_BADERR         3    /* severe error in zipfile */
+#define PK_MEM            4    /* insufficient memory */
+#define PK_MEM2           5    /* insufficient memory */
+#define PK_MEM3           6    /* insufficient memory */
+#define PK_MEM4           7    /* insufficient memory */
+#define PK_MEM5           8    /* insufficient memory */
+#define PK_NOZIP          9    /* zipfile not found */
+#define PK_PARAM          10   /* bad or illegal parameters specified */
+#define PK_FIND           11   /* no files found */
+#define PK_DISK           50   /* disk full */
+#define PK_EOF            51   /* unexpected EOF */
+
+/*---------------------------------------------------------------------------
+    True sizes of the various headers, as defined by PKWARE--so it is not
+    likely that these will ever change.  But if they do, make sure both these
+    defines AND the typedefs below get updated accordingly.
+  ---------------------------------------------------------------------------*/
+#define LREC_SIZE     26    /* lengths of local file headers, central */
+#define CREC_SIZE     42    /*  directory headers, and the end-of-    */
+#define ECREC_SIZE    18    /*  central-dir record, respectively      */
+
+
+#ifndef SEEK_SET
+#  define SEEK_SET  0
+#  define SEEK_CUR  1
+#  define SEEK_END  2
+#endif
+
+/**************/
+/*  Typedefs  */
+/**************/
+
+typedef char              boolean;
+typedef unsigned char     uch;  /* code assumes unsigned bytes; these type-  */
+typedef unsigned short    ush;  /*  defs replace byte/UWORD/ULONG (which are */
+typedef unsigned long     ulg;  /*  predefined on some systems) & match zip  */
+
+/*---------------------------------------------------------------------------
+    Zipfile layout declarations.  If these headers ever change, make sure the
+    xxREC_SIZE defines (above) change with them!
+  ---------------------------------------------------------------------------*/
+
+   typedef uch   local_byte_hdr[ LREC_SIZE ];
+#      define L_VERSION_NEEDED_TO_EXTRACT_0     0
+#      define L_VERSION_NEEDED_TO_EXTRACT_1     1
+#      define L_GENERAL_PURPOSE_BIT_FLAG        2
+#      define L_COMPRESSION_METHOD              4
+#      define L_LAST_MOD_FILE_TIME              6
+#      define L_LAST_MOD_FILE_DATE              8
+#      define L_CRC32                           10
+#      define L_COMPRESSED_SIZE                 14
+#      define L_UNCOMPRESSED_SIZE               18
+#      define L_FILENAME_LENGTH                 22
+#      define L_EXTRA_FIELD_LENGTH              24
+
+  typedef uch   cdir_byte_hdr[ CREC_SIZE ];
+#      define C_VERSION_MADE_BY_0               0
+#      define C_VERSION_MADE_BY_1               1
+#      define C_VERSION_NEEDED_TO_EXTRACT_0     2
+#      define C_VERSION_NEEDED_TO_EXTRACT_1     3
+#      define C_GENERAL_PURPOSE_BIT_FLAG        4
+#      define C_COMPRESSION_METHOD              6
+#      define C_LAST_MOD_FILE_TIME              8
+#      define C_LAST_MOD_FILE_DATE              10
+#      define C_CRC32                           12
+#      define C_COMPRESSED_SIZE                 16
+#      define C_UNCOMPRESSED_SIZE               20
+#      define C_FILENAME_LENGTH                 24
+#      define C_EXTRA_FIELD_LENGTH              26
+#      define C_FILE_COMMENT_LENGTH             28
+#      define C_DISK_NUMBER_START               30
+#      define C_INTERNAL_FILE_ATTRIBUTES        32
+#      define C_EXTERNAL_FILE_ATTRIBUTES        34
+#      define C_RELATIVE_OFFSET_LOCAL_HEADER    38
+
+   typedef uch   ec_byte_rec[ ECREC_SIZE+4 ];
+/*     define SIGNATURE                         0   space-holder only */
+#      define NUMBER_THIS_DISK                  4
+#      define NUM_DISK_WITH_START_CENTRAL_DIR   6
+#      define NUM_ENTRIES_CENTRL_DIR_THS_DISK   8
+#      define TOTAL_ENTRIES_CENTRAL_DIR         10
+#      define SIZE_CENTRAL_DIRECTORY            12
+#      define OFFSET_START_CENTRAL_DIRECTORY    16
+#      define ZIPFILE_COMMENT_LENGTH            20
+
+
+   typedef struct local_file_header {                 /* LOCAL */
+       uch version_needed_to_extract[2];
+       ush general_purpose_bit_flag;
+       ush compression_method;
+       ush last_mod_file_time;
+       ush last_mod_file_date;
+       ulg crc32;
+       ulg csize;
+       ulg ucsize;
+       ush filename_length;
+       ush extra_field_length;
+   } local_file_hdr;
+
+   typedef struct central_directory_file_header {     /* CENTRAL */
+       uch version_made_by[2];
+       uch version_needed_to_extract[2];
+       ush general_purpose_bit_flag;
+       ush compression_method;
+       ush last_mod_file_time;
+       ush last_mod_file_date;
+       ulg crc32;
+       ulg csize;
+       ulg ucsize;
+       ush filename_length;
+       ush extra_field_length;
+       ush file_comment_length;
+       ush disk_number_start;
+       ush internal_file_attributes;
+       ulg external_file_attributes;
+       ulg relative_offset_local_header;
+   } cdir_file_hdr;
+
+   typedef struct end_central_dir_record {            /* END CENTRAL */
+       ush number_this_disk;
+       ush num_disk_with_start_central_dir;
+       ush num_entries_centrl_dir_ths_disk;
+       ush total_entries_central_dir;
+       ulg size_central_directory;
+       ulg offset_start_central_directory;
+       ush zipfile_comment_length;
+   } ecdir_rec;
+
+
+/************/
+/*  Macros  */
+/************/
+
+#ifndef MAX
+#  define MAX(a,b)   ((a) > (b) ? (a) : (b))
+#endif
+#ifndef MIN
+#  define MIN(a,b)   ((a) < (b) ? (a) : (b))
+#endif
+
+
+/***********************/
+/* Function makeword() */
+/***********************/
+
+static ush makeword(b)
+    uch *b;
+{
+    /*
+     * Convert Intel style 'short' integer to non-Intel non-16-bit
+     * host format.  This routine also takes care of byte-ordering.
+     */
+    return (ush)((b[1] << 8) | b[0]);
+}
+
+
+/***********************/
+/* Function makelong() */
+/***********************/
+
+static ulg makelong(sig)
+    uch *sig;
+{
+    /*
+     * Convert intel style 'long' variable to non-Intel non-16-bit
+     * host format.  This routine also takes care of byte-ordering.
+     */
+    return (((ulg)sig[3]) << 24)
+        + (((ulg)sig[2]) << 16)
+        + (((ulg)sig[1]) << 8)
+        + ((ulg)sig[0]);
+}
+
+int
+read_zip_archive (zipf)
+     register ZipFile *zipf;
+{
+  int i;
+  int dir_last_pad;
+  char *dir_ptr;
+  char buffer[100];
+
+  zipf->size = lseek (zipf->fd, 0L, SEEK_END);
+
+  if (zipf->size < (ECREC_SIZE+4) || lseek (zipf->fd, (long)(-(ECREC_SIZE+4)), SEEK_CUR) <= 0)
+    return -1;
+  if (read (zipf->fd, buffer, ECREC_SIZE+4) != ECREC_SIZE+4)
+    return -2;
+  zipf->count = makeword(&buffer[TOTAL_ENTRIES_CENTRAL_DIR]);
+  zipf->dir_size = makelong(&buffer[SIZE_CENTRAL_DIRECTORY]);
+#define ALLOC malloc
+  /* Allocate 1 more to allow appending '\0' to last filename. */
+  zipf->central_directory = ALLOC (zipf->dir_size+1);
+  if (lseek (zipf->fd, -(zipf->dir_size+ECREC_SIZE+4), SEEK_CUR) < 0)
+    return -2;
+  if (read (zipf->fd, zipf->central_directory, zipf->dir_size) < 0)
+    return -2;
+
+#ifdef TEST
+  printf ("number_this_disk = %d\n", makeword(&buffer[NUMBER_THIS_DISK]));
+  printf ("num_disk_with_start_central_dir = %d\n", makeword(&buffer[NUM_DISK_WITH_START_CENTRAL_DIR]));
+
+  printf ("num_entries_centrl_dir_ths_disk = %d\n",
+        makeword(&buffer[NUM_ENTRIES_CENTRL_DIR_THS_DISK]));
+  printf ("total_entries_central_dir = %d\n",
+        makeword(&buffer[TOTAL_ENTRIES_CENTRAL_DIR]));
+  printf ("size_central_directory = %d\n",
+        makelong(&buffer[SIZE_CENTRAL_DIRECTORY]));
+  printf ("offset_start_central_directory = %d\n",
+        makelong(&buffer[OFFSET_START_CENTRAL_DIRECTORY]));
+  printf ("zipfile_comment_length = %d\n",
+        makeword(&buffer[ZIPFILE_COMMENT_LENGTH]));
+#endif
+
+  dir_last_pad = 0;
+  dir_ptr = zipf->central_directory;
+  for (i = 0; i < zipf->count; i++)
+    {
+      ZipDirectory *zipd = (ZipDirectory*)(dir_ptr + dir_last_pad);
+      long uncompressed_size = makelong (&dir_ptr[4+C_UNCOMPRESSED_SIZE]);
+      long filename_length = makeword (&dir_ptr[4+C_FILENAME_LENGTH]);
+      long extra_field_length = makeword (&dir_ptr[4+C_EXTRA_FIELD_LENGTH]);
+      long file_comment_length = makeword (&dir_ptr[4+C_FILE_COMMENT_LENGTH]);
+      int unpadded_direntry_length;
+      if ((dir_ptr-zipf->central_directory)+filename_length+CREC_SIZE+4>zipf->dir_size)
+       return -1;
+
+      zipd->filename_length = filename_length;
+      zipd->size = uncompressed_size;
+#ifdef __GNUC__
+#define DIR_ALIGN __alignof__(ZipDirectory)
+#else
+#define DIR_ALIGN sizeof(long)
+#endif
+      zipd->filestart = makelong (&dir_ptr[4+C_RELATIVE_OFFSET_LOCAL_HEADER])
+         + (LREC_SIZE+4) + filename_length + file_comment_length +
+         + (extra_field_length ? extra_field_length+4 : 0);
+      /* About the last term of the expression above. Should the same
+        apply if file_comment_length is not zero ?  I've never seen
+        the comment field uses so far. FIXME.  */
+      zipd->filename_offset = CREC_SIZE+4 - dir_last_pad;
+      unpadded_direntry_length 
+         = zipd->filename_offset + zipd->filename_length + extra_field_length;
+      zipd->direntry_size =
+       ((unpadded_direntry_length + DIR_ALIGN) / DIR_ALIGN) * DIR_ALIGN;
+      dir_last_pad = zipd->direntry_size - unpadded_direntry_length;
+      dir_ptr = (char*)zipd + unpadded_direntry_length;
+      *dir_ptr = '\0';
+    }
+  return 0;
+}
+
+#ifdef TEST
+main ()
+{
+  ZipFile zipf[1];
+  ZipDirectory *zipd;
+  int i;
+
+  zipf->fd = 0;
+
+  i = read_zip_archive (zipf);
+  if (i)
+    {
+      fprintf (stderr, "Bad zip file.\n");
+      exit (i);
+    }
+
+  zipd = (ZipDirectory*) zipf->central_directory;
+  for (i = 0; i < zipf->count; i++, zipd = ZIPDIR_NEXT (zipd))
+    {
+      printf ("%d: size:%d, name(#%d)%s, offset:%d\n",
+             i, zipd->size, zipd->filename_length,
+             ZIPDIR_FILENAME (zipd),
+             zipd->filestart);
+    }
+}
+#endif
diff --git a/gcc/java/zipfile.h b/gcc/java/zipfile.h
new file mode 100644 (file)
index 0000000..5d71184
--- /dev/null
@@ -0,0 +1,58 @@
+/* Definitions for using a zipped' archive.
+
+   Copyright (C) 1996  Free Software Foundation, Inc.
+
+This program is free software; you can redistribute it and/or modify
+it under the terms of the GNU General Public License as published by
+the Free Software Foundation; either version 2, or (at your option)
+any later version.
+
+This program is distributed in the hope that it will be useful,
+but WITHOUT ANY WARRANTY; without even the implied warranty of
+MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
+GNU General Public License for more details.
+
+You should have received a copy of the GNU General Public License
+along with GNU CC; see the file COPYING.  If not, write to
+the Free Software Foundation, 59 Temple Place - Suite 330,
+Boston, MA 02111-1307, USA.  
+
+Java and all Java-based marks are trademarks or registered trademarks
+of Sun Microsystems, Inc. in the United States and other countries.
+The Free Software Foundation is independent of Sun Microsystems, Inc.  */
+
+struct ZipFile {
+  int fd;
+  long size;
+  long count;
+  long dir_size;
+  char *central_directory;
+};
+
+typedef struct ZipFile ZipFile;
+
+struct ZipDirectory {
+  int direntry_size;
+  int filename_offset;
+  long size; /* length of file */
+  long filestart;  /* start of file in archive */
+  long filename_length;
+  /* char mid_padding[...]; */
+  /* char filename[filename_length]; */
+  /* char end_padding[...]; */
+};
+
+typedef struct ZipDirectory ZipDirectory;
+
+struct ZipFileCache {
+  struct ZipFile z;
+  struct ZipFileCache *next;
+  char *name;
+};
+
+extern struct ZipFileCache *SeenZipFiles;
+
+#define ZIPDIR_FILENAME(ZIPD) ((char*)(ZIPD)+(ZIPD)->filename_offset)
+#define ZIPDIR_NEXT(ZIPD) \
+   ((ZipDirectory*)((char*)(ZIPD)+(ZIPD)->direntry_size))
+#define ZIPMAGIC 0x504b0304