1 /* src/native/vm/gnu/VMClassLoader.c
3 Copyright (C) 1996-2005, 2006, 2007 R. Grafl, A. Krall, C. Kruegel,
4 C. Oates, R. Obermaisser, M. Platter, M. Probst, S. Ring,
5 E. Steiner, C. Thalinger, D. Thuernbeck, P. Tomsich, C. Ullrich,
6 J. Wenninger, Institut f. Computersprachen - TU Wien
8 This file is part of CACAO.
10 This program is free software; you can redistribute it and/or
11 modify it under the terms of the GNU General Public License as
12 published by the Free Software Foundation; either version 2, or (at
13 your option) any later version.
15 This program is distributed in the hope that it will be useful, but
16 WITHOUT ANY WARRANTY; without even the implied warranty of
17 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
18 General Public License for more details.
20 You should have received a copy of the GNU General Public License
21 along with this program; if not, write to the Free Software
22 Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
35 #include "mm/memory.h"
37 #include "native/jni.h"
38 #include "native/llni.h"
39 #include "native/native.h"
40 #include "native/include/java_lang_Class.h"
41 #include "native/include/java_lang_String.h"
42 #include "native/include/java_security_ProtectionDomain.h" /* required by... */
43 #include "native/include/java_lang_ClassLoader.h"
44 #include "native/include/java_util_Vector.h"
45 #include "native/include/java_util_HashMap.h"
46 #include "native/include/java_util_Map.h"
47 #include "native/include/java_lang_Boolean.h"
49 #include "native/include/java_lang_VMClassLoader.h"
51 #include "native/vm/java_lang_ClassLoader.h"
53 #include "toolbox/logging.h"
54 #include "toolbox/list.h"
56 #if defined(ENABLE_ASSERTION)
57 #include "vm/assertion.h"
60 #include "vm/builtin.h"
61 #include "vm/exceptions.h"
62 #include "vm/initialize.h"
63 #include "vm/primitive.h"
64 #include "vm/stringlocal.h"
67 #include "vm/jit/asmpart.h"
69 #include "vmcore/class.h"
70 #include "vmcore/classcache.h"
71 #include "vmcore/linker.h"
72 #include "vmcore/loader.h"
73 #include "vmcore/options.h"
74 #include "vmcore/statistics.h"
75 #include "vmcore/suck.h"
76 #include "vmcore/zip.h"
78 #if defined(ENABLE_JVMTI)
79 #include "native/jvmti/cacaodbg.h"
82 /* native methods implemented by this file ************************************/
84 static JNINativeMethod methods[] = {
85 { "defineClass", "(Ljava/lang/ClassLoader;Ljava/lang/String;[BIILjava/security/ProtectionDomain;)Ljava/lang/Class;", (void *) (ptrint) &Java_java_lang_VMClassLoader_defineClass },
86 { "getPrimitiveClass", "(C)Ljava/lang/Class;", (void *) (ptrint) &Java_java_lang_VMClassLoader_getPrimitiveClass },
87 { "resolveClass", "(Ljava/lang/Class;)V", (void *) (ptrint) &Java_java_lang_VMClassLoader_resolveClass },
88 { "loadClass", "(Ljava/lang/String;Z)Ljava/lang/Class;", (void *) (ptrint) &Java_java_lang_VMClassLoader_loadClass },
89 { "nativeGetResources", "(Ljava/lang/String;)Ljava/util/Vector;", (void *) (ptrint) &Java_java_lang_VMClassLoader_nativeGetResources },
90 { "defaultAssertionStatus", "()Z", (void *) (ptrint) &Java_java_lang_VMClassLoader_defaultAssertionStatus },
91 { "defaultUserAssertionStatus", "()Z", (void *) (ptrint) &Java_java_lang_VMClassLoader_defaultUserAssertionStatus },
92 { "packageAssertionStatus0", "(Ljava/lang/Boolean;Ljava/lang/Boolean;)Ljava/util/Map;", (void *) (ptrint) &Java_java_lang_VMClassLoader_packageAssertionStatus0 },
93 { "classAssertionStatus0", "(Ljava/lang/Boolean;Ljava/lang/Boolean;)Ljava/util/Map;", (void *) (ptrint) &Java_java_lang_VMClassLoader_classAssertionStatus0 },
94 { "findLoadedClass", "(Ljava/lang/ClassLoader;Ljava/lang/String;)Ljava/lang/Class;", (void *) (ptrint) &Java_java_lang_VMClassLoader_findLoadedClass },
98 /* _Jv_java_lang_VMClassLoader_init ********************************************
100 Register native functions.
102 *******************************************************************************/
104 void _Jv_java_lang_VMClassLoader_init(void)
108 u = utf_new_char("java/lang/VMClassLoader");
110 native_method_register(u, methods, NATIVE_METHODS_COUNT);
115 * Class: java/lang/VMClassLoader
116 * Method: defineClass
117 * Signature: (Ljava/lang/ClassLoader;Ljava/lang/String;[BIILjava/security/ProtectionDomain;)Ljava/lang/Class;
119 JNIEXPORT java_lang_Class* JNICALL Java_java_lang_VMClassLoader_defineClass(JNIEnv *env, jclass clazz, java_lang_ClassLoader *cl, java_lang_String *name, java_handle_bytearray_t *data, s4 offset, s4 len, java_security_ProtectionDomain *pd)
121 return _Jv_java_lang_ClassLoader_defineClass(cl, name, data, offset, len, pd);
126 * Class: java/lang/VMClassLoader
127 * Method: getPrimitiveClass
128 * Signature: (C)Ljava/lang/Class;
130 JNIEXPORT java_lang_Class* JNICALL Java_java_lang_VMClassLoader_getPrimitiveClass(JNIEnv *env, jclass clazz, s4 type)
134 c = primitive_class_get_by_char(type);
137 exceptions_throw_classnotfoundexception(utf_null);
141 return LLNI_classinfo_wrap(c);
146 * Class: java/lang/VMClassLoader
147 * Method: resolveClass
148 * Signature: (Ljava/lang/Class;)V
150 JNIEXPORT void JNICALL Java_java_lang_VMClassLoader_resolveClass(JNIEnv *env, jclass clazz, java_lang_Class *c)
154 ci = LLNI_classinfo_unwrap(c);
157 exceptions_throw_nullpointerexception();
163 if (!(ci->state & CLASS_LINKED))
164 (void) link_class(ci);
171 * Class: java/lang/VMClassLoader
173 * Signature: (Ljava/lang/String;Z)Ljava/lang/Class;
175 JNIEXPORT java_lang_Class* JNICALL Java_java_lang_VMClassLoader_loadClass(JNIEnv *env, jclass clazz, java_lang_String *name, s4 resolve)
181 exceptions_throw_nullpointerexception();
185 /* create utf string in which '.' is replaced by '/' */
187 u = javastring_toutf((java_handle_t *) name, true);
191 c = load_class_bootstrap(u);
196 /* resolve class -- if requested */
202 return LLNI_classinfo_wrap(c);
207 * Class: java/lang/VMClassLoader
208 * Method: nativeGetResources
209 * Signature: (Ljava/lang/String;)Ljava/util/Vector;
211 JNIEXPORT java_util_Vector* JNICALL Java_java_lang_VMClassLoader_nativeGetResources(JNIEnv *env, jclass clazz, java_lang_String *name)
213 java_handle_t *o; /* vector being created */
214 methodinfo *m; /* "add" method of vector */
215 java_handle_t *path; /* path to be added */
216 list_classpath_entry *lce; /* classpath entry */
217 utf *utfname; /* utf to look for */
218 char *buffer; /* char buffer */
219 char *namestart; /* start of name to use */
220 char *tmppath; /* temporary buffer */
221 s4 namelen; /* length of name to use */
222 s4 searchlen; /* length of name to search */
223 s4 bufsize; /* size of buffer allocated */
224 s4 pathlen; /* name of path to assemble */
225 struct stat buf; /* buffer for stat */
226 jboolean ret; /* return value of "add" */
228 /* get the resource name as utf string */
230 utfname = javastring_toutf((java_handle_t *) name, false);
235 /* copy it to a char buffer */
237 namelen = utf_bytes(utfname);
239 bufsize = namelen + strlen("0");
240 buffer = MNEW(char, bufsize);
242 utf_copy(buffer, utfname);
245 /* skip leading '/' */
247 if (namestart[0] == '/') {
253 /* remove trailing `.class' */
255 if (namelen >= 6 && strcmp(namestart + (namelen - 6), ".class") == 0) {
259 /* create a new needle to look for, if necessary */
261 if (searchlen != bufsize-1) {
262 utfname = utf_new(namestart, searchlen);
269 o = native_new_and_init(class_java_util_Vector);
274 /* get Vector.add() method */
276 m = class_resolveclassmethod(class_java_util_Vector,
278 utf_new_char("(Ljava/lang/Object;)Z"),
285 /* iterate over all classpath entries */
287 for (lce = list_first(list_classpath_entries); lce != NULL;
288 lce = list_next(list_classpath_entries, lce)) {
289 /* clear path pointer */
292 #if defined(ENABLE_ZLIB)
293 if (lce->type == CLASSPATH_ARCHIVE) {
295 if (zip_find(lce, utfname)) {
296 pathlen = strlen("jar:file://") + lce->pathlen + strlen("!/") +
297 namelen + strlen("0");
299 tmppath = MNEW(char, pathlen);
301 sprintf(tmppath, "jar:file://%s!/%s", lce->path, namestart);
302 path = javastring_new_from_utf_string(tmppath),
304 MFREE(tmppath, char, pathlen);
308 #endif /* defined(ENABLE_ZLIB) */
309 pathlen = strlen("file://") + lce->pathlen + namelen + strlen("0");
311 tmppath = MNEW(char, pathlen);
313 sprintf(tmppath, "file://%s%s", lce->path, namestart);
315 /* Does this file exist? */
317 if (stat(tmppath + strlen("file://") - 1, &buf) == 0)
318 if (!S_ISDIR(buf.st_mode))
319 path = javastring_new_from_utf_string(tmppath);
321 MFREE(tmppath, char, pathlen);
322 #if defined(ENABLE_ZLIB)
326 /* if a resource was found, add it to the vector */
329 ret = vm_call_method_int(m, o, path);
331 if (exceptions_get_exception() != NULL)
339 MFREE(buffer, char, bufsize);
341 return (java_util_Vector *) o;
344 MFREE(buffer, char, bufsize);
351 * Class: java/lang/VMClassLoader
352 * Method: defaultAssertionStatus
355 JNIEXPORT s4 JNICALL Java_java_lang_VMClassLoader_defaultAssertionStatus(JNIEnv *env, jclass clazz)
357 #if defined(ENABLE_ASSERTION)
358 return assertion_system_enabled;
365 * Class: java/lang/VMClassLoader
366 * Method: userAssertionStatus
369 JNIEXPORT s4 JNICALL Java_java_lang_VMClassLoader_defaultUserAssertionStatus(JNIEnv *env, jclass clazz)
371 #if defined(ENABLE_ASSERTION)
372 return assertion_user_enabled;
379 * Class: java/lang/VMClassLoader
380 * Method: packageAssertionStatus
381 * Signature: ()Ljava_util_Map;
383 JNIEXPORT java_util_Map* JNICALL Java_java_lang_VMClassLoader_packageAssertionStatus0(JNIEnv *env, jclass clazz, java_lang_Boolean *jtrue, java_lang_Boolean *jfalse)
386 #if defined(ENABLE_ASSERTION)
389 assertion_name_t *item;
394 hm = native_new_and_init(class_java_util_HashMap);
399 #if defined(ENABLE_ASSERTION)
400 /* if nothing todo, return now */
402 if (assertion_package_count == 0) {
403 return (java_util_Map *) hm;
406 /* get HashMap.put method */
408 m = class_resolveclassmethod(class_java_util_HashMap,
410 utf_new_char("(Ljava/lang/Object;Ljava/lang/Object;)Ljava/lang/Object;"),
418 item = (assertion_name_t *)list_first(list_assertion_names);
420 while (item != NULL) {
421 if (item->package == false) {
422 item = (assertion_name_t *)list_next(list_assertion_names, item);
426 if (strcmp(item->name, "") == 0) {
427 /* unnamed package wanted */
431 js = javastring_new_from_ascii(item->name);
437 if (item->enabled == true) {
438 vm_call_method(m, hm, js, jtrue);
441 vm_call_method(m, hm, js, jfalse);
444 item = (assertion_name_t *)list_next(list_assertion_names, item);
448 return (java_util_Map *) hm;
452 * Class: java/lang/VMClassLoader
453 * Method: classAssertionStatus
454 * Signature: ()Ljava_util_Map;
456 JNIEXPORT java_util_Map* JNICALL Java_java_lang_VMClassLoader_classAssertionStatus0(JNIEnv *env, jclass clazz, java_lang_Boolean *jtrue, java_lang_Boolean *jfalse)
459 #if defined(ENABLE_ASSERTION)
462 assertion_name_t *item;
467 hm = native_new_and_init(class_java_util_HashMap);
472 #if defined(ENABLE_ASSERTION)
473 /* if nothing todo, return now */
475 if (assertion_class_count == 0) {
476 return (java_util_Map *) hm;
479 /* get HashMap.put method */
481 m = class_resolveclassmethod(class_java_util_HashMap,
483 utf_new_char("(Ljava/lang/Object;Ljava/lang/Object;)Ljava/lang/Object;"),
491 item = (assertion_name_t *)list_first(list_assertion_names);
493 while (item != NULL) {
494 if (item->package == true) {
495 item = (assertion_name_t *)list_next(list_assertion_names, item);
499 js = javastring_new_from_ascii(item->name);
504 if (item->enabled == true) {
505 vm_call_method(m, hm, js, jtrue);
508 vm_call_method(m, hm, js, jfalse);
511 item = (assertion_name_t *)list_next(list_assertion_names, item);
515 return (java_util_Map *) hm;
520 * Class: java/lang/VMClassLoader
521 * Method: findLoadedClass
522 * Signature: (Ljava/lang/ClassLoader;Ljava/lang/String;)Ljava/lang/Class;
524 JNIEXPORT java_lang_Class* JNICALL Java_java_lang_VMClassLoader_findLoadedClass(JNIEnv *env, jclass clazz, java_lang_ClassLoader *loader, java_lang_String *name)
530 /* XXX is it correct to add the classloader to the hashtable here? */
532 cl = loader_hashtable_classloader_add((java_handle_t *) loader);
534 /* replace `.' by `/', this is required by the classcache */
536 u = javastring_toutf((java_handle_t *) name, true);
538 /* lookup for defining classloader */
540 c = classcache_lookup_defined(cl, u);
542 /* if not found, lookup for initiating classloader */
545 c = classcache_lookup(cl, u);
547 return LLNI_classinfo_wrap(c);
552 * These are local overrides for various environment variables in Emacs.
553 * Please do not remove this and leave it at the end of the file, where
554 * Emacs will automagically detect them.
555 * ---------------------------------------------------------------------
558 * indent-tabs-mode: t
562 * vim:noexpandtab:sw=4:ts=4: