/* src/cacaoh/headers.c - functions for header generation Copyright (C) 1996-2005, 2006, 2007 R. Grafl, A. Krall, C. Kruegel, C. Oates, R. Obermaisser, M. Platter, M. Probst, S. Ring, E. Steiner, C. Thalinger, D. Thuernbeck, P. Tomsich, C. Ullrich, J. Wenninger, Institut f. Computersprachen - TU Wien This file is part of CACAO. This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. */ #include "config.h" #include #include #include #include #include #include #if defined(ENABLE_THREADS) # if defined(__DARWIN__) # include # endif # include #endif #include "mm/gc-common.h" #include "mm/memory.h" #include "toolbox/chain.h" #include "toolbox/logging.h" #include "vm/builtin.h" #include "vm/global.h" #include "vm/stringlocal.h" #include "vmcore/class.h" #include "vmcore/method.h" #include "vmcore/loader.h" #include "vmcore/options.h" /************************ global variables **********************/ #define ACC_NATIVELY_OVERLOADED 0x10000000 #if defined(ENABLE_HANDLES) # define HEAP_PREFIX "heap_" #else # define HEAP_PREFIX "" #endif chain *ident_chain; /* chain with method and field names in current class */ FILE *file = NULL; static uint32_t outputsize; static bool dopadding; static void printIDpart(int c) { if ((c >= 'a' && c <= 'z') || (c >= 'A' && c <= 'Z') || (c >= '0' && c <= '9') || (c == '_')) putc(c, file); else putc('_', file); } void printID(utf *u) { char *utf_ptr = u->text; int i; for (i = 0; i < utf_get_number_of_u2s(u); i++) printIDpart(utf_nextu2(&utf_ptr)); } static void addoutputsize(int len) { uint32_t newsize; int32_t i; if (!dopadding) return; newsize = MEMORY_ALIGN(outputsize, len); for (i = outputsize; i < newsize; i++) fprintf(file, " uint8_t pad%d\n", (int) i); outputsize = newsize; } void printOverloadPart(utf *desc) { char *utf_ptr=desc->text; uint16_t c; fprintf(file, "__"); while ((c = utf_nextu2(&utf_ptr)) != ')') { switch (c) { case 'I': case 'S': case 'B': case 'C': case 'Z': case 'J': case 'F': case 'D': fprintf(file, "%c", (char) c); break; case '[': fprintf(file, "_3"); break; case 'L': putc('L', file); while ((c = utf_nextu2(&utf_ptr)) != ';') printIDpart(c); fprintf(file, "_2"); break; case '(': break; default: log_text("invalid method descriptor"); assert(0); } } } static char *printtype(char *utf_ptr, char *prefix, char *infix) { uint16_t c; switch (utf_nextu2(&utf_ptr)) { case 'V': fprintf(file, "void"); break; case 'I': case 'S': case 'B': case 'C': case 'Z': addoutputsize(4); fprintf(file, "int32_t"); break; case 'J': addoutputsize(8); fprintf(file, "int64_t"); break; case 'F': addoutputsize(4); fprintf(file, "float"); break; case 'D': addoutputsize(8); fprintf(file, "double"); break; case '[': addoutputsize ( sizeof(java_array_t*) ); switch (utf_nextu2(&utf_ptr)) { case 'I': fprintf (file, "java%s_intarray_t*", infix); break; case 'J': fprintf (file, "java%s_longarray_t*", infix); break; case 'Z': fprintf (file, "java%s_booleanarray_t*", infix); break; case 'B': fprintf (file, "java%s_bytearray_t*", infix); break; case 'S': fprintf (file, "java%s_shortarray_t*", infix); break; case 'C': fprintf (file, "java%s_chararray_t*", infix); break; case 'F': fprintf (file, "java%s_floatarray_t*", infix); break; case 'D': fprintf (file, "java%s_doublearray_t*", infix); break; case '[': fprintf(file, "java%s_objectarray_t*", infix); while ((c = utf_nextu2(&utf_ptr)) == '['); if (c == 'L') while (utf_nextu2(&utf_ptr) != ';'); break; case 'L': fprintf(file, "java%s_objectarray_t*", infix); while (utf_nextu2(&utf_ptr) != ';'); break; default: log_text("invalid type descriptor"); assert(0); } break; case 'L': addoutputsize ( sizeof(java_object_t*)); fprintf (file, "struct %s", prefix); while ( (c = utf_nextu2(&utf_ptr)) != ';' ) printIDpart (c); fprintf (file, "*"); break; default: log_text("Unknown type in field descriptor"); assert(0); } return utf_ptr; } /***** determine the number of entries of a utf string in the ident chain *****/ static int searchidentchain_utf(utf *ident) { utf *u = chain_first(ident_chain); /* first element of list */ int count = 0; while (u) { if (u==ident) count++; /* string found */ u = chain_next(ident_chain); /* next element in list */ } return count; } /************** print structure for direct access to objects ******************/ static void printfields(classinfo *c) { int32_t i; fieldinfo *f; int ident_count; if (!c) { addoutputsize(sizeof(java_object_t)); fprintf(file, " java_object_t header;\n"); return; } printfields(c->super.cls); for (i = 0; i < c->fieldscount; i++) { f = &(c->fields[i]); if (!(f->flags & ACC_STATIC)) { fprintf(file, " "); printtype(f->descriptor->text, HEAP_PREFIX, ""); fprintf(file, " "); utf_fprint_printable_ascii(file, f->name); /* rename multiple fieldnames */ if ((ident_count = searchidentchain_utf(f->name))) fprintf(file, "%d", ident_count - 1); chain_addlast(ident_chain, f->name); fprintf(file, ";\n"); } } } /***************** store prototype for native method in file ******************/ void printmethod(methodinfo *m) { char *utf_ptr; int32_t paramnum = 1; /* search for return-type in descriptor */ utf_ptr = m->descriptor->text; while (utf_nextu2(&utf_ptr) != ')'); /* create remarks */ fprintf(file, "\n/*\n * Class: "); utf_fprint_printable_ascii(file, m->class->name); fprintf(file, "\n * Method: "); utf_fprint_printable_ascii(file, m->name); fprintf(file, "\n * Signature: "); utf_fprint_printable_ascii(file, m->descriptor); fprintf(file, "\n */\n"); /* create prototype */ fprintf(file, "JNIEXPORT "); printtype(utf_ptr, "", "_handle"); fprintf(file, " JNICALL Java_"); printID(m->class->name); chain_addlast(ident_chain, m->name); fprintf(file, "_"); printID(m->name); /* ATTENTION: We use a dummy flag here. */ if (m->flags & ACC_NATIVELY_OVERLOADED) printOverloadPart(m->descriptor); fprintf(file, "(JNIEnv *env"); utf_ptr = m->descriptor->text + 1; if (!(m->flags & ACC_STATIC)) { fprintf(file, ", struct "); printID(m->class->name); fprintf(file, "* this"); } else { fprintf(file, ", jclass clazz"); } if ((*utf_ptr) != ')') fprintf(file, ", "); while ((*utf_ptr) != ')') { utf_ptr = printtype(utf_ptr, "", "_handle"); fprintf(file, " par%d", paramnum++); if ((*utf_ptr)!=')') fprintf(file, ", "); } fprintf(file, ");\n\n"); } /******* remove package-name in fully-qualified classname *********************/ void gen_header_filename(char *buffer, utf *u) { int32_t i; for (i = 0; i < utf_get_number_of_u2s(u); i++) { if ((u->text[i] == '/') || (u->text[i] == '$')) { buffer[i] = '_'; /* convert '$' and '/' to '_' */ } else { buffer[i] = u->text[i]; } } buffer[utf_get_number_of_u2s(u)] = '\0'; } /* create headerfile for classes and store native methods in chain ************/ void headerfile_generate(classinfo *c, char *opt_directory) { char header_filename[1024] = ""; char classname[1024]; char uclassname[1024]; int32_t i; methodinfo *m; int32_t j; methodinfo *m2; bool nativelyoverloaded; /* prevent compiler warnings */ nativelyoverloaded = false; /* open headerfile for class */ gen_header_filename(classname, c->name); /* create chain for renaming fields */ ident_chain = chain_new(); if (opt_directory) { sprintf(header_filename, "%s/%s.h", opt_directory, classname); } else { sprintf(header_filename, "%s.h", classname); } file = fopen(header_filename, "w"); if (!file) { log_text("Can not open file to store header information"); assert(0); } fprintf(file, "/* This file is machine generated, don't edit it! */\n\n"); /* convert to uppercase */ for (i = 0; classname[i]; i++) { uclassname[i] = toupper(classname[i]); } uclassname[i] = '\0'; fprintf(file, "#ifndef _%s_H\n#define _%s_H\n\n", uclassname, uclassname); /* create structure for direct access to objects */ fprintf(file, "/* Structure information for class: "); utf_fprint_printable_ascii(file, c->name); fprintf(file, " */\n\n"); fprintf(file, "typedef struct %s", HEAP_PREFIX); printID(c->name); fprintf(file, " {\n"); outputsize = 0; dopadding = true; printfields(c); fprintf(file, "} %s", HEAP_PREFIX); printID(c->name); fprintf(file, ";\n\n"); #if defined(ENABLE_HANDLES) /* create structure for indirection cell */ fprintf(file, "typedef struct "); printID(c->name); fprintf(file, " {\n"); fprintf(file, " %s", HEAP_PREFIX); printID(c->name); fprintf(file, " *heap_object;\n"); fprintf(file, "} "); printID(c->name); fprintf(file, ";\n\n"); #endif /* create chain for renaming overloaded methods */ chain_free(ident_chain); ident_chain = chain_new(); /* create method-prototypes */ /* find overloaded methods */ for (i = 0; i < c->methodscount; i++) { m = &(c->methods[i]); if (!(m->flags & ACC_NATIVE)) continue; /* We use a dummy flag here. */ if (!(m->flags & ACC_NATIVELY_OVERLOADED)) { nativelyoverloaded = false; for (j = i + 1; j < c->methodscount; j++) { m2 = &(c->methods[j]); if (!(m2->flags & ACC_NATIVE)) continue; if (m->name == m2->name) { m2->flags |= ACC_NATIVELY_OVERLOADED; nativelyoverloaded = true; } } } if (nativelyoverloaded == true) m->flags |= ACC_NATIVELY_OVERLOADED; } for (i = 0; i < c->methodscount; i++) { m = &(c->methods[i]); if (m->flags & ACC_NATIVE) printmethod(m); } chain_free(ident_chain); fprintf(file, "#endif\n\n"); fclose(file); } /******** print classname, '$' used to seperate inner-class name ***********/ void print_classname(classinfo *clazz) { utf *u = clazz->name; char *endpos = u->text + u->blength; char *utf_ptr = u->text; uint16_t c; while (utf_ptr < endpos) { if ((c = utf_nextu2(&utf_ptr)) == '_') putc('$', file); else putc(c, file); } } /* * These are local overrides for various environment variables in Emacs. * Please do not remove this and leave it at the end of the file, where * Emacs will automagically detect them. * --------------------------------------------------------------------- * Local variables: * mode: c * indent-tabs-mode: t * c-basic-offset: 4 * tab-width: 4 * End: * vim:noexpandtab:sw=4:ts=4: */