1 /* src/vm/exceptions.c - exception related functions
3 Copyright (C) 1996-2005 R. Grafl, A. Krall, C. Kruegel, C. Oates,
4 R. Obermaisser, M. Platter, M. Probst, S. Ring, E. Steiner,
5 C. Thalinger, D. Thuernbeck, P. Tomsich, C. Ullrich, J. Wenninger,
6 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., 59 Temple Place - Suite 330, Boston, MA
25 Contact: cacao@complang.tuwien.ac.at
27 Authors: Christian Thalinger
31 $Id: exceptions.c 2458 2005-05-12 23:02:07Z twisti $
42 #include "mm/memory.h"
43 #include "native/native.h"
44 #include "native/include/java_lang_String.h"
45 #include "native/include/java_lang_Throwable.h"
46 #include "toolbox/logging.h"
47 #include "toolbox/util.h"
49 #include "vm/exceptions.h"
50 #include "vm/global.h"
51 #include "vm/loader.h"
52 #include "vm/options.h"
53 #include "vm/stringlocal.h"
54 #include "vm/tables.h"
55 #include "vm/jit/asmpart.h"
56 #include "vm/jit/jit.h"
59 /* for raising exceptions from native methods */
61 #if !defined(USE_THREADS) || !defined(NATIVE_THREADS)
62 java_objectheader* _exceptionptr = NULL;
63 u1 _dontfillinexceptionstacktrace = false;
67 /* init_system_exceptions ******************************************************
69 Load and link exceptions used in the system.
71 *******************************************************************************/
73 bool exceptions_init(void)
75 /* java/lang/Throwable */
77 if (!load_class_bootstrap(utf_java_lang_Throwable,
78 &class_java_lang_Throwable) ||
79 !link_class(class_java_lang_Throwable))
83 /* java/lang/VMThrowable */
85 if (!load_class_bootstrap(utf_java_lang_VMThrowable,
86 &class_java_lang_VMThrowable) ||
87 !link_class(class_java_lang_VMThrowable))
93 if (!load_class_bootstrap(utf_java_lang_Error, &class_java_lang_Error) ||
94 !link_class(class_java_lang_Error))
98 /* java/lang/Exception */
100 if (!load_class_bootstrap(utf_java_lang_Exception,
101 &class_java_lang_Exception) ||
102 !link_class(class_java_lang_Exception))
106 /* java/lang/NoClassDefFoundError */
108 if (!load_class_bootstrap(utf_java_lang_NoClassDefFoundError,
109 &class_java_lang_NoClassDefFoundError) ||
110 !link_class(class_java_lang_NoClassDefFoundError))
114 /* java/lang/OutOfMemoryError */
116 if (!load_class_bootstrap(utf_java_lang_OutOfMemoryError,
117 &class_java_lang_OutOfMemoryError) ||
118 !link_class(class_java_lang_OutOfMemoryError))
122 /* java/lang/ClassNotFoundException */
124 if (!load_class_bootstrap(utf_java_lang_ClassNotFoundException,
125 &class_java_lang_ClassNotFoundException) ||
126 !link_class(class_java_lang_ClassNotFoundException))
133 static void throw_exception_exit_intern(bool doexit)
135 java_objectheader *xptr;
139 xptr = *exceptionptr;
142 /* clear exception, because we are calling jit code again */
143 *exceptionptr = NULL;
145 c = xptr->vftbl->class;
147 pss = class_resolveclassmethod(c,
150 class_java_lang_Object,
153 /* print the stacktrace */
155 asm_calljavafunction(pss, xptr, NULL, NULL, NULL);
157 /* this normally means, we are EXTREMLY out of memory, but may be
158 any other exception */
160 utf_fprint_classname(stderr, c->name);
161 fprintf(stderr, "\n");
165 utf_fprint_classname(stderr, c->name);
166 fprintf(stderr, ": printStackTrace()V not found!\n");
179 void throw_exception(void)
181 throw_exception_exit_intern(false);
185 void throw_exception_exit(void)
187 throw_exception_exit_intern(true);
191 void throw_main_exception(void)
193 fprintf(stderr, "Exception in thread \"main\" ");
196 throw_exception_exit_intern(false);
200 void throw_main_exception_exit(void)
202 fprintf(stderr, "Exception in thread \"main\" ");
205 throw_exception_exit_intern(true);
209 void throw_cacao_exception_exit(const char *exception, const char *message, ...)
216 len = strlen(exception);
217 tmp = MNEW(char, len + 1);
218 strncpy(tmp, exception, len);
221 /* convert to classname */
223 for (i = len - 1; i >= 0; i--)
224 if (tmp[i] == '/') tmp[i] = '.';
226 fprintf(stderr, "Exception in thread \"main\" %s", tmp);
228 MFREE(tmp, char, len);
230 if (strlen(message) > 0) {
231 fprintf(stderr, ": ");
233 va_start(ap, message);
234 vfprintf(stderr, message, ap);
238 fprintf(stderr, "\n");
247 java_objectheader *new_exception(const char *classname)
251 if (!load_class_bootstrap(utf_new_char(classname), &c))
252 return *exceptionptr;
254 return native_new_and_init(c);
257 java_objectheader *new_exception_message(const char *classname, const char *message)
261 if (!load_class_bootstrap(utf_new_char(classname), &c))
262 return *exceptionptr;
265 return native_new_and_init_string(c, javastring_new_char(message));
269 java_objectheader *new_exception_throwable(const char *classname, java_lang_Throwable *throwable)
273 if (!load_class_bootstrap(utf_new_char(classname), &c))
274 return *exceptionptr;
277 return native_new_and_init_throwable(c, throwable);
281 java_objectheader *new_exception_utfmessage(const char *classname, utf *message)
285 if (!load_class_bootstrap(utf_new_char(classname), &c))
286 return *exceptionptr;
289 return native_new_and_init_string(c, javastring_new(message));
293 java_objectheader *new_exception_javastring(const char *classname, java_lang_String *message)
297 if (!load_class_bootstrap(utf_new_char(classname), &c))
298 return *exceptionptr;
301 return native_new_and_init_string(c, message);
305 java_objectheader *new_exception_int(const char *classname, s4 i)
309 if (!load_class_bootstrap(utf_new_char(classname), &c))
310 return *exceptionptr;
312 return native_new_and_init_int(c, i);
316 /* new_classformaterror ********************************************************
318 generates a java.lang.ClassFormatError for the classloader
320 *******************************************************************************/
322 java_objectheader *new_classformaterror(classinfo *c, const char *message, ...)
324 java_objectheader *o;
329 /* calculate message length */
332 msglen = utf_strlen(c->name) + strlen(" (");
334 va_start(ap, message);
335 msglen += get_variable_message_length(message, ap);
339 msglen += strlen(")");
341 msglen += strlen("0");
343 /* allocate a buffer */
345 msg = MNEW(char, msglen);
347 /* print message into allocated buffer */
350 utf_sprint_classname(msg, c->name);
354 va_start(ap, message);
355 vsprintf(msg + strlen(msg), message, ap);
361 o = new_exception_message(string_java_lang_ClassFormatError, msg);
363 MFREE(msg, char, msglen);
369 /* new_internalerror ***********************************************************
371 Generates a java.lang.InternalError for the VM.
373 *******************************************************************************/
375 java_objectheader *new_internalerror(const char *message, ...)
377 char msg[MAXLOGTEXT];
380 va_start(ap, message);
381 vsprintf(msg, message, ap);
384 return new_exception_message(string_java_lang_InternalError, msg);
388 /* new_unsupportedclassversionerror ********************************************
390 generates a java.lang.UnsupportedClassVersionError for the classloader
392 *******************************************************************************/
394 java_objectheader *new_unsupportedclassversionerror(classinfo *c, const char *message, ...)
396 char msg[MAXLOGTEXT];
399 utf_sprint_classname(msg, c->name);
400 sprintf(msg + strlen(msg), " (");
402 va_start(ap, message);
403 vsprintf(msg + strlen(msg), message, ap);
406 sprintf(msg + strlen(msg), ")");
408 return new_exception_message(string_java_lang_UnsupportedClassVersionError,
413 /* new_verifyerror *************************************************************
415 generates a java.lang.VerifyError for the jit compiler
417 *******************************************************************************/
419 java_objectheader *new_verifyerror(methodinfo *m, const char *message)
421 java_objectheader *o;
425 useinlining = false; /* at least until sure inlining works with exceptions*/
426 len = 8 + utf_strlen(m->class->name) +
427 10 + utf_strlen(m->name) +
428 13 + utf_strlen(m->descriptor) +
429 2 + strlen(message) + 1;
431 msg = MNEW(char, len);
433 sprintf(msg, "(class: ");
434 utf_sprint(msg + strlen(msg), m->class->name);
435 sprintf(msg + strlen(msg), ", method: ");
436 utf_sprint(msg + strlen(msg), m->name);
437 sprintf(msg + strlen(msg), ", signature: ");
438 utf_sprint(msg + strlen(msg), m->descriptor);
439 sprintf(msg + strlen(msg), ") %s", message);
441 o = new_exception_message(string_java_lang_VerifyError, msg);
449 /* new_arithmeticexception *****************************************************
451 generates a java.lang.ArithmeticException for the jit compiler
453 *******************************************************************************/
455 java_objectheader *new_arithmeticexception(void)
457 java_objectheader *e;
459 e = new_exception_message(string_java_lang_ArithmeticException,
460 string_java_lang_ArithmeticException_message);
463 return *exceptionptr;
469 /* new_arrayindexoutofboundsexception ******************************************
471 generates a java.lang.ArrayIndexOutOfBoundsException for the jit compiler
473 *******************************************************************************/
475 java_objectheader *new_arrayindexoutofboundsexception(s4 index)
477 java_objectheader *e;
481 /* convert the index into a String, like Sun does */
483 m = class_resolveclassmethod(class_java_lang_String,
484 utf_new_char("valueOf"),
485 utf_new_char("(I)Ljava/lang/String;"),
486 class_java_lang_Object,
490 return *exceptionptr;
492 s = (java_lang_String *) asm_calljavafunction(m,
503 return *exceptionptr;
505 e = new_exception_javastring(string_java_lang_ArrayIndexOutOfBoundsException,
509 return *exceptionptr;
515 /* new_arraystoreexception *****************************************************
517 generates a java.lang.ArrayStoreException for the jit compiler
519 *******************************************************************************/
521 java_objectheader *new_arraystoreexception(void)
523 java_objectheader *e;
525 e = new_exception(string_java_lang_ArrayStoreException);
528 return *exceptionptr;
534 /* new_classcastexception ******************************************************
536 generates a java.lang.ClassCastException for the jit compiler
538 *******************************************************************************/
540 java_objectheader *new_classcastexception(void)
542 java_objectheader *e;
544 e = new_exception(string_java_lang_ClassCastException);
547 return *exceptionptr;
553 /* new_negativearraysizeexception **********************************************
555 generates a java.lang.NegativeArraySizeException for the jit compiler
557 *******************************************************************************/
559 java_objectheader *new_negativearraysizeexception(void)
561 java_objectheader *e;
563 e = new_exception(string_java_lang_NegativeArraySizeException);
566 return *exceptionptr;
572 /* new_nullpointerexception ****************************************************
574 generates a java.lang.NullPointerException for the jit compiler
576 *******************************************************************************/
578 java_objectheader *new_nullpointerexception(void)
580 java_objectheader *e;
582 e = new_exception(string_java_lang_NullPointerException);
585 return *exceptionptr;
592 * These are local overrides for various environment variables in Emacs.
593 * Please do not remove this and leave it at the end of the file, where
594 * Emacs will automagically detect them.
595 * ---------------------------------------------------------------------
598 * indent-tabs-mode: t