1 /* src/native/vm/gnu/java_lang_VMClassLoader.c
3 Copyright (C) 1996-2005, 2006, 2007, 2008
4 CACAOVM - Verein zur Foerderung der freien virtuellen Maschine CACAO
6 This file is part of CACAO.
8 This program is free software; you can redistribute it and/or
9 modify it under the terms of the GNU General Public License as
10 published by the Free Software Foundation; either version 2, or (at
11 your option) any later version.
13 This program is distributed in the hope that it will be useful, but
14 WITHOUT ANY WARRANTY; without even the implied warranty of
15 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
16 General Public License for more details.
18 You should have received a copy of the GNU General Public License
19 along with this program; if not, write to the Free Software
20 Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
32 #include "mm/memory.h"
34 #include "native/jni.h"
35 #include "native/llni.h"
36 #include "native/native.h"
37 #include "native/include/java_lang_Class.h"
38 #include "native/include/java_lang_String.h"
39 #include "native/include/java_security_ProtectionDomain.h" /* required by... */
40 #include "native/include/java_lang_ClassLoader.h"
41 #include "native/include/java_util_Vector.h"
42 #include "native/include/java_util_HashMap.h"
43 #include "native/include/java_util_Map.h"
44 #include "native/include/java_lang_Boolean.h"
46 #include "native/include/java_lang_VMClassLoader.h"
48 #include "toolbox/logging.h"
49 #include "toolbox/list.h"
51 #if defined(ENABLE_ASSERTION)
52 #include "vm/assertion.h"
55 #include "vm/builtin.h"
56 #include "vm/exceptions.h"
57 #include "vm/initialize.h"
58 #include "vm/primitive.h"
59 #include "vm/stringlocal.h"
62 #include "vm/jit/asmpart.h"
64 #include "vmcore/class.h"
65 #include "vmcore/classcache.h"
66 #include "vmcore/linker.h"
67 #include "vmcore/loader.h"
68 #include "vmcore/options.h"
69 #include "vmcore/statistics.h"
70 #include "vmcore/suck.h"
71 #include "vmcore/zip.h"
73 #if defined(ENABLE_JVMTI)
74 #include "native/jvmti/cacaodbg.h"
77 /* native methods implemented by this file ************************************/
79 static JNINativeMethod methods[] = {
80 { "defineClass", "(Ljava/lang/ClassLoader;Ljava/lang/String;[BIILjava/security/ProtectionDomain;)Ljava/lang/Class;", (void *) (uintptr_t) &Java_java_lang_VMClassLoader_defineClass },
81 { "getPrimitiveClass", "(C)Ljava/lang/Class;", (void *) (uintptr_t) &Java_java_lang_VMClassLoader_getPrimitiveClass },
82 { "resolveClass", "(Ljava/lang/Class;)V", (void *) (uintptr_t) &Java_java_lang_VMClassLoader_resolveClass },
83 { "loadClass", "(Ljava/lang/String;Z)Ljava/lang/Class;", (void *) (uintptr_t) &Java_java_lang_VMClassLoader_loadClass },
84 { "nativeGetResources", "(Ljava/lang/String;)Ljava/util/Vector;", (void *) (uintptr_t) &Java_java_lang_VMClassLoader_nativeGetResources },
85 { "defaultAssertionStatus", "()Z", (void *) (uintptr_t) &Java_java_lang_VMClassLoader_defaultAssertionStatus },
86 { "defaultUserAssertionStatus", "()Z", (void *) (uintptr_t) &Java_java_lang_VMClassLoader_defaultUserAssertionStatus },
87 { "packageAssertionStatus0", "(Ljava/lang/Boolean;Ljava/lang/Boolean;)Ljava/util/Map;", (void *) (uintptr_t) &Java_java_lang_VMClassLoader_packageAssertionStatus0 },
88 { "classAssertionStatus0", "(Ljava/lang/Boolean;Ljava/lang/Boolean;)Ljava/util/Map;", (void *) (uintptr_t) &Java_java_lang_VMClassLoader_classAssertionStatus0 },
89 { "findLoadedClass", "(Ljava/lang/ClassLoader;Ljava/lang/String;)Ljava/lang/Class;", (void *) (uintptr_t) &Java_java_lang_VMClassLoader_findLoadedClass },
93 /* _Jv_java_lang_VMClassLoader_init ********************************************
95 Register native functions.
97 *******************************************************************************/
99 void _Jv_java_lang_VMClassLoader_init(void)
103 u = utf_new_char("java/lang/VMClassLoader");
105 native_method_register(u, methods, NATIVE_METHODS_COUNT);
110 * Class: java/lang/VMClassLoader
111 * Method: defineClass
112 * Signature: (Ljava/lang/ClassLoader;Ljava/lang/String;[BIILjava/security/ProtectionDomain;)Ljava/lang/Class;
114 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, int32_t offset, int32_t len, java_security_ProtectionDomain *pd)
118 classloader_t *loader;
121 #if defined(ENABLE_JVMTI)
122 jint new_class_data_len = 0;
123 unsigned char* new_class_data = NULL;
126 /* check if data was passed */
129 exceptions_throw_nullpointerexception();
133 /* check the indexes passed */
135 if ((offset < 0) || (len < 0) || ((offset + len) > LLNI_array_size(data))) {
136 exceptions_throw_arrayindexoutofboundsexception();
140 /* add classloader to classloader hashtable */
142 loader = loader_hashtable_classloader_add((java_handle_t *) cl);
145 /* convert '.' to '/' in java string */
147 utfname = javastring_toutf((java_handle_t *) name, true);
153 #if defined(ENABLE_JVMTI)
154 /* XXX again this will not work because of the indirection cell for classloaders */
156 /* fire Class File Load Hook JVMTI event */
159 jvmti_ClassFileLoadHook(utfname, len, (unsigned char *) data->data,
160 loader, (java_handle_t *) pd,
161 &new_class_data_len, &new_class_data);
164 /* define the class */
166 #if defined(ENABLE_JVMTI)
167 /* check if the JVMTI wants to modify the class */
169 if (new_class_data == NULL)
170 c = class_define(utfname, loader, new_class_data_len, new_class_data, pd);
173 c = class_define(utfname, loader, len, (uint8_t *) &LLNI_array_direct(data, offset), (java_handle_t *) pd);
178 /* for convenience */
180 o = LLNI_classinfo_wrap(c);
182 #if defined(WITH_CLASSPATH_GNU)
183 /* set ProtectionDomain */
185 LLNI_field_set_ref(o, pd, pd);
193 * Class: java/lang/VMClassLoader
194 * Method: getPrimitiveClass
195 * Signature: (C)Ljava/lang/Class;
197 JNIEXPORT java_lang_Class* JNICALL Java_java_lang_VMClassLoader_getPrimitiveClass(JNIEnv *env, jclass clazz, int32_t type)
201 c = primitive_class_get_by_char(type);
204 exceptions_throw_classnotfoundexception(utf_null);
208 return LLNI_classinfo_wrap(c);
213 * Class: java/lang/VMClassLoader
214 * Method: resolveClass
215 * Signature: (Ljava/lang/Class;)V
217 JNIEXPORT void JNICALL Java_java_lang_VMClassLoader_resolveClass(JNIEnv *env, jclass clazz, java_lang_Class *c)
221 ci = LLNI_classinfo_unwrap(c);
224 exceptions_throw_nullpointerexception();
230 if (!(ci->state & CLASS_LINKED))
231 (void) link_class(ci);
238 * Class: java/lang/VMClassLoader
240 * Signature: (Ljava/lang/String;Z)Ljava/lang/Class;
242 JNIEXPORT java_lang_Class* JNICALL Java_java_lang_VMClassLoader_loadClass(JNIEnv *env, jclass clazz, java_lang_String *name, int32_t resolve)
248 exceptions_throw_nullpointerexception();
252 /* create utf string in which '.' is replaced by '/' */
254 u = javastring_toutf((java_handle_t *) name, true);
258 c = load_class_bootstrap(u);
263 /* resolve class -- if requested */
269 return LLNI_classinfo_wrap(c);
274 * Class: java/lang/VMClassLoader
275 * Method: nativeGetResources
276 * Signature: (Ljava/lang/String;)Ljava/util/Vector;
278 JNIEXPORT java_util_Vector* JNICALL Java_java_lang_VMClassLoader_nativeGetResources(JNIEnv *env, jclass clazz, java_lang_String *name)
280 java_handle_t *o; /* vector being created */
281 methodinfo *m; /* "add" method of vector */
282 java_handle_t *path; /* path to be added */
283 list_classpath_entry *lce; /* classpath entry */
284 utf *utfname; /* utf to look for */
285 char *buffer; /* char buffer */
286 char *namestart; /* start of name to use */
287 char *tmppath; /* temporary buffer */
288 int32_t namelen; /* length of name to use */
289 int32_t searchlen; /* length of name to search */
290 int32_t bufsize; /* size of buffer allocated */
291 int32_t pathlen; /* name of path to assemble */
292 struct stat buf; /* buffer for stat */
293 jboolean ret; /* return value of "add" */
295 /* get the resource name as utf string */
297 utfname = javastring_toutf((java_handle_t *) name, false);
302 /* copy it to a char buffer */
304 namelen = utf_bytes(utfname);
306 bufsize = namelen + strlen("0");
307 buffer = MNEW(char, bufsize);
309 utf_copy(buffer, utfname);
312 /* skip leading '/' */
314 if (namestart[0] == '/') {
320 /* remove trailing `.class' */
322 if (namelen >= 6 && strcmp(namestart + (namelen - 6), ".class") == 0) {
326 /* create a new needle to look for, if necessary */
328 if (searchlen != bufsize-1) {
329 utfname = utf_new(namestart, searchlen);
336 o = native_new_and_init(class_java_util_Vector);
341 /* get Vector.add() method */
343 m = class_resolveclassmethod(class_java_util_Vector,
345 utf_new_char("(Ljava/lang/Object;)Z"),
352 /* iterate over all classpath entries */
354 for (lce = list_first(list_classpath_entries); lce != NULL;
355 lce = list_next(list_classpath_entries, lce)) {
356 /* clear path pointer */
359 #if defined(ENABLE_ZLIB)
360 if (lce->type == CLASSPATH_ARCHIVE) {
362 if (zip_find(lce, utfname)) {
363 pathlen = strlen("jar:file://") + lce->pathlen + strlen("!/") +
364 namelen + strlen("0");
366 tmppath = MNEW(char, pathlen);
368 sprintf(tmppath, "jar:file://%s!/%s", lce->path, namestart);
369 path = javastring_new_from_utf_string(tmppath),
371 MFREE(tmppath, char, pathlen);
375 #endif /* defined(ENABLE_ZLIB) */
376 pathlen = strlen("file://") + lce->pathlen + namelen + strlen("0");
378 tmppath = MNEW(char, pathlen);
380 sprintf(tmppath, "file://%s%s", lce->path, namestart);
382 /* Does this file exist? */
384 if (stat(tmppath + strlen("file://") - 1, &buf) == 0)
385 if (!S_ISDIR(buf.st_mode))
386 path = javastring_new_from_utf_string(tmppath);
388 MFREE(tmppath, char, pathlen);
389 #if defined(ENABLE_ZLIB)
393 /* if a resource was found, add it to the vector */
396 ret = vm_call_method_int(m, o, path);
398 if (exceptions_get_exception() != NULL)
406 MFREE(buffer, char, bufsize);
408 return (java_util_Vector *) o;
411 MFREE(buffer, char, bufsize);
418 * Class: java/lang/VMClassLoader
419 * Method: defaultAssertionStatus
422 JNIEXPORT int32_t JNICALL Java_java_lang_VMClassLoader_defaultAssertionStatus(JNIEnv *env, jclass clazz)
424 #if defined(ENABLE_ASSERTION)
425 return assertion_system_enabled;
432 * Class: java/lang/VMClassLoader
433 * Method: userAssertionStatus
436 JNIEXPORT int32_t JNICALL Java_java_lang_VMClassLoader_defaultUserAssertionStatus(JNIEnv *env, jclass clazz)
438 #if defined(ENABLE_ASSERTION)
439 return assertion_user_enabled;
446 * Class: java/lang/VMClassLoader
447 * Method: packageAssertionStatus
448 * Signature: ()Ljava_util_Map;
450 JNIEXPORT java_util_Map* JNICALL Java_java_lang_VMClassLoader_packageAssertionStatus0(JNIEnv *env, jclass clazz, java_lang_Boolean *jtrue, java_lang_Boolean *jfalse)
453 #if defined(ENABLE_ASSERTION)
456 assertion_name_t *item;
461 hm = native_new_and_init(class_java_util_HashMap);
466 #if defined(ENABLE_ASSERTION)
467 /* if nothing todo, return now */
469 if (assertion_package_count == 0) {
470 return (java_util_Map *) hm;
473 /* get HashMap.put method */
475 m = class_resolveclassmethod(class_java_util_HashMap,
477 utf_new_char("(Ljava/lang/Object;Ljava/lang/Object;)Ljava/lang/Object;"),
485 item = (assertion_name_t *)list_first(list_assertion_names);
487 while (item != NULL) {
488 if (item->package == false) {
489 item = (assertion_name_t *)list_next(list_assertion_names, item);
493 if (strcmp(item->name, "") == 0) {
494 /* unnamed package wanted */
498 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;
519 * Class: java/lang/VMClassLoader
520 * Method: classAssertionStatus
521 * Signature: ()Ljava_util_Map;
523 JNIEXPORT java_util_Map* JNICALL Java_java_lang_VMClassLoader_classAssertionStatus0(JNIEnv *env, jclass clazz, java_lang_Boolean *jtrue, java_lang_Boolean *jfalse)
526 #if defined(ENABLE_ASSERTION)
529 assertion_name_t *item;
534 hm = native_new_and_init(class_java_util_HashMap);
539 #if defined(ENABLE_ASSERTION)
540 /* if nothing todo, return now */
542 if (assertion_class_count == 0) {
543 return (java_util_Map *) hm;
546 /* get HashMap.put method */
548 m = class_resolveclassmethod(class_java_util_HashMap,
550 utf_new_char("(Ljava/lang/Object;Ljava/lang/Object;)Ljava/lang/Object;"),
558 item = (assertion_name_t *)list_first(list_assertion_names);
560 while (item != NULL) {
561 if (item->package == true) {
562 item = (assertion_name_t *)list_next(list_assertion_names, item);
566 js = javastring_new_from_ascii(item->name);
571 if (item->enabled == true) {
572 vm_call_method(m, hm, js, jtrue);
575 vm_call_method(m, hm, js, jfalse);
578 item = (assertion_name_t *)list_next(list_assertion_names, item);
582 return (java_util_Map *) hm;
587 * Class: java/lang/VMClassLoader
588 * Method: findLoadedClass
589 * Signature: (Ljava/lang/ClassLoader;Ljava/lang/String;)Ljava/lang/Class;
591 JNIEXPORT java_lang_Class* JNICALL Java_java_lang_VMClassLoader_findLoadedClass(JNIEnv *env, jclass clazz, java_lang_ClassLoader *loader, java_lang_String *name)
597 /* XXX is it correct to add the classloader to the hashtable here? */
599 cl = loader_hashtable_classloader_add((java_handle_t *) loader);
601 /* replace `.' by `/', this is required by the classcache */
603 u = javastring_toutf((java_handle_t *) name, true);
605 /* lookup for defining classloader */
607 c = classcache_lookup_defined(cl, u);
609 /* if not found, lookup for initiating classloader */
612 c = classcache_lookup(cl, u);
614 return LLNI_classinfo_wrap(c);
619 * These are local overrides for various environment variables in Emacs.
620 * Please do not remove this and leave it at the end of the file, where
621 * Emacs will automagically detect them.
622 * ---------------------------------------------------------------------
625 * indent-tabs-mode: t
629 * vim:noexpandtab:sw=4:ts=4: