* gjavah.c (print_stub_or_jni): Mark functions only JNIEXPORT, not
[official-gcc.git] / gcc / java / gjavah.c
blob19958f3832a33d1d666908c2ea550915ed189acd
1 /* Program to write C++-suitable header files from a Java(TM) .class
2 file. This is similar to SUN's javah.
4 Copyright (C) 1996, 1998, 1999, 2000, 2001, 2002, 2003, 2004
5 Free Software Foundation, Inc.
7 This file is part of GCC.
9 GCC is free software; you can redistribute it and/or modify
10 it under the terms of the GNU General Public License as published by
11 the Free Software Foundation; either version 2, or (at your option)
12 any later version.
14 GCC is distributed in the hope that it will be useful,
15 but WITHOUT ANY WARRANTY; without even the implied warranty of
16 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
17 GNU General Public License for more details.
19 You should have received a copy of the GNU General Public License
20 along with GCC; see the file COPYING. If not, write to
21 the Free Software Foundation, 59 Temple Place - Suite 330,
22 Boston, MA 02111-1307, USA.
24 Java and all Java-based marks are trademarks or registered trademarks
25 of Sun Microsystems, Inc. in the United States and other countries.
26 The Free Software Foundation is independent of Sun Microsystems, Inc. */
28 /* Written by Per Bothner <bothner@cygnus.com>, February 1996. */
30 #include "config.h"
31 #include "system.h"
32 #include "coretypes.h"
33 #include "tm.h"
34 #include <math.h>
36 #include "jcf.h"
37 #include "tree.h"
38 #include "javaop.h"
39 #include "java-tree.h"
40 #include "java-opcodes.h"
41 #include "ggc.h"
42 #include "hashtab.h"
43 #include "intl.h"
45 #include <getopt.h>
49 /* The output file. */
50 FILE *out = NULL;
52 /* Nonzero on failure. */
53 static int found_error = 0;
55 /* Nonzero if we're generating JNI output. */
56 int flag_jni = 0;
58 /* When nonzero, warn when source file is newer than matching class
59 file. */
60 int flag_newer = 1;
62 /* Directory to place resulting files in. Set by -d option. */
63 const char *output_directory = "";
65 /* Directory to place temporary file. Set by -td option. Currently unused. */
66 const char *temp_directory = "/tmp";
68 /* Number of friend functions we have to declare. */
69 static int friend_count;
71 /* A class can optionally have a `friend' function declared. If
72 non-NULL, this is that function. */
73 static char **friend_specs = NULL;
75 /* Number of lines we are prepending before the class. */
76 static int prepend_count;
78 /* We can prepend extra lines before the class's start. */
79 static char **prepend_specs = NULL;
81 /* Number of lines we are appending at the end of the class. */
82 static int add_count;
84 /* We can append extra lines just before the class's end. */
85 static char **add_specs = NULL;
87 /* Number of lines we are appending after the class. */
88 static int append_count;
90 /* We can append extra lines after the class's end. */
91 static char **append_specs = NULL;
93 int verbose = 0;
95 int stubs = 0;
97 struct JCF *current_jcf;
99 /* This holds access information for the last field we examined. They
100 let us generate "private:", "public:", and "protected:" properly.
101 If 0 then we haven't previously examined any field. */
102 static JCF_u2 last_access;
104 /* Pass this macro the flags for a class and for a method. It will
105 return true if the method should be considered `final'. */
106 #define METHOD_IS_FINAL(Class, Method) \
107 (((Class) & ACC_FINAL) || ((Method) & (ACC_FINAL | ACC_PRIVATE)))
109 /* Pass this macro the flags for a method. It will return true if the
110 method is native. */
111 #define METHOD_IS_NATIVE(Method) \
112 ((Method) & ACC_NATIVE)
114 /* We keep a linked list of all method names we have seen. This lets
115 us determine if a method name and a field name are in conflict. */
116 struct method_name
118 unsigned char *name;
119 int length;
120 unsigned char *signature;
121 int sig_length;
122 int is_native;
123 struct method_name *next;
126 /* List of method names we've seen. */
127 static struct method_name *method_name_list;
129 static void print_field_info (FILE*, JCF*, int, int, JCF_u2);
130 static void print_mangled_classname (FILE*, JCF*, const char*, int);
131 static int print_cxx_classname (FILE*, const char*, JCF*, int, int);
132 static void print_method_info (FILE*, JCF*, int, int, JCF_u2);
133 static void print_c_decl (FILE*, JCF*, int, int, int, const char *, int);
134 static void print_stub_or_jni (FILE*, JCF*, int, int, int, const char *, int);
135 static void print_full_cxx_name (FILE*, JCF*, int, int, int, const char *, int);
136 static void decompile_method (FILE*, JCF*, int);
137 static void add_class_decl (FILE*, JCF*, JCF_u2);
139 static void print_name (FILE *, JCF *, int);
140 static void print_base_classname (FILE *, JCF *, int);
141 static int utf8_cmp (const unsigned char *, int, const char *);
142 static char *cxx_keyword_subst (const unsigned char *, int);
143 static void generate_access (FILE *, JCF_u2);
144 static int name_is_method_p (const unsigned char *, int);
145 static char *get_field_name (JCF *, int, JCF_u2);
146 static void print_field_name (FILE *, JCF *, int, JCF_u2);
147 static const unsigned char *super_class_name (JCF *, int *);
148 static void print_include (FILE *, const unsigned char *, int);
149 static int gcjh_streq (const void *p1, const void *p2);
150 static int throwable_p (const unsigned char *signature);
151 static const unsigned char *
152 decode_signature_piece (FILE *, const unsigned char *,
153 const unsigned char *, int *);
154 static void print_class_decls (FILE *, JCF *, int);
155 static void error (const char *msgid, ...);
156 static void usage (void) ATTRIBUTE_NORETURN;
157 static void help (void) ATTRIBUTE_NORETURN;
158 static void version (void) ATTRIBUTE_NORETURN;
159 static int overloaded_jni_method_exists_p (const unsigned char *, int,
160 const char *, int);
161 static void jni_print_char (FILE *, int);
162 static void jni_print_float (FILE *, jfloat);
163 static void jni_print_double (FILE *, jdouble);
164 static void decompile_return_statement (FILE *, JCF *, int, int, int);
166 static void handle_inner_classes (int);
168 JCF_u2 current_field_name;
169 JCF_u2 current_field_value;
170 JCF_u2 current_field_signature;
171 JCF_u2 current_field_flags;
173 #define HANDLE_START_FIELD(ACCESS_FLAGS, NAME, SIGNATURE, ATTRIBUTE_COUNT) \
174 ( current_field_name = (NAME), current_field_signature = (SIGNATURE), \
175 current_field_flags = (ACCESS_FLAGS), current_field_value = 0)
177 /* We pass over fields twice. The first time we just note the types
178 of the fields and then the start of the methods. Then we go back
179 and parse the fields for real. This is ugly. */
180 static int field_pass;
181 /* Likewise we pass over methods twice. The first time we generate
182 class decl information; the second time we generate actual method
183 decls. */
184 static int method_pass;
186 #define HANDLE_END_FIELD() \
187 if (field_pass) \
189 if (out && ! stubs && ! flag_jni) \
190 print_field_info (out, jcf, current_field_name, \
191 current_field_signature, \
192 current_field_flags); \
194 else if (! stubs && ! flag_jni) \
195 add_class_decl (out, jcf, current_field_signature);
197 #define HANDLE_CONSTANTVALUE(VALUEINDEX) current_field_value = (VALUEINDEX)
199 static int method_declared = 0;
200 static int method_access = 0;
201 static int method_printed = 0;
202 static int method_synthetic = 0;
203 static int method_signature = 0;
205 /* Set to 1 while the very first data member of a class is being handled. */
206 static int is_first_data_member = 0;
208 #define HANDLE_METHOD(ACCESS_FLAGS, NAME, SIGNATURE, ATTRIBUTE_COUNT) \
210 method_synthetic = 0; \
211 method_printed = 0; \
212 decompiled = 0; \
213 method_signature = SIGNATURE; \
214 if (ATTRIBUTE_COUNT) \
215 method_synthetic = peek_attribute (jcf, ATTRIBUTE_COUNT, \
216 (const char *)"Synthetic", 9); \
217 /* If a synthetic methods have been declared, its attribute aren't \
218 worth reading (and triggering side-effects). We skip them an \
219 set ATTRIBUTE_COUNT to zero so that they'll be skipped in \
220 jcf_parse_one_method. */ \
221 if (method_synthetic) \
223 skip_attribute (jcf, ATTRIBUTE_COUNT); \
224 ATTRIBUTE_COUNT = 0; \
226 if (method_pass && !method_synthetic) \
228 if (out) \
229 print_method_info (out, jcf, NAME, SIGNATURE, \
230 ACCESS_FLAGS); \
232 else if (!method_synthetic) \
234 print_method_info (NULL, jcf, NAME, SIGNATURE, \
235 ACCESS_FLAGS); \
236 if (! stubs && ! flag_jni) \
237 add_class_decl (out, jcf, SIGNATURE); \
241 #define HANDLE_CODE_ATTRIBUTE(MAX_STACK, MAX_LOCALS, CODE_LENGTH) \
242 if (out && method_declared) decompile_method (out, jcf, CODE_LENGTH);
244 static int decompiled = 0;
245 #define HANDLE_END_METHOD() \
246 if (out && method_printed && !method_synthetic) \
247 fputs (decompiled || stubs ? "\n" : ";\n", out);
249 #define HANDLE_INNERCLASSES_ATTRIBUTE(COUNT) handle_inner_classes (COUNT)
251 /* We're going to need {peek,skip}_attribute, enable their definition. */
252 #define NEED_PEEK_ATTRIBUTE
253 #define NEED_SKIP_ATTRIBUTE
255 #include "jcf-reader.c"
257 /* Print an error message and set found_error. */
258 static void
259 error (const char *msgid, ...)
261 va_list ap;
263 va_start (ap, msgid);
265 fprintf (stderr, "gcjh: ");
266 vfprintf (stderr, _(msgid), ap);
267 va_end (ap);
268 fprintf (stderr, "\n");
269 found_error = 1;
272 /* Print a single-precision float, suitable for parsing by g++. */
273 static void
274 jni_print_float (FILE *stream, jfloat f)
276 /* It'd be nice to use __builtin_nan/__builtin_inf here but they don't
277 work in data initializers. FIXME. */
278 if (JFLOAT_FINITE (f))
280 fputs (" = ", stream);
281 if (f.negative)
282 putc ('-', stream);
283 if (f.exponent)
284 fprintf (stream, "0x1.%.6xp%+df",
285 ((unsigned int)f.mantissa) << 1,
286 f.exponent - JFLOAT_EXP_BIAS);
287 else
288 /* Exponent of 0x01 is -125; exponent of 0x00 is *also* -125,
289 because the implicit leading 1 bit is no longer present. */
290 fprintf (stream, "0x0.%.6xp%+df",
291 ((unsigned int)f.mantissa) << 1,
292 f.exponent + 1 - JFLOAT_EXP_BIAS);
294 fputs (";\n", stream);
297 /* Print a double-precision float, suitable for parsing by g++. */
298 static void
299 jni_print_double (FILE *stream, jdouble f)
301 /* It'd be nice to use __builtin_nan/__builtin_inf here but they don't
302 work in data initializers. FIXME. */
303 if (JDOUBLE_FINITE (f))
305 fputs (" = ", stream);
306 if (f.negative)
307 putc ('-', stream);
308 if (f.exponent)
309 fprintf (stream, "0x1.%.5x%.8xp%+d",
310 f.mantissa0, f.mantissa1,
311 f.exponent - JDOUBLE_EXP_BIAS);
312 else
313 /* Exponent of 0x001 is -1022; exponent of 0x000 is *also* -1022,
314 because the implicit leading 1 bit is no longer present. */
315 fprintf (stream, "0x0.%.5x%.8xp%+d",
316 f.mantissa0, f.mantissa1,
317 f.exponent + 1 - JDOUBLE_EXP_BIAS);
319 fputs (";\n", stream);
322 /* Print a character, appropriately mangled for JNI. */
324 static void
325 jni_print_char (FILE *stream, int ch)
327 if (! flag_jni)
328 jcf_print_char (stream, ch);
329 else if (ch == '(' || ch == ')')
331 /* Ignore. */
333 else if (ch == '_')
334 fputs ("_1", stream);
335 else if (ch == ';')
336 fputs ("_2", stream);
337 else if (ch == '[')
338 fputs ("_3", stream);
339 else if (ch == '/')
340 fputs ("_", stream);
341 else if (ISALNUM (ch))
342 fputc (ch, stream);
343 else
345 /* "Unicode" character. */
346 fprintf (stream, "_0%04x", ch);
350 /* Print a name from the class data. If the index does not point to a
351 string, an error results. */
353 static void
354 print_name (FILE* stream, JCF* jcf, int name_index)
356 if (JPOOL_TAG (jcf, name_index) != CONSTANT_Utf8)
358 fprintf (stream, "<not a UTF8 constant>");
359 found_error = 1;
361 else if (! flag_jni)
362 jcf_print_utf8 (stream, JPOOL_UTF_DATA (jcf, name_index),
363 JPOOL_UTF_LENGTH (jcf, name_index));
364 else
366 /* For JNI we must correctly quote each character. */
367 const unsigned char *str = JPOOL_UTF_DATA (jcf, name_index);
368 int length = JPOOL_UTF_LENGTH (jcf, name_index);
369 const unsigned char *limit = str + length;
370 while (str < limit)
372 int ch = UTF8_GET (str, limit);
373 if (ch < 0)
375 fprintf (stream, "\\<invalid>");
376 return;
378 jni_print_char (stream, ch);
383 /* Print base name of class. The base name is everything after the
384 final separator. */
386 static void
387 print_base_classname (FILE *stream, JCF *jcf, int index)
389 int name_index = JPOOL_USHORT1 (jcf, index);
390 int len;
391 const unsigned char *s, *p, *limit;
393 s = JPOOL_UTF_DATA (jcf, name_index);
394 len = JPOOL_UTF_LENGTH (jcf, name_index);
395 limit = s + len;
396 p = s;
397 while (s < limit)
399 int c = UTF8_GET (s, limit);
400 if (c == '/')
401 p = s;
404 while (p < limit)
406 int ch = UTF8_GET (p, limit);
407 if (ch == '/')
408 fputs ("::", stream);
409 else
410 jcf_print_char (stream, ch);
414 /* Return 0 if NAME is equal to STR, -1 if STR is "less" than NAME,
415 and 1 if STR is "greater" than NAME. */
417 static int
418 utf8_cmp (const unsigned char *str, int length, const char *name)
420 const unsigned char *limit = str + length;
421 int i;
423 for (i = 0; name[i]; ++i)
425 int ch = UTF8_GET (str, limit);
426 if (ch != name[i])
427 return ch - name[i];
430 return str == limit ? 0 : 1;
433 /* This is a sorted list of all C++ keywords. */
435 static const char *const cxx_keywords[] =
437 "_Complex",
438 "__alignof",
439 "__alignof__",
440 "__asm",
441 "__asm__",
442 "__attribute",
443 "__attribute__",
444 "__builtin_va_arg",
445 "__complex",
446 "__complex__",
447 "__const",
448 "__const__",
449 "__extension__",
450 "__imag",
451 "__imag__",
452 "__inline",
453 "__inline__",
454 "__label__",
455 "__null",
456 "__real",
457 "__real__",
458 "__restrict",
459 "__restrict__",
460 "__signed",
461 "__signed__",
462 "__typeof",
463 "__typeof__",
464 "__volatile",
465 "__volatile__",
466 "and",
467 "and_eq",
468 "asm",
469 "auto",
470 "bitand",
471 "bitor",
472 "bool",
473 "break",
474 "case",
475 "catch",
476 "char",
477 "class",
478 "compl",
479 "const",
480 "const_cast",
481 "continue",
482 "default",
483 "delete",
484 "do",
485 "double",
486 "dynamic_cast",
487 "else",
488 "enum",
489 "explicit",
490 "export",
491 "extern",
492 "false",
493 "float",
494 "for",
495 "friend",
496 "goto",
497 "if",
498 "inline",
499 "int",
500 "long",
501 "mutable",
502 "namespace",
503 "new",
504 "not",
505 "not_eq",
506 "operator",
507 "or",
508 "or_eq",
509 "private",
510 "protected",
511 "public",
512 "register",
513 "reinterpret_cast",
514 "return",
515 "short",
516 "signed",
517 "sizeof",
518 "static",
519 "static_cast",
520 "struct",
521 "switch",
522 "template",
523 "this",
524 "throw",
525 "true",
526 "try",
527 "typedef",
528 "typeid",
529 "typename",
530 "typeof",
531 "union",
532 "unsigned",
533 "using",
534 "virtual",
535 "void",
536 "volatile",
537 "wchar_t",
538 "while",
539 "xor",
540 "xor_eq"
544 /* If NAME is the name of a C++ keyword, then return an override name.
545 This is a name that can be used in place of the keyword.
546 Otherwise, return NULL. The return value is malloc()d. */
548 static char *
549 cxx_keyword_subst (const unsigned char *str, int length)
551 int last = ARRAY_SIZE (cxx_keywords);
552 int first = 0;
553 int mid = (last + first) / 2;
554 int old = -1;
556 for (mid = (last + first) / 2;
557 mid != old;
558 old = mid, mid = (last + first) / 2)
560 int kwl = strlen (cxx_keywords[mid]);
561 int min_length = kwl > length ? length : kwl;
562 int r = utf8_cmp (str, min_length, cxx_keywords[mid]);
564 if (r == 0)
566 int i;
568 /* Skip all trailing `$'. */
569 for (i = min_length; i < length && str[i] == '$'; ++i)
571 /* We've only found a match if all the remaining characters
572 are `$'. */
573 if (i == length)
575 char *dup = xmalloc (2 + length - min_length + kwl);
576 strcpy (dup, cxx_keywords[mid]);
577 for (i = kwl; i < length + 1; ++i)
578 dup[i] = '$';
579 dup[i] = '\0';
580 return dup;
582 r = 1;
585 if (r < 0)
586 last = mid;
587 else
588 first = mid;
590 return NULL;
593 /* Generate an access control keyword based on FLAGS. */
595 static void
596 generate_access (FILE *stream, JCF_u2 flags)
598 if ((flags & ACC_VISIBILITY) == last_access)
599 return;
600 last_access = (flags & ACC_VISIBILITY);
602 switch (last_access)
604 case 0:
605 fputs ("public: // actually package-private\n", stream);
606 break;
607 case ACC_PUBLIC:
608 fputs ("public:\n", stream);
609 break;
610 case ACC_PRIVATE:
611 fputs ("private:\n", stream);
612 break;
613 case ACC_PROTECTED:
614 fputs ("public: // actually protected\n", stream);
615 break;
616 default:
617 found_error = 1;
618 fprintf (stream, "#error unrecognized visibility %d\n",
619 (flags & ACC_VISIBILITY));
620 break;
624 /* See if NAME is already the name of a method. */
625 static int
626 name_is_method_p (const unsigned char *name, int length)
628 struct method_name *p;
630 for (p = method_name_list; p != NULL; p = p->next)
632 if (p->length == length && ! memcmp (p->name, name, length))
633 return 1;
635 return 0;
638 /* If there is already a native method named NAME, whose signature is not
639 SIGNATURE, then return true. Otherwise return false. */
640 static int
641 overloaded_jni_method_exists_p (const unsigned char *name, int length,
642 const char *signature, int sig_length)
644 struct method_name *p;
646 for (p = method_name_list; p != NULL; p = p->next)
648 if (p->is_native
649 && p->length == length
650 && ! memcmp (p->name, name, length)
651 && (p->sig_length != sig_length
652 || memcmp (p->signature, signature, sig_length)))
653 return 1;
655 return 0;
658 /* Get name of a field. This handles renamings due to C++ clash. */
659 static char *
660 get_field_name (JCF *jcf, int name_index, JCF_u2 flags)
662 unsigned char *name = JPOOL_UTF_DATA (jcf, name_index);
663 int length = JPOOL_UTF_LENGTH (jcf, name_index);
664 char *override;
666 if (name_is_method_p (name, length))
668 /* This field name matches a method. So override the name with
669 a dummy name. This is yucky, but it isn't clear what else to
670 do. FIXME: if the field is static, then we'll be in real
671 trouble. */
672 if ((flags & ACC_STATIC))
674 error ("static field has same name as method");
675 return NULL;
678 override = xmalloc (length + 3);
679 memcpy (override, name, length);
680 strcpy (override + length, "__");
682 else
683 override = cxx_keyword_subst (name, length);
685 return override;
688 /* Print a field name. Convenience function for use with
689 get_field_name. */
690 static void
691 print_field_name (FILE *stream, JCF *jcf, int name_index, JCF_u2 flags)
693 char *override = get_field_name (jcf, name_index, flags);
695 if (override)
697 fputs (override, stream);
698 free (override);
700 else
701 jcf_print_utf8 (stream, JPOOL_UTF_DATA (jcf, name_index),
702 JPOOL_UTF_LENGTH (jcf, name_index));
705 static void
706 print_field_info (FILE *stream, JCF* jcf, int name_index, int sig_index,
707 JCF_u2 flags)
709 char *override = NULL;
711 generate_access (stream, flags);
712 if (JPOOL_TAG (jcf, name_index) != CONSTANT_Utf8)
714 fprintf (stream, "<not a UTF8 constant>");
715 found_error = 1;
716 return;
719 fputs (" ", out);
720 if ((flags & ACC_STATIC))
721 fputs ("static ", out);
723 if ((flags & ACC_FINAL))
725 if (current_field_value > 0)
727 char buffer[25];
728 int done = 1;
730 switch (JPOOL_TAG (jcf, current_field_value))
732 case CONSTANT_Integer:
734 jint num;
735 int most_negative = 0;
736 fputs ("const jint ", out);
737 print_field_name (out, jcf, name_index, 0);
738 fputs (" = ", out);
739 num = JPOOL_INT (jcf, current_field_value);
740 /* We single out the most negative number to print
741 specially. This avoids later warnings from g++. */
742 if (num == (jint) 0x80000000)
744 most_negative = 1;
745 ++num;
747 format_int (buffer, (jlong) num, 10);
748 fprintf (out, "%sL%s;\n", buffer, most_negative ? " - 1" : "");
750 break;
751 case CONSTANT_Long:
753 jlong num;
754 int most_negative = 0;
755 fputs ("const jlong ", out);
756 print_field_name (out, jcf, name_index, 0);
757 fputs (" = ", out);
758 num = JPOOL_LONG (jcf, current_field_value);
759 /* We single out the most negative number to print
760 specially.. This avoids later warnings from g++. */
761 if (num == (jlong) 0x8000000000000000LL)
763 most_negative = 1;
764 ++num;
766 format_int (buffer, num, 10);
767 fprintf (out, "%sLL%s;\n", buffer, most_negative ? " - 1" :"");
769 break;
770 case CONSTANT_Float:
772 jfloat fnum = JPOOL_FLOAT (jcf, current_field_value);
773 fputs ("const jfloat ", out);
774 print_field_name (out, jcf, name_index, 0);
775 jni_print_float (out, fnum);
777 break;
778 case CONSTANT_Double:
780 jdouble dnum = JPOOL_DOUBLE (jcf, current_field_value);
781 fputs ("const jdouble ", out);
782 print_field_name (out, jcf, name_index, 0);
783 jni_print_double (out, dnum);
785 break;
786 default:
787 /* We can't print this as a constant, but we can still
788 print something sensible. */
789 done = 0;
790 break;
793 if (done)
794 return;
798 override = get_field_name (jcf, name_index, flags);
799 print_c_decl (out, jcf, name_index, sig_index, 0, override, flags);
800 fputs (";\n", out);
802 if (override)
803 free (override);
807 static void
808 print_method_info (FILE *stream, JCF* jcf, int name_index, int sig_index,
809 JCF_u2 flags)
811 const unsigned char *str;
812 int length, is_init = 0;
813 char *override = NULL;
815 method_declared = 0;
816 method_access = flags;
817 if (stream && JPOOL_TAG (jcf, name_index) != CONSTANT_Utf8)
818 fprintf (stream, "<not a UTF8 constant>");
819 str = JPOOL_UTF_DATA (jcf, name_index);
820 length = JPOOL_UTF_LENGTH (jcf, name_index);
822 if (str[0] == '<')
824 /* Ignore the internally generated method <clinit>. However,
825 treat <init> as a constructor. */
826 if (! utf8_cmp (str, length, "<init>"))
827 is_init = 1;
828 else if (! METHOD_IS_FINAL (jcf->access_flags, flags)
829 && ! (flags & ACC_STATIC))
831 /* FIXME: i18n bug here. Order of prints should not be
832 fixed. */
833 fprintf (stderr, _("ignored method `"));
834 jcf_print_utf8 (stderr, str, length);
835 fprintf (stderr, _("' marked virtual\n"));
836 found_error = 1;
837 return;
839 else
840 return;
843 /* During the first method pass, build a list of method names. This will
844 be used to determine if field names conflict with method names. */
845 if (! stream)
847 struct method_name *nn;
849 nn = xmalloc (sizeof (struct method_name));
850 nn->name = xmalloc (length);
851 memcpy (nn->name, str, length);
852 nn->length = length;
853 nn->next = method_name_list;
854 nn->sig_length = JPOOL_UTF_LENGTH (jcf, sig_index);
855 nn->signature = xmalloc (nn->sig_length);
856 nn->is_native = METHOD_IS_NATIVE (flags);
857 memcpy (nn->signature, JPOOL_UTF_DATA (jcf, sig_index),
858 nn->sig_length);
859 method_name_list = nn;
861 /* The rest of this function doesn't matter. */
862 return;
865 /* We don't worry about overrides in JNI mode. */
866 if (! flag_jni)
868 /* We can't generate a method whose name is a C++ reserved word.
869 We can't just ignore the function, because that will cause
870 incorrect code to be generated if the function is virtual
871 (not only for calls to this function for for other functions
872 after it in the vtbl). So we give it a dummy name instead. */
873 override = cxx_keyword_subst (str, length);
876 if (! stubs && ! flag_jni)
878 method_printed = 1;
880 generate_access (stream, flags);
882 fputs (" ", out);
883 if ((flags & ACC_STATIC))
884 fputs ("static ", out);
885 else if (! METHOD_IS_FINAL (jcf->access_flags, flags))
887 /* Don't print `virtual' if we have a constructor. */
888 if (! is_init)
889 fputs ("virtual ", out);
891 print_c_decl (out, jcf, name_index, sig_index, is_init, override, flags);
893 if ((flags & ACC_ABSTRACT))
894 fputs (" = 0", out);
895 else
896 method_declared = 1;
898 else
900 if (METHOD_IS_NATIVE (flags))
902 method_printed = 1;
903 print_stub_or_jni (out, jcf, name_index, sig_index,
904 is_init, override, flags);
908 if (override)
909 free (override);
912 /* A helper for the decompiler which prints a `return' statement where
913 the type is a reference type. If METHODTYPE and OBJECTTYPE are not
914 identical, we emit a cast. We do this because the C++ compiler
915 doesn't know that a reference can be cast to the type of an
916 interface it implements. METHODTYPE is the index of the method's
917 signature. NAMEINDEX is the index of the field name; -1 for
918 `this'. OBJECTTYPE is the index of the object's type. */
919 static void
920 decompile_return_statement (FILE *out, JCF *jcf, int methodtype,
921 int nameindex, int objecttype)
923 int cast = 0;
924 int obj_name_len, method_name_len;
925 const unsigned char *obj_data, *method_data;
927 obj_name_len = JPOOL_UTF_LENGTH (jcf, objecttype);
928 obj_data = JPOOL_UTF_DATA (jcf, objecttype);
930 method_name_len = JPOOL_UTF_LENGTH (jcf, methodtype);
931 method_data = JPOOL_UTF_DATA (jcf, methodtype);
933 /* Skip forward to return type part of method. */
934 while (*method_data != ')')
936 ++method_data;
937 --method_name_len;
939 /* Skip past `)'. */
940 ++method_data;
941 --method_name_len;
943 /* If we see an `L', skip it and the trailing `;'. */
944 if (method_data[0] == 'L' && method_data[method_name_len - 1] == ';')
946 ++method_data;
947 method_name_len -= 2;
949 if (obj_data[0] == 'L' && obj_data[obj_name_len - 1] == ';')
951 ++obj_data;
952 obj_name_len -= 2;
955 /* FIXME: if METHODTYPE is a superclass of OBJECTTYPE then we don't
956 need a cast. Right now there is no way to determine if this is
957 the case. */
958 if (method_name_len != obj_name_len)
959 cast = 1;
960 else
962 int i;
963 for (i = 0; i < method_name_len; ++i)
965 if (method_data[i] != obj_data[i])
967 cast = 1;
968 break;
973 fputs (" { return ", out);
975 if (cast)
977 int array_depth = 0;
978 const unsigned char *limit;
980 fputs ("reinterpret_cast<", out);
982 while (*method_data == '[')
984 ++method_data;
985 ++array_depth;
986 --method_name_len;
987 fputs ("JArray<", out);
990 /* Leading space to avoid C++ digraphs. */
991 fputs (" ::", out);
993 /* If we see an `L', skip it and the trailing `;'. Only do this
994 if we've seen an array specification. If we don't have an
995 array then the `L' was stripped earlier. */
996 if (array_depth && method_data[0] == 'L'
997 && method_data[method_name_len - 1] == ';')
999 ++method_data;
1000 method_name_len -= 2;
1003 limit = method_data + method_name_len;
1004 while (method_data < limit)
1006 int ch = UTF8_GET (method_data, limit);
1007 if (ch == '/')
1008 fputs ("::", out);
1009 else
1010 jcf_print_char (out, ch);
1012 fputs (" *", out);
1014 /* Close each array. */
1015 while (array_depth > 0)
1017 fputs ("> *", out);
1018 --array_depth;
1021 /* Close the cast. */
1022 fputs ("> (", out);
1025 if (nameindex == -1)
1026 fputs ("this", out);
1027 else
1028 print_field_name (out, jcf, nameindex, 0);
1030 if (cast)
1031 fputs (")", out);
1033 fputs ("; }", out);
1037 /* Try to decompile a method body. Right now we just try to handle a
1038 simple case that we can do. Expand as desired. */
1039 static void
1040 decompile_method (FILE *out, JCF *jcf, int code_len)
1042 const unsigned char *codes = jcf->read_ptr;
1043 int index;
1044 uint16 name_and_type, name;
1046 /* If the method is synchronized, don't touch it. */
1047 if ((method_access & ACC_SYNCHRONIZED))
1048 return;
1050 if (code_len == 5
1051 && codes[0] == OPCODE_aload_0
1052 && codes[1] == OPCODE_getfield
1053 && (codes[4] == OPCODE_areturn
1054 || codes[4] == OPCODE_dreturn
1055 || codes[4] == OPCODE_freturn
1056 || codes[4] == OPCODE_ireturn
1057 || codes[4] == OPCODE_lreturn))
1059 /* Found code like `return FIELD'. */
1060 index = (codes[2] << 8) | codes[3];
1061 /* FIXME: ensure that tag is CONSTANT_Fieldref. */
1062 name_and_type = JPOOL_USHORT2 (jcf, index);
1063 /* FIXME: ensure that tag is CONSTANT_NameAndType. */
1064 name = JPOOL_USHORT1 (jcf, name_and_type);
1065 if (codes[4] == OPCODE_areturn)
1066 decompile_return_statement (out, jcf, method_signature,
1067 name, JPOOL_USHORT2 (jcf, name_and_type));
1068 else
1070 fputs (" { return ", out);
1071 /* FIXME: flags. */
1072 print_field_name (out, jcf, name, 0);
1073 fputs ("; }", out);
1075 decompiled = 1;
1077 else if (code_len == 2
1078 && codes[0] == OPCODE_aload_0
1079 && codes[1] == OPCODE_areturn
1080 /* We're going to generate `return this'. This only makes
1081 sense for non-static methods. */
1082 && ! (method_access & ACC_STATIC))
1084 decompile_return_statement (out, jcf, method_signature, -1,
1085 JPOOL_USHORT1 (jcf, jcf->this_class));
1086 decompiled = 1;
1088 else if (code_len == 1 && codes[0] == OPCODE_return)
1090 /* Found plain `return'. */
1091 fputs (" { }", out);
1092 decompiled = 1;
1094 else if (code_len == 2
1095 && codes[0] == OPCODE_aconst_null
1096 && codes[1] == OPCODE_areturn)
1098 /* Found `return null'. We don't want to depend on NULL being
1099 defined. */
1100 fputs (" { return 0; }", out);
1101 decompiled = 1;
1105 /* Like strcmp, but invert the return result for the hash table. This
1106 should probably be in hashtab.c to complement the existing string
1107 hash function. */
1108 static int
1109 gcjh_streq (const void *p1, const void *p2)
1111 return ! strcmp ((char *) p1, (char *) p2);
1114 /* Return 1 if the initial part of CLNAME names a subclass of throwable,
1115 or 0 if not. CLNAME may be extracted from a signature, and can be
1116 terminated with either `;' or NULL. */
1117 static int
1118 throwable_p (const unsigned char *clname)
1120 int length;
1121 unsigned char *current;
1122 int i;
1123 int result = 0;
1125 /* We keep two hash tables of class names. In one we list all the
1126 classes which are subclasses of Throwable. In the other we will
1127 all other classes. We keep two tables to make the code a bit
1128 simpler; we don't have to have a structure mapping class name to
1129 a `throwable?' bit. */
1130 static htab_t throw_hash;
1131 static htab_t non_throw_hash;
1132 static int init_done = 0;
1134 if (! init_done)
1136 void **slot;
1137 unsigned char *str;
1139 /* Self-initializing. The cost of this really doesn't matter.
1140 We also don't care about freeing these, either. */
1141 throw_hash = htab_create (10, htab_hash_string, gcjh_streq,
1142 (htab_del) free);
1143 non_throw_hash = htab_create (10, htab_hash_string, gcjh_streq,
1144 (htab_del) free);
1146 /* Make sure the root classes show up in the tables. */
1147 str = xstrdup ("java.lang.Throwable");
1148 slot = htab_find_slot (throw_hash, str, INSERT);
1149 *slot = str;
1151 str = xstrdup ("java.lang.Object");
1152 slot = htab_find_slot (non_throw_hash, str, INSERT);
1153 *slot = str;
1155 init_done = 1;
1158 for (length = 0; clname[length] != ';' && clname[length] != '\0'; ++length)
1160 current = ALLOC (length + 1);
1161 for (i = 0; i < length; ++i)
1162 current[i] = clname[i] == '/' ? '.' : clname[i];
1163 current[length] = '\0';
1165 /* We don't compute the hash slot here because the table might be
1166 modified by the recursion. In that case the slot could be
1167 invalidated. */
1168 if (htab_find (throw_hash, current))
1169 result = 1;
1170 else if (htab_find (non_throw_hash, current))
1171 result = 0;
1172 else
1174 JCF jcf;
1175 void **slot;
1176 unsigned char *super, *tmp;
1177 int super_length = -1;
1178 const char *classfile_name = find_class (current, strlen (current),
1179 &jcf, 0);
1181 if (! classfile_name)
1183 error ("couldn't find class %s", current);
1184 return 0;
1186 if (jcf_parse_preamble (&jcf) != 0
1187 || jcf_parse_constant_pool (&jcf) != 0
1188 || verify_constant_pool (&jcf) > 0)
1190 error ("parse error while reading %s", classfile_name);
1191 return 0;
1193 jcf_parse_class (&jcf);
1195 tmp = (unsigned char *) super_class_name (&jcf, &super_length);
1196 super = ALLOC (super_length + 1);
1197 memcpy (super, tmp, super_length);
1198 super[super_length] = '\0';
1200 result = throwable_p (super);
1201 slot = htab_find_slot (result ? throw_hash : non_throw_hash,
1202 current, INSERT);
1203 *slot = current;
1204 current = NULL;
1206 JCF_FINISH (&jcf);
1209 return result;
1212 /* Print one piece of a signature. Returns pointer to next parseable
1213 character on success, NULL on error. */
1214 static const unsigned char *
1215 decode_signature_piece (FILE *stream, const unsigned char *signature,
1216 const unsigned char *limit, int *need_space)
1218 const char *ctype;
1219 int array_depth = 0;
1221 switch (signature[0])
1223 case '[':
1224 /* More spaghetti. */
1226 array_loop:
1227 for (signature++; (signature < limit
1228 && ISDIGIT (*signature)); signature++)
1230 switch (*signature)
1232 case 'B':
1233 ctype = "jbyteArray";
1234 break;
1235 case 'C':
1236 ctype = "jcharArray";
1237 break;
1238 case 'D':
1239 ctype = "jdoubleArray";
1240 break;
1241 case 'F':
1242 ctype = "jfloatArray";
1243 break;
1244 case 'I':
1245 ctype = "jintArray";
1246 break;
1247 case 'S':
1248 ctype = "jshortArray";
1249 break;
1250 case 'J':
1251 ctype = "jlongArray";
1252 break;
1253 case 'Z':
1254 ctype = "jbooleanArray";
1255 break;
1256 case '[':
1257 /* We have a nested array. */
1258 ++array_depth;
1259 if (! flag_jni)
1260 fputs ("JArray<", stream);
1261 goto array_loop;
1263 case 'L':
1264 /* We have to generate a reference to JArray here, so that
1265 our output matches what the compiler does. */
1266 ++signature;
1267 /* Space between `<' and `:' to avoid C++ digraphs. */
1268 if (! flag_jni)
1269 fputs ("JArray< ::", stream);
1270 while (signature < limit && *signature != ';')
1272 int ch = UTF8_GET (signature, limit);
1273 if (! flag_jni)
1275 if (ch == '/')
1276 fputs ("::", stream);
1277 else
1278 jcf_print_char (stream, ch);
1281 if (! flag_jni)
1282 fputs (" *> *", stream);
1283 *need_space = 0;
1284 ctype = NULL;
1285 break;
1286 default:
1287 /* Unparseable signature. */
1288 return NULL;
1291 /* If the previous iterations left us with something to print,
1292 print it. For JNI, we always print `jobjectArray' in the
1293 nested cases. */
1294 if (flag_jni && (ctype == NULL || array_depth > 0))
1296 ctype = "jobjectArray";
1297 *need_space = 1;
1299 /* The `printit' case will advance SIGNATURE for us. If we
1300 don't go there, we must advance past the `;' ourselves. */
1301 if (ctype != NULL)
1302 goto printit;
1303 ++signature;
1304 break;
1306 case '(':
1307 case ')':
1308 /* This shouldn't happen. */
1309 return NULL;
1311 case 'B': ctype = "jbyte"; goto printit;
1312 case 'C': ctype = "jchar"; goto printit;
1313 case 'D': ctype = "jdouble"; goto printit;
1314 case 'F': ctype = "jfloat"; goto printit;
1315 case 'I': ctype = "jint"; goto printit;
1316 case 'J': ctype = "jlong"; goto printit;
1317 case 'S': ctype = "jshort"; goto printit;
1318 case 'Z': ctype = "jboolean"; goto printit;
1319 case 'V': ctype = "void"; goto printit;
1320 case 'L':
1321 if (flag_jni)
1323 /* We know about certain types and special-case their names. */
1324 if (! strncmp (signature, "Ljava/lang/String;",
1325 sizeof ("Ljava/lang/String;") -1))
1326 ctype = "jstring";
1327 else if (! strncmp (signature, "Ljava/lang/Class;",
1328 sizeof ("Ljava/lang/Class;") - 1))
1329 ctype = "jclass";
1330 /* Skip leading 'L' for throwable_p call. */
1331 else if (throwable_p (signature + 1))
1332 ctype = "jthrowable";
1333 else
1334 ctype = "jobject";
1336 while (*signature && *signature != ';')
1337 ++signature;
1339 goto printit;
1341 /* Print a leading "::" so we look in the right namespace. */
1342 fputs ("::", stream);
1343 ++signature;
1344 while (*signature && *signature != ';')
1346 int ch = UTF8_GET (signature, limit);
1347 if (ch == '/')
1348 fputs ("::", stream);
1349 else
1350 jcf_print_char (stream, ch);
1352 fputs (" *", stream);
1353 if (*signature == ';')
1354 signature++;
1355 *need_space = 0;
1356 break;
1357 default:
1358 *need_space = 1;
1359 jni_print_char (stream, *signature++);
1360 break;
1361 printit:
1362 signature++;
1363 *need_space = 1;
1364 fputs (ctype, stream);
1365 break;
1368 if (! flag_jni)
1370 while (array_depth-- > 0)
1371 fputs ("> *", stream);
1374 return signature;
1377 static void
1378 print_c_decl (FILE* stream, JCF* jcf, int name_index, int signature_index,
1379 int is_init, const char *name_override, int flags)
1381 if (JPOOL_TAG (jcf, signature_index) != CONSTANT_Utf8)
1383 fprintf (stream, "<not a UTF8 constant>");
1384 found_error = 1;
1386 else
1388 int length = JPOOL_UTF_LENGTH (jcf, signature_index);
1389 const unsigned char *str0 = JPOOL_UTF_DATA (jcf, signature_index);
1390 const unsigned char *str = str0;
1391 const unsigned char *limit = str + length;
1392 int need_space = 0;
1393 int is_method = str[0] == '(';
1394 const unsigned char *next;
1396 /* If printing a method, skip to the return signature and print
1397 that first. However, there is no return value if this is a
1398 constructor. */
1399 if (is_method && ! is_init)
1401 while (str < limit)
1403 int ch = *str++;
1404 if (ch == ')')
1405 break;
1409 /* If printing a field or an ordinary method, then print the
1410 "return value" now. */
1411 if (! is_method || ! is_init)
1413 next = decode_signature_piece (stream, str, limit, &need_space);
1414 if (! next)
1416 error ("unparseable signature: `%s'", str0);
1417 return;
1421 /* Force the alignment of the first data member. This is
1422 because the "new" C++ ABI changed the alignment of non-POD
1423 classes. gcj, however, still uses the "old" alignment. */
1424 if (is_first_data_member && ! (flags & ACC_STATIC) && ! is_method)
1426 is_first_data_member = 0;
1427 print_cxx_classname (out, " __attribute__((aligned(__alignof__( ",
1428 jcf, jcf->super_class, 1);
1429 fputs (" )))) ", stream);
1432 /* Now print the name of the thing. */
1433 if (need_space)
1434 fputs (" ", stream);
1435 print_full_cxx_name (stream, jcf, name_index,
1436 signature_index, is_init, name_override,
1437 flags);
1441 /* Print the unqualified method name followed by the signature. */
1442 static void
1443 print_full_cxx_name (FILE* stream, JCF* jcf, int name_index,
1444 int signature_index, int is_init,
1445 const char *name_override, int flags)
1447 int length = JPOOL_UTF_LENGTH (jcf, signature_index);
1448 const unsigned char *str0 = JPOOL_UTF_DATA (jcf, signature_index);
1449 const unsigned char *str = str0;
1450 const unsigned char *limit = str + length;
1451 int need_space = 0;
1452 int is_method = str[0] == '(';
1453 const unsigned char *next;
1455 if (name_override)
1456 fputs (name_override, stream);
1457 else if (name_index)
1459 /* Declare constructors specially. */
1460 if (is_init)
1461 print_base_classname (stream, jcf, jcf->this_class);
1462 else
1463 print_name (stream, jcf, name_index);
1466 if (flag_jni)
1468 unsigned char *signature = JPOOL_UTF_DATA (jcf, signature_index);
1469 int sig_len = JPOOL_UTF_LENGTH (jcf, signature_index);
1470 if (overloaded_jni_method_exists_p (JPOOL_UTF_DATA (jcf, name_index),
1471 JPOOL_UTF_LENGTH (jcf, name_index),
1472 signature, sig_len))
1474 /* If this method is overloaded by another native method,
1475 then include the argument information in the mangled
1476 name. */
1477 unsigned char *limit = signature + sig_len;
1478 fputs ("__", stream);
1479 while (signature < limit)
1481 int ch = UTF8_GET (signature, limit);
1482 jni_print_char (stream, ch);
1483 if (ch == ')')
1485 /* Done. */
1486 break;
1492 if (is_method)
1494 /* Have a method or a constructor. Print signature pieces
1495 until done. */
1496 fputs (" (", stream);
1498 str = str0 + 1;
1500 /* In JNI mode, add extra arguments. */
1501 if (flag_jni)
1503 /* FIXME: it would be nice to know if we are printing a decl
1504 or a definition, and only print `env' for the latter. */
1505 fputs ("JNIEnv *env", stream);
1507 fputs ((flags & ACC_STATIC) ? ", jclass" : ", jobject", stream);
1509 if (*str != ')')
1510 fputs (", ", stream);
1513 while (str < limit && *str != ')')
1515 next = decode_signature_piece (stream, str, limit, &need_space);
1516 if (! next)
1518 error ("unparseable signature: `%s'", str0);
1519 return;
1522 if (next < limit && *next != ')')
1523 fputs (", ", stream);
1524 str = next;
1527 fputs (")", stream);
1531 /* This is a helper for print_stub_or_jni. */
1532 static void
1533 print_name_for_stub_or_jni (FILE *stream, JCF *jcf, int name_index,
1534 int signature_index, int is_init,
1535 const char *name_override, int flags)
1537 const char *const prefix = flag_jni ? "Java_" : "";
1538 print_cxx_classname (stream, prefix, jcf, jcf->this_class, 1);
1539 fputs (flag_jni ? "_" : "::", stream);
1540 print_full_cxx_name (stream, jcf, name_index,
1541 signature_index, is_init, name_override,
1542 flags);
1545 static void
1546 print_stub_or_jni (FILE* stream, JCF* jcf, int name_index,
1547 int signature_index, int is_init,
1548 const char *name_override, int flags)
1550 if (JPOOL_TAG (jcf, signature_index) != CONSTANT_Utf8)
1552 fprintf (stream, "<not a UTF8 constant>");
1553 found_error = 1;
1555 else
1557 int length = JPOOL_UTF_LENGTH (jcf, signature_index);
1558 const unsigned char *str0 = JPOOL_UTF_DATA (jcf, signature_index);
1559 const unsigned char *str = str0;
1560 const unsigned char *limit = str + length;
1561 int need_space = 0;
1562 int is_method = str[0] == '(';
1563 const unsigned char *next;
1565 /* Don't print fields in the JNI case. */
1566 if (! is_method && flag_jni)
1567 return;
1569 if (flag_jni && ! stubs)
1570 fputs ("JNIEXPORT ", stream);
1572 /* If printing a method, skip to the return signature and print
1573 that first. However, there is no return value if this is a
1574 constructor. */
1575 if (is_method && ! is_init)
1577 while (str < limit)
1579 int ch = *str++;
1580 if (ch == ')')
1581 break;
1585 /* If printing a field or an ordinary method, then print the
1586 "return value" now. Note that a constructor can't be native,
1587 so we don't bother checking this in the JNI case. */
1588 if (! is_method || ! is_init)
1590 next = decode_signature_piece (stream, str, limit, &need_space);
1591 if (! next)
1593 error ("unparseable signature: `%s'", str0);
1594 return;
1598 /* When printing a JNI header we need to respect the space. In
1599 other cases we're just going to insert a newline anyway. */
1600 fputs (need_space && ! stubs ? " " : "\n", stream);
1602 if (flag_jni && ! stubs)
1603 fputs ("JNICALL ", stream);
1605 /* Now print the name of the thing. */
1606 print_name_for_stub_or_jni (stream, jcf, name_index,
1607 signature_index, is_init, name_override,
1608 flags);
1610 /* Print the body. */
1611 if (stubs)
1613 if (flag_jni)
1614 fputs ("\n{\n (*env)->FatalError (env, \"", stream);
1615 else
1616 fputs ("\n{\n throw new ::java::lang::UnsupportedOperationException (JvNewStringLatin1 (\"", stream);
1617 print_name_for_stub_or_jni (stream, jcf, name_index,
1618 signature_index, is_init,
1619 name_override,
1620 flags);
1621 fprintf (stream, " not implemented\")%s;\n}\n\n",
1622 flag_jni ? "" : ")");
1627 static void
1628 print_mangled_classname (FILE *stream, JCF *jcf, const char *prefix, int index)
1630 int name_index = JPOOL_USHORT1 (jcf, index);
1631 fputs (prefix, stream);
1632 jcf_print_utf8_replace (out,
1633 JPOOL_UTF_DATA (jcf, name_index),
1634 JPOOL_UTF_LENGTH (jcf, name_index),
1635 '/', '_');
1638 /* Print PREFIX, then a class name in C++ format. If the name refers
1639 to an array, ignore it and don't print PREFIX. Returns 1 if
1640 something was printed, 0 otherwise. */
1641 static int
1642 print_cxx_classname (FILE *stream, const char *prefix,
1643 JCF *jcf, int index, int add_scope)
1645 int name_index = JPOOL_USHORT1 (jcf, index);
1646 int len, c;
1647 const unsigned char *s, *p, *limit;
1649 s = JPOOL_UTF_DATA (jcf, name_index);
1650 len = JPOOL_UTF_LENGTH (jcf, name_index);
1651 limit = s + len;
1653 /* Explicitly omit arrays here. */
1654 p = s;
1655 c = UTF8_GET (p, limit);
1656 if (c == '[')
1657 return 0;
1659 fputs (prefix, stream);
1661 /* Print a leading "::" so we look in the right namespace. */
1662 if (! flag_jni && ! stubs && add_scope)
1663 fputs ("::", stream);
1665 while (s < limit)
1667 c = UTF8_GET (s, limit);
1668 if (c == '/')
1669 fputs (flag_jni ? "_" : "::", stream);
1670 else
1671 jni_print_char (stream, c);
1674 return 1;
1677 int written_class_count = 0;
1679 /* Return name of superclass. If LEN is not NULL, fill it with length
1680 of name. */
1681 static const unsigned char *
1682 super_class_name (JCF *derived_jcf, int *len)
1684 int supername_index = JPOOL_USHORT1 (derived_jcf, derived_jcf->super_class);
1685 int supername_length = JPOOL_UTF_LENGTH (derived_jcf, supername_index);
1686 const unsigned char *supername =
1687 JPOOL_UTF_DATA (derived_jcf, supername_index);
1689 if (len)
1690 *len = supername_length;
1692 return supername;
1695 static void
1696 handle_inner_classes (int count)
1698 int i;
1700 if (out && ! flag_jni && ! stubs && count > 0)
1701 fprintf (out, "\n");
1703 for (i = 0; i < count; ++i)
1705 JCF_u2 inner_info_index = JCF_readu2 (current_jcf);
1707 /* There are a few more values here, but we don't care about
1708 them. The (void) cast is apparently the only way to avoid a
1709 warning here. */
1710 (void) JCF_readu2 (current_jcf);
1711 (void) JCF_readu2 (current_jcf);
1712 (void) JCF_readu2 (current_jcf);
1714 if (out && ! flag_jni && ! stubs)
1716 print_mangled_classname (out, current_jcf, " friend class ",
1717 inner_info_index);
1718 fprintf (out, ";\n");
1725 /* We keep track of all the `#include's we generate, so we can avoid
1726 duplicates. */
1727 struct include
1729 char *name;
1730 struct include *next;
1733 /* List of all includes. */
1734 static struct include *all_includes = NULL;
1736 /* Generate a #include. */
1737 static void
1738 print_include (FILE *out, const unsigned char *utf8, int len)
1740 struct include *incl;
1742 if (! out)
1743 return;
1745 if (len == -1)
1746 len = strlen (utf8);
1748 for (incl = all_includes; incl; incl = incl->next)
1750 /* We check the length because we might have a proper prefix. */
1751 if (len == (int) strlen (incl->name)
1752 && ! strncmp (incl->name, utf8, len))
1753 return;
1756 incl = xmalloc (sizeof (struct include));
1757 incl->name = xmalloc (len + 1);
1758 strncpy (incl->name, utf8, len);
1759 incl->name[len] = '\0';
1760 incl->next = all_includes;
1761 all_includes = incl;
1763 fputs ("#include <", out);
1764 jcf_print_utf8_replace (out, utf8, len,
1765 '/',
1766 flag_jni ? '_' : '/');
1767 fputs (".h>\n", out);
1772 /* This is used to represent part of a package or class name. */
1773 struct namelet
1775 /* The text of this part of the name. */
1776 char *name;
1777 /* True if this represents a class. */
1778 int is_class;
1779 /* Linked list of all classes and packages inside this one. */
1780 struct namelet *subnamelets;
1781 /* Pointer to next sibling. */
1782 struct namelet *next;
1785 static void add_namelet (const unsigned char *, const unsigned char *,
1786 struct namelet *);
1787 static void print_namelet (FILE *, struct namelet *, int);
1789 /* The special root namelet. */
1790 static struct namelet root =
1792 NULL,
1794 NULL,
1795 NULL
1798 /* This extracts the next name segment from the full UTF-8 encoded
1799 package or class name and links it into the tree. It does this
1800 recursively. */
1801 static void
1802 add_namelet (const unsigned char *name, const unsigned char *name_limit,
1803 struct namelet *parent)
1805 const unsigned char *p;
1806 struct namelet *n = NULL, *np;
1808 /* We want to skip the standard namespaces that we assume the
1809 runtime already knows about. We only do this at the top level,
1810 though, hence the check for `root'. */
1811 if (parent == &root)
1813 #define JAVALANG "java/lang/"
1814 #define JAVAIO "java/io/"
1815 #define JAVAUTIL "java/util/"
1816 if ((name_limit - name >= (int) sizeof (JAVALANG) - 1
1817 && ! strncmp (name, JAVALANG, sizeof (JAVALANG) - 1))
1818 || (name_limit - name >= (int) sizeof (JAVAUTIL) - 1
1819 && ! strncmp (name, JAVAUTIL, sizeof (JAVAUTIL) - 1))
1820 || (name_limit - name >= (int) sizeof (JAVAIO) - 1
1821 && ! strncmp (name, JAVAIO, sizeof (JAVAIO) - 1)))
1822 return;
1825 for (p = name; p < name_limit && *p != '/'; ++p)
1828 /* Search for this name beneath the PARENT node. */
1829 for (np = parent->subnamelets; np != NULL; np = np->next)
1831 /* We check the length because we might have a proper prefix. */
1832 if ((int) strlen (np->name) == p - name &&
1833 ! strncmp (name, np->name, p - name))
1835 n = np;
1836 break;
1840 if (n == NULL)
1842 n = xmalloc (sizeof (struct namelet));
1843 n->name = xmalloc (p - name + 1);
1844 strncpy (n->name, name, p - name);
1845 n->name[p - name] = '\0';
1846 n->is_class = (p == name_limit);
1847 n->subnamelets = NULL;
1848 n->next = parent->subnamelets;
1849 parent->subnamelets = n;
1852 /* We recurse if there is more text, and if the trailing piece does
1853 not represent an inner class. */
1854 if (p < name_limit)
1855 add_namelet (p + 1, name_limit, n);
1858 /* Print a single namelet. Destroys namelets while printing. */
1859 static void
1860 print_namelet (FILE *out, struct namelet *name, int depth)
1862 int i, term = 0;
1863 struct namelet *c;
1865 if (name->name)
1867 for (i = 0; i < depth; ++i)
1868 fputc (' ', out);
1869 fprintf (out, "%s %s", name->is_class ? "class" : "namespace",
1870 name->name);
1871 if (name->is_class && name->subnamelets == NULL)
1872 fputs (";\n", out);
1873 else
1875 term = 1;
1876 fputs ("\n", out);
1877 for (i = 0; i < depth; ++i)
1878 fputc (' ', out);
1879 fputs ("{\n", out);
1883 c = name->subnamelets;
1884 while (c != NULL)
1886 struct namelet *next = c->next;
1887 print_namelet (out, c, depth + 2);
1888 c = next;
1890 name->subnamelets = NULL;
1892 if (name->name)
1894 if (term)
1896 for (i = 0; i < depth; ++i)
1897 fputc (' ', out);
1898 fputs ("}\n", out);
1899 /* Only print a `;' when printing a class. C++ is evil. */
1900 if (name->is_class)
1901 fputs (";", out);
1904 free (name->name);
1905 free (name);
1909 /* This is called to add some classes to the list of classes for which
1910 we need decls. The signature argument can be a function
1911 signature. */
1912 static void
1913 add_class_decl (FILE *out, JCF *jcf, JCF_u2 signature)
1915 const unsigned char *s = JPOOL_UTF_DATA (jcf, signature);
1916 int len = JPOOL_UTF_LENGTH (jcf, signature);
1917 int i;
1919 for (i = 0; i < len; ++i)
1921 int start;
1923 /* If we see an array, then we include the array header. */
1924 if (s[i] == '[')
1926 print_include (out, "gcj/array", -1);
1927 continue;
1930 /* We're looking for `L<stuff>;' -- everything else is
1931 ignorable. */
1932 if (s[i] != 'L')
1933 continue;
1935 for (start = ++i; i < len && s[i] != ';'; ++i)
1938 add_namelet (&s[start], &s[i], &root);
1942 /* Print declarations for all classes required by this class. Any
1943 class or package in the `java' package is assumed to be handled
1944 statically in libjava; we don't generate declarations for these.
1945 This makes the generated headers a bit easier to read. */
1946 static void
1947 print_class_decls (FILE *out, JCF *jcf, int self)
1949 /* Make sure to always add the current class to the list of things
1950 that should be declared. */
1951 int name_index = JPOOL_USHORT1 (jcf, self);
1952 int len;
1953 const unsigned char *s;
1955 s = JPOOL_UTF_DATA (jcf, name_index);
1956 len = JPOOL_UTF_LENGTH (jcf, name_index);
1957 add_namelet (s, s + len, &root);
1959 if (root.subnamelets)
1961 fputs ("extern \"Java\"\n{\n", out);
1962 /* We use an initial offset of 0 because the root namelet
1963 doesn't cause anything to print. */
1964 print_namelet (out, &root, 0);
1965 fputs ("}\n\n", out);
1971 static void
1972 process_file (JCF *jcf, FILE *out)
1974 int code, i;
1975 uint32 field_start, method_end, method_start;
1977 current_jcf = jcf;
1979 last_access = -1;
1981 if (jcf_parse_preamble (jcf) != 0)
1983 error ("Not a valid Java .class file.");
1984 return;
1987 /* Parse and possibly print constant pool */
1988 code = jcf_parse_constant_pool (jcf);
1989 if (code != 0)
1991 error ("error while parsing constant pool");
1992 return;
1994 code = verify_constant_pool (jcf);
1995 if (code > 0)
1997 error ("error in constant pool entry #%d", code);
1998 return;
2001 jcf_parse_class (jcf);
2003 if (written_class_count++ == 0 && out)
2005 const char *cstart, *cstart2, *mode, *cend, *what, *jflag;
2006 if (flag_jni)
2008 cstart = "/*";
2009 cstart2 = " ";
2010 cend = " */";
2011 mode = "";
2012 what = "JNI";
2013 jflag = " -jni";
2015 else
2017 cstart = "//";
2018 cstart2 = "//";
2019 cend = "";
2020 mode = " -*- c++ -*-";
2021 what = "CNI";
2022 jflag = "";
2025 if (! stubs)
2026 fprintf (out, "%s DO NOT EDIT THIS FILE - it is machine generated%s%s\n\n",
2027 cstart, mode, cend);
2028 else
2030 fprintf (out, "%s This file was created by `gcjh -stubs%s'.%s\n\
2031 %s\n\
2032 %s This file is intended to give you a head start on implementing native\n\
2033 %s methods using %s.\n\
2034 %s Be aware: running `gcjh -stubs %s' once more for this class may\n\
2035 %s overwrite any edits you have made to this file.%s\n\n",
2036 cstart, jflag, mode,
2037 cstart2,
2038 cstart2,
2039 cstart2,
2040 what,
2041 cstart2,
2042 jflag,
2043 cstart2,
2044 cend);
2048 if (out)
2050 if (! stubs)
2052 print_mangled_classname (out, jcf, "#ifndef __", jcf->this_class);
2053 fprintf (out, "__\n");
2055 print_mangled_classname (out, jcf, "#define __", jcf->this_class);
2056 fprintf (out, "__\n\n");
2058 if (flag_jni)
2060 fprintf (out, "#include <jni.h>\n\n");
2061 fprintf (out, "#ifdef __cplusplus\n");
2062 fprintf (out, "extern \"C\"\n");
2063 fprintf (out, "{\n");
2064 fprintf (out, "#endif\n");
2066 else
2068 /* We do this to ensure that inline methods won't be
2069 `outlined' by g++. This works as long as method and
2070 fields are not added by the user. */
2071 fprintf (out, "#pragma interface\n");
2073 if (jcf->super_class)
2075 int super_length;
2076 const unsigned char *supername =
2077 super_class_name (jcf, &super_length);
2079 fputs ("\n", out);
2080 print_include (out, supername, super_length);
2084 else
2086 /* Strip off the ".class" portion of the name when printing
2087 the include file name. */
2088 char *name;
2089 int i, len = strlen (jcf->classname);
2090 if (len > 6 && ! strcmp (&jcf->classname[len - 6], ".class"))
2091 len -= 6;
2092 /* Turn the class name into a file name. */
2093 name = xmalloc (len + 1);
2094 for (i = 0; i < len; ++i)
2095 name[i] = jcf->classname[i] == '.' ? '/' : jcf->classname[i];
2096 name[i] = '\0';
2097 print_include (out, name, len);
2098 free (name);
2100 if (! flag_jni)
2102 print_include (out, "gcj/cni", -1);
2103 print_include (out, "java/lang/UnsupportedOperationException",
2104 -1);
2109 /* We want to parse the methods first. But we need to find where
2110 they start. So first we skip the fields, then parse the methods.
2111 Then we parse the fields and skip the methods. This is ugly, but
2112 not too bad since we need two full passes to get class decl
2113 information anyway. */
2114 field_pass = 0;
2115 field_start = JCF_TELL (jcf);
2116 jcf_parse_fields (jcf);
2118 method_start = JCF_TELL (jcf);
2119 method_pass = 0;
2120 jcf_parse_methods (jcf);
2122 if (out)
2123 fputs ("\n", out);
2125 if (out && ! flag_jni)
2127 if (! stubs)
2128 print_class_decls (out, jcf, jcf->this_class);
2130 for (i = 0; i < prepend_count; ++i)
2131 fprintf (out, "%s\n", prepend_specs[i]);
2132 if (prepend_count > 0)
2133 fputc ('\n', out);
2135 if (! stubs)
2137 if (! print_cxx_classname (out, "class ", jcf,
2138 jcf->this_class, 0))
2140 error ("class is of array type\n");
2141 return;
2143 if (jcf->super_class)
2145 if (! print_cxx_classname (out, " : public ",
2146 jcf, jcf->super_class, 1))
2148 error ("base class is of array type");
2149 return;
2153 fputs ("\n{\n", out);
2157 /* Now go back for second pass over methods and fields. */
2158 is_first_data_member = 1;
2160 JCF_SEEK (jcf, method_start);
2161 method_pass = 1;
2162 jcf_parse_methods (jcf);
2163 method_end = JCF_TELL (jcf);
2165 field_pass = 1;
2166 JCF_SEEK (jcf, field_start);
2167 jcf_parse_fields (jcf);
2168 JCF_SEEK (jcf, method_end);
2170 jcf_parse_final_attributes (jcf);
2172 if (out && ! stubs)
2174 if (flag_jni)
2176 fprintf (out, "\n#ifdef __cplusplus\n");
2177 fprintf (out, "}\n");
2178 fprintf (out, "#endif\n");
2180 else
2182 /* Generate friend decl if we still must. */
2183 for (i = 0; i < friend_count; ++i)
2184 fprintf (out, " friend %s\n", friend_specs[i]);
2186 /* Generate extra declarations. */
2187 if (add_count > 0)
2188 fputc ('\n', out);
2189 for (i = 0; i < add_count; ++i)
2190 fprintf (out, " %s\n", add_specs[i]);
2192 /* Generate an entry for the class object. */
2193 generate_access (out, ACC_PUBLIC);
2194 fprintf (out, "\n static ::java::lang::Class class$;\n");
2196 fputs ("}", out);
2198 if (jcf->access_flags & ACC_INTERFACE)
2199 fputs (" __attribute__ ((java_interface))", out);
2201 fputs (";\n", out);
2203 if (append_count > 0)
2204 fputc ('\n', out);
2205 for (i = 0; i < append_count; ++i)
2206 fprintf (out, "%s\n", append_specs[i]);
2209 print_mangled_classname (out, jcf,
2210 "\n#endif /* __", jcf->this_class);
2211 fprintf (out, "__ */\n");
2217 /* This is used to mark options with no short value. */
2218 #define LONG_OPT(Num) ((Num) + 128)
2220 #define OPT_classpath LONG_OPT (0)
2221 #define OPT_CLASSPATH OPT_classpath
2222 #define OPT_bootclasspath LONG_OPT (1)
2223 #define OPT_extdirs LONG_OPT (2)
2224 #define OPT_HELP LONG_OPT (3)
2225 #define OPT_TEMP LONG_OPT (4)
2226 #define OPT_VERSION LONG_OPT (5)
2227 #define OPT_PREPEND LONG_OPT (6)
2228 #define OPT_FRIEND LONG_OPT (7)
2229 #define OPT_ADD LONG_OPT (8)
2230 #define OPT_APPEND LONG_OPT (9)
2231 #define OPT_M LONG_OPT (10)
2232 #define OPT_MM LONG_OPT (11)
2233 #define OPT_MG LONG_OPT (12)
2234 #define OPT_MD LONG_OPT (13)
2235 #define OPT_MMD LONG_OPT (14)
2237 static const struct option options[] =
2239 { "classpath", required_argument, NULL, OPT_classpath },
2240 { "bootclasspath", required_argument, NULL, OPT_bootclasspath },
2241 { "extdirs", required_argument, NULL, OPT_extdirs },
2242 { "CLASSPATH", required_argument, NULL, OPT_CLASSPATH },
2243 { "help", no_argument, NULL, OPT_HELP },
2244 { "stubs", no_argument, &stubs, 1 },
2245 { "td", required_argument, NULL, OPT_TEMP },
2246 { "verbose", no_argument, NULL, 'v' },
2247 { "version", no_argument, NULL, OPT_VERSION },
2248 { "prepend", required_argument, NULL, OPT_PREPEND },
2249 { "friend", required_argument, NULL, OPT_FRIEND },
2250 { "add", required_argument, NULL, OPT_ADD },
2251 { "append", required_argument, NULL, OPT_APPEND },
2252 { "M", no_argument, NULL, OPT_M },
2253 { "MM", no_argument, NULL, OPT_MM },
2254 { "MG", no_argument, NULL, OPT_MG },
2255 { "MD", no_argument, NULL, OPT_MD },
2256 { "MMD", no_argument, NULL, OPT_MMD },
2257 { "jni", no_argument, &flag_jni, 1 },
2258 { NULL, no_argument, NULL, 0 }
2261 static void
2262 usage (void)
2264 fprintf (stderr, _("Try `gcjh --help' for more information.\n"));
2265 exit (1);
2268 static void
2269 help (void)
2271 printf (_("Usage: gcjh [OPTION]... CLASS...\n\n"));
2272 printf (_("Generate C++ header files from .class files\n\n"));
2273 printf (_(" -stubs Generate an implementation stub file\n"));
2274 printf (_(" -jni Generate a JNI header or stub\n"));
2275 printf ("\n");
2276 printf (_(" -add TEXT Insert TEXT into class body\n"));
2277 printf (_(" -append TEXT Insert TEXT after class declaration\n"));
2278 printf (_(" -friend TEXT Insert TEXT as `friend' declaration\n"));
2279 printf (_(" -prepend TEXT Insert TEXT before start of class\n"));
2280 printf ("\n");
2281 printf (_(" --classpath PATH Set path to find .class files\n"));
2282 printf (_(" -IDIR Append directory to class path\n"));
2283 printf (_(" --bootclasspath PATH Override built-in class path\n"));
2284 printf (_(" --extdirs PATH Set extensions directory path\n"));
2285 printf (_(" -d DIRECTORY Set output directory name\n"));
2286 printf (_(" -o FILE Set output file name\n"));
2287 printf (_(" -td DIRECTORY Set temporary directory name\n"));
2288 printf ("\n");
2289 printf (_(" --help Print this help, then exit\n"));
2290 printf (_(" --version Print version number, then exit\n"));
2291 printf (_(" -v, --verbose Print extra information while running\n"));
2292 printf ("\n");
2293 printf (_(" -M Print all dependencies to stdout;\n"
2294 " suppress ordinary output\n"));
2295 printf (_(" -MM Print non-system dependencies to stdout;\n"
2296 " suppress ordinary output\n"));
2297 printf (_(" -MD Print all dependencies to stdout\n"));
2298 printf (_(" -MMD Print non-system dependencies to stdout\n"));
2299 /* We omit -MG until it is implemented. */
2300 printf ("\n");
2301 printf (_("For bug reporting instructions, please see:\n"
2302 "%s.\n"), bug_report_url);
2303 exit (0);
2306 static void
2307 version (void)
2309 printf ("gcjh (GCC) %s\n\n", version_string);
2310 printf ("Copyright %s 2004 Free Software Foundation, Inc.\n", _("(C)"));
2311 printf (_("This is free software; see the source for copying conditions. There is NO\n"
2312 "warranty; not even for MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.\n\n"));
2313 exit (0);
2317 main (int argc, char** argv)
2319 JCF jcf;
2320 int argi;
2321 char *output_file = NULL;
2322 int emit_dependencies = 0, suppress_output = 0;
2323 int opt;
2325 gcc_init_libintl ();
2327 if (argc <= 1)
2329 error ("no classes specified");
2330 usage ();
2333 jcf_path_init ();
2335 /* We use getopt_long_only to allow single `-' long options. For
2336 some of our options this is more natural. */
2337 while ((opt = getopt_long_only (argc, argv, "I:d:o:v", options, NULL)) != -1)
2339 switch (opt)
2341 case 0:
2342 /* Already handled. */
2343 break;
2345 case 'o':
2346 output_file = optarg;
2347 break;
2349 case 'd':
2350 output_directory = optarg;
2351 break;
2353 case 'I':
2354 jcf_path_include_arg (optarg);
2355 break;
2357 case 'v':
2358 verbose++;
2359 break;
2361 case OPT_classpath:
2362 jcf_path_classpath_arg (optarg);
2363 break;
2365 case OPT_bootclasspath:
2366 jcf_path_bootclasspath_arg (optarg);
2367 break;
2369 case OPT_extdirs:
2370 jcf_path_extdirs_arg (optarg);
2371 break;
2373 case OPT_HELP:
2374 help ();
2375 break;
2377 case OPT_TEMP:
2378 temp_directory = optarg;
2379 break;
2381 case OPT_VERSION:
2382 version ();
2383 break;
2385 case OPT_PREPEND:
2386 if (prepend_count == 0)
2387 prepend_specs = ALLOC (argc * sizeof (char*));
2388 prepend_specs[prepend_count++] = optarg;
2389 break;
2391 case OPT_FRIEND:
2392 if (friend_count == 0)
2393 friend_specs = ALLOC (argc * sizeof (char*));
2394 friend_specs[friend_count++] = optarg;
2395 break;
2397 case OPT_ADD:
2398 if (add_count == 0)
2399 add_specs = ALLOC (argc * sizeof (char*));
2400 add_specs[add_count++] = optarg;
2401 break;
2403 case OPT_APPEND:
2404 if (append_count == 0)
2405 append_specs = ALLOC (argc * sizeof (char*));
2406 append_specs[append_count++] = optarg;
2407 break;
2409 case OPT_M:
2410 emit_dependencies = 1;
2411 suppress_output = 1;
2412 jcf_dependency_init (1);
2413 break;
2415 case OPT_MM:
2416 emit_dependencies = 1;
2417 suppress_output = 1;
2418 jcf_dependency_init (0);
2419 break;
2421 case OPT_MG:
2422 error ("`-MG' option is unimplemented");
2423 exit (1);
2425 case OPT_MD:
2426 emit_dependencies = 1;
2427 jcf_dependency_init (1);
2428 break;
2430 case OPT_MMD:
2431 emit_dependencies = 1;
2432 jcf_dependency_init (0);
2433 break;
2435 default:
2436 usage ();
2437 break;
2441 if (optind == argc)
2443 error ("no classes specified");
2444 usage ();
2447 jcf_path_seal (verbose);
2449 if (output_file && emit_dependencies)
2451 error ("can't specify both -o and -MD");
2452 exit (1);
2455 for (argi = optind; argi < argc; argi++)
2457 char *classname = argv[argi];
2458 char *current_output_file;
2459 const char *classfile_name;
2461 if (verbose)
2462 printf (_("Processing %s\n"), classname);
2463 if (! output_file)
2464 jcf_dependency_reset ();
2465 classfile_name = find_class (classname, strlen (classname), &jcf, 0);
2466 if (classfile_name == NULL)
2468 error ("%s: no such class", classname);
2469 exit (1);
2471 if (verbose)
2472 printf (_("Found in %s\n"), classfile_name);
2473 if (output_file)
2475 if (strcmp (output_file, "-") == 0)
2476 out = stdout;
2477 else if (out == NULL)
2479 out = fopen (output_file, "w");
2481 if (out == NULL)
2483 perror (output_file);
2484 exit (1);
2486 current_output_file = output_file;
2488 else
2490 int dir_len = strlen (output_directory);
2491 int i, classname_length = strlen (classname);
2492 current_output_file = ALLOC (dir_len + classname_length + 5);
2493 strcpy (current_output_file, output_directory);
2494 if (dir_len > 0 && output_directory[dir_len-1] != '/')
2495 current_output_file[dir_len++] = '/';
2496 for (i = 0; classname[i] != '\0'; i++)
2498 char ch = classname[i];
2499 if (ch == '.')
2500 ch = '/';
2501 if (flag_jni && ch == '/')
2502 ch = '_';
2503 current_output_file[dir_len++] = ch;
2505 if (emit_dependencies)
2507 if (suppress_output)
2509 jcf_dependency_set_dep_file ("-");
2510 out = NULL;
2512 else
2514 /* We use `.hd' and not `.d' to avoid clashes with
2515 dependency tracking from straight compilation. */
2516 strcpy (current_output_file + dir_len, ".hd");
2517 jcf_dependency_set_dep_file (current_output_file);
2520 strcpy (current_output_file + dir_len,
2521 stubs ? (flag_jni ? ".c" : ".cc") : ".h");
2522 jcf_dependency_set_target (current_output_file);
2523 if (! suppress_output)
2525 out = fopen (current_output_file, "w");
2526 if (out == NULL)
2528 perror (current_output_file);
2529 exit (1);
2533 process_file (&jcf, out);
2534 JCF_FINISH (&jcf);
2535 if (current_output_file != output_file)
2536 free (current_output_file);
2537 jcf_dependency_write ();
2540 if (out != NULL && out != stdout)
2541 fclose (out);
2543 return found_error;