]> git.ipfire.org Git - thirdparty/gcc.git/blob - libjava/prims.cc
d94cd92cbc73fbb852240290223386f77a916a22
[thirdparty/gcc.git] / libjava / prims.cc
1 // prims.cc - Code for core of runtime environment.
2
3 /* Copyright (C) 1998, 1999, 2000, 2001, 2002, 2003, 2004, 2005, 2006, 2007 Free Software Foundation
4
5 This file is part of libgcj.
6
7 This software is copyrighted work licensed under the terms of the
8 Libgcj License. Please consult the file "LIBGCJ_LICENSE" for
9 details. */
10
11 #include <config.h>
12 #include <platform.h>
13
14 #include <stdlib.h>
15 #include <stdarg.h>
16 #include <stdio.h>
17 #include <string.h>
18 #include <signal.h>
19
20 #ifdef HAVE_UNISTD_H
21 #include <unistd.h>
22 #endif
23
24 #include <gcj/cni.h>
25 #include <jvm.h>
26 #include <java-signal.h>
27 #include <java-threads.h>
28 #include <java-interp.h>
29
30 #ifdef ENABLE_JVMPI
31 #include <jvmpi.h>
32 #include <java/lang/ThreadGroup.h>
33 #endif
34
35 #ifdef INTERPRETER
36 #include <jvmti.h>
37 #include "jvmti-int.h"
38 #endif
39
40 #ifndef DISABLE_GETENV_PROPERTIES
41 #include <ctype.h>
42 #include <java-props.h>
43 #define PROCESS_GCJ_PROPERTIES process_gcj_properties()
44 #else
45 #define PROCESS_GCJ_PROPERTIES
46 #endif // DISABLE_GETENV_PROPERTIES
47
48 #include <java/lang/Class.h>
49 #include <java/lang/ClassLoader.h>
50 #include <java/lang/Runtime.h>
51 #include <java/lang/String.h>
52 #include <java/lang/Thread.h>
53 #include <java/lang/ThreadGroup.h>
54 #include <java/lang/ArrayIndexOutOfBoundsException.h>
55 #include <java/lang/ArithmeticException.h>
56 #include <java/lang/ClassFormatError.h>
57 #include <java/lang/ClassNotFoundException.h>
58 #include <java/lang/InternalError.h>
59 #include <java/lang/NegativeArraySizeException.h>
60 #include <java/lang/NoClassDefFoundError.h>
61 #include <java/lang/NullPointerException.h>
62 #include <java/lang/OutOfMemoryError.h>
63 #include <java/lang/System.h>
64 #include <java/lang/VMClassLoader.h>
65 #include <java/lang/reflect/Modifier.h>
66 #include <java/io/PrintStream.h>
67 #include <java/lang/UnsatisfiedLinkError.h>
68 #include <java/lang/VirtualMachineError.h>
69 #include <gnu/gcj/runtime/ExtensionClassLoader.h>
70 #include <gnu/gcj/runtime/FinalizerThread.h>
71 #include <execution.h>
72
73 #ifdef INTERPRETER
74 #include <gnu/classpath/jdwp/Jdwp.h>
75 #include <gnu/classpath/jdwp/VMVirtualMachine.h>
76 #endif // INTERPRETER
77
78 #include <gnu/java/lang/MainThread.h>
79
80 #ifdef USE_LTDL
81 #include <ltdl.h>
82 #endif
83
84 // Execution engine for compiled code.
85 _Jv_CompiledEngine _Jv_soleCompiledEngine;
86
87 // Execution engine for code compiled with -findirect-classes
88 _Jv_IndirectCompiledEngine _Jv_soleIndirectCompiledEngine;
89
90 // We allocate a single OutOfMemoryError exception which we keep
91 // around for use if we run out of memory.
92 static java::lang::OutOfMemoryError *no_memory;
93
94 // Number of bytes in largest array object we create. This could be
95 // increased to the largest size_t value, so long as the appropriate
96 // functions are changed to take a size_t argument instead of jint.
97 #define MAX_OBJECT_SIZE (((size_t)1<<31) - 1)
98
99 // Properties set at compile time.
100 const char **_Jv_Compiler_Properties = NULL;
101 int _Jv_Properties_Count = 0;
102
103 #ifndef DISABLE_GETENV_PROPERTIES
104 // Property key/value pairs.
105 property_pair *_Jv_Environment_Properties;
106 #endif
107
108 // Stash the argv pointer to benefit native libraries that need it.
109 const char **_Jv_argv;
110 int _Jv_argc;
111
112 // Debugging options
113 static bool remoteDebug = false;
114 #ifdef INTERPRETER
115 static char defaultJdwpOptions[] = "";
116 static char *jdwpOptions = defaultJdwpOptions;
117
118 // Typedefs for JVMTI agent functions.
119 typedef jint jvmti_agent_onload_func (JavaVM *vm, char *options,
120 void *reserved);
121 typedef jint jvmti_agent_onunload_func (JavaVM *vm);
122
123 // JVMTI agent function pointers.
124 static jvmti_agent_onload_func *jvmti_agentonload = NULL;
125 static jvmti_agent_onunload_func *jvmti_agentonunload = NULL;
126 static char *jvmti_agent_opts;
127 #endif // INTERPRETER
128
129 // Argument support.
130 int
131 _Jv_GetNbArgs (void)
132 {
133 // _Jv_argc is 0 if not explicitly initialized.
134 return _Jv_argc;
135 }
136
137 const char *
138 _Jv_GetSafeArg (int index)
139 {
140 if (index >=0 && index < _Jv_GetNbArgs ())
141 return _Jv_argv[index];
142 else
143 return "";
144 }
145
146 void
147 _Jv_SetArgs (int argc, const char **argv)
148 {
149 _Jv_argc = argc;
150 _Jv_argv = argv;
151 }
152
153 #ifdef ENABLE_JVMPI
154 // Pointer to JVMPI notification functions.
155 void (*_Jv_JVMPI_Notify_OBJECT_ALLOC) (JVMPI_Event *event);
156 void (*_Jv_JVMPI_Notify_THREAD_START) (JVMPI_Event *event);
157 void (*_Jv_JVMPI_Notify_THREAD_END) (JVMPI_Event *event);
158 #endif
159 \f
160
161 #if defined (HANDLE_SEGV) || defined(HANDLE_FPE)
162 /* Unblock a signal. Unless we do this, the signal may only be sent
163 once. */
164 static void
165 unblock_signal (int signum __attribute__ ((__unused__)))
166 {
167 #ifdef _POSIX_VERSION
168 sigset_t sigs;
169
170 sigemptyset (&sigs);
171 sigaddset (&sigs, signum);
172 sigprocmask (SIG_UNBLOCK, &sigs, NULL);
173 #endif
174 }
175 #endif
176
177 #ifdef HANDLE_SEGV
178 SIGNAL_HANDLER (catch_segv)
179 {
180 unblock_signal (SIGSEGV);
181 MAKE_THROW_FRAME (nullp);
182 java::lang::NullPointerException *nullp
183 = new java::lang::NullPointerException;
184 throw nullp;
185 }
186 #endif
187
188 #ifdef HANDLE_FPE
189 SIGNAL_HANDLER (catch_fpe)
190 {
191 unblock_signal (SIGFPE);
192 #ifdef HANDLE_DIVIDE_OVERFLOW
193 HANDLE_DIVIDE_OVERFLOW;
194 #else
195 MAKE_THROW_FRAME (arithexception);
196 #endif
197 java::lang::ArithmeticException *arithexception
198 = new java::lang::ArithmeticException (JvNewStringLatin1 ("/ by zero"));
199 throw arithexception;
200 }
201 #endif
202
203
204 jboolean
205 _Jv_equalUtf8Consts (const Utf8Const* a, const Utf8Const *b)
206 {
207 int len;
208 const _Jv_ushort *aptr, *bptr;
209 if (a == b)
210 return true;
211 if (a->hash != b->hash)
212 return false;
213 len = a->length;
214 if (b->length != len)
215 return false;
216 aptr = (const _Jv_ushort *)a->data;
217 bptr = (const _Jv_ushort *)b->data;
218 len = (len + 1) >> 1;
219 while (--len >= 0)
220 if (*aptr++ != *bptr++)
221 return false;
222 return true;
223 }
224
225 /* True iff A is equal to STR.
226 HASH is STR->hashCode().
227 */
228
229 jboolean
230 _Jv_equal (Utf8Const* a, jstring str, jint hash)
231 {
232 if (a->hash != (_Jv_ushort) hash)
233 return false;
234 jint len = str->length();
235 jint i = 0;
236 jchar *sptr = _Jv_GetStringChars (str);
237 unsigned char* ptr = (unsigned char*) a->data;
238 unsigned char* limit = ptr + a->length;
239 for (;; i++, sptr++)
240 {
241 int ch = UTF8_GET (ptr, limit);
242 if (i == len)
243 return ch < 0;
244 if (ch != *sptr)
245 return false;
246 }
247 return true;
248 }
249
250 /* Like _Jv_equal, but stop after N characters. */
251 jboolean
252 _Jv_equaln (Utf8Const *a, jstring str, jint n)
253 {
254 jint len = str->length();
255 jint i = 0;
256 jchar *sptr = _Jv_GetStringChars (str);
257 unsigned char* ptr = (unsigned char*) a->data;
258 unsigned char* limit = ptr + a->length;
259 for (; n-- > 0; i++, sptr++)
260 {
261 int ch = UTF8_GET (ptr, limit);
262 if (i == len)
263 return ch < 0;
264 if (ch != *sptr)
265 return false;
266 }
267 return true;
268 }
269
270 // Determines whether the given Utf8Const object contains
271 // a type which is primitive or some derived form of it, eg.
272 // an array or multi-dimensional array variant.
273 jboolean
274 _Jv_isPrimitiveOrDerived(const Utf8Const *a)
275 {
276 unsigned char *aptr = (unsigned char *) a->data;
277 unsigned char *alimit = aptr + a->length;
278 int ac = UTF8_GET(aptr, alimit);
279
280 // Skips any leading array marks.
281 while (ac == '[')
282 ac = UTF8_GET(aptr, alimit);
283
284 // There should not be another character. This implies that
285 // the type name is only one character long.
286 if (UTF8_GET(aptr, alimit) == -1)
287 switch ( ac )
288 {
289 case 'Z':
290 case 'B':
291 case 'C':
292 case 'S':
293 case 'I':
294 case 'J':
295 case 'F':
296 case 'D':
297 return true;
298 default:
299 break;
300 }
301
302 return false;
303 }
304
305 // Find out whether two _Jv_Utf8Const candidates contain the same
306 // classname.
307 // The method is written to handle the different formats of classnames.
308 // Eg. "Ljava/lang/Class;", "Ljava.lang.Class;", "java/lang/Class" and
309 // "java.lang.Class" will be seen as equal.
310 // Warning: This function is not smart enough to declare "Z" and "boolean"
311 // and similar cases as equal (and is not meant to be used this way)!
312 jboolean
313 _Jv_equalUtf8Classnames (const Utf8Const *a, const Utf8Const *b)
314 {
315 // If the class name's length differs by two characters
316 // it is possible that we have candidates which are given
317 // in the two different formats ("Lp1/p2/cn;" vs. "p1/p2/cn")
318 switch (a->length - b->length)
319 {
320 case -2:
321 case 0:
322 case 2:
323 break;
324 default:
325 return false;
326 }
327
328 unsigned char *aptr = (unsigned char *) a->data;
329 unsigned char *alimit = aptr + a->length;
330 unsigned char *bptr = (unsigned char *) b->data;
331 unsigned char *blimit = bptr + b->length;
332
333 if (alimit[-1] == ';')
334 alimit--;
335
336 if (blimit[-1] == ';')
337 blimit--;
338
339 int ac = UTF8_GET(aptr, alimit);
340 int bc = UTF8_GET(bptr, blimit);
341
342 // Checks whether both strings have the same amount of leading [ characters.
343 while (ac == '[')
344 {
345 if (bc == '[')
346 {
347 ac = UTF8_GET(aptr, alimit);
348 bc = UTF8_GET(bptr, blimit);
349 continue;
350 }
351
352 return false;
353 }
354
355 // Skips leading L character.
356 if (ac == 'L')
357 ac = UTF8_GET(aptr, alimit);
358
359 if (bc == 'L')
360 bc = UTF8_GET(bptr, blimit);
361
362 // Compares the remaining characters.
363 while (ac != -1 && bc != -1)
364 {
365 // Replaces package separating dots with slashes.
366 if (ac == '.')
367 ac = '/';
368
369 if (bc == '.')
370 bc = '/';
371
372 // Now classnames differ if there is at least one non-matching
373 // character.
374 if (ac != bc)
375 return false;
376
377 ac = UTF8_GET(aptr, alimit);
378 bc = UTF8_GET(bptr, blimit);
379 }
380
381 return (ac == bc);
382 }
383
384 /* Count the number of Unicode chars encoded in a given Ut8 string. */
385 int
386 _Jv_strLengthUtf8(const char* str, int len)
387 {
388 unsigned char* ptr;
389 unsigned char* limit;
390 int str_length;
391
392 ptr = (unsigned char*) str;
393 limit = ptr + len;
394 str_length = 0;
395 for (; ptr < limit; str_length++)
396 {
397 if (UTF8_GET (ptr, limit) < 0)
398 return (-1);
399 }
400 return (str_length);
401 }
402
403 /* Calculate a hash value for a string encoded in Utf8 format.
404 * This returns the same hash value as specified or java.lang.String.hashCode.
405 */
406 jint
407 _Jv_hashUtf8String (const char* str, int len)
408 {
409 unsigned char* ptr = (unsigned char*) str;
410 unsigned char* limit = ptr + len;
411 jint hash = 0;
412
413 for (; ptr < limit;)
414 {
415 int ch = UTF8_GET (ptr, limit);
416 /* Updated specification from
417 http://www.javasoft.com/docs/books/jls/clarify.html. */
418 hash = (31 * hash) + ch;
419 }
420 return hash;
421 }
422
423 void
424 _Jv_Utf8Const::init(const char *s, int len)
425 {
426 ::memcpy (data, s, len);
427 data[len] = 0;
428 length = len;
429 hash = _Jv_hashUtf8String (s, len) & 0xFFFF;
430 }
431
432 _Jv_Utf8Const *
433 _Jv_makeUtf8Const (const char* s, int len)
434 {
435 if (len < 0)
436 len = strlen (s);
437 Utf8Const* m
438 = (Utf8Const*) _Jv_AllocBytes (_Jv_Utf8Const::space_needed(s, len));
439 m->init(s, len);
440 return m;
441 }
442
443 _Jv_Utf8Const *
444 _Jv_makeUtf8Const (jstring string)
445 {
446 jint hash = string->hashCode ();
447 jint len = _Jv_GetStringUTFLength (string);
448
449 Utf8Const* m = (Utf8Const*)
450 _Jv_AllocBytes (sizeof(Utf8Const) + len + 1);
451
452 m->hash = hash;
453 m->length = len;
454
455 _Jv_GetStringUTFRegion (string, 0, string->length (), m->data);
456 m->data[len] = 0;
457
458 return m;
459 }
460
461 \f
462
463 #ifdef DEBUG
464 void
465 _Jv_Abort (const char *function, const char *file, int line,
466 const char *message)
467 #else
468 void
469 _Jv_Abort (const char *, const char *, int, const char *message)
470 #endif
471 {
472 #ifdef DEBUG
473 fprintf (stderr,
474 "libgcj failure: %s\n in function %s, file %s, line %d\n",
475 message, function, file, line);
476 #else
477 fprintf (stderr, "libgcj failure: %s\n", message);
478 #endif
479 fflush (stderr);
480 abort ();
481 }
482
483 static void
484 fail_on_finalization (jobject)
485 {
486 JvFail ("object was finalized");
487 }
488
489 void
490 _Jv_GCWatch (jobject obj)
491 {
492 _Jv_RegisterFinalizer (obj, fail_on_finalization);
493 }
494
495 void
496 _Jv_ThrowBadArrayIndex(jint bad_index)
497 {
498 throw new java::lang::ArrayIndexOutOfBoundsException
499 (java::lang::String::valueOf (bad_index));
500 }
501
502 void
503 _Jv_ThrowNullPointerException ()
504 {
505 throw new java::lang::NullPointerException;
506 }
507
508 // Resolve an entry in the constant pool and return the target
509 // address.
510 void *
511 _Jv_ResolvePoolEntry (jclass this_class, jint index)
512 {
513 _Jv_Constants *pool = &this_class->constants;
514
515 if ((pool->tags[index] & JV_CONSTANT_ResolvedFlag) != 0)
516 return pool->data[index].field->u.addr;
517
518 JvSynchronize sync (this_class);
519 return (_Jv_Linker::resolve_pool_entry (this_class, index))
520 .field->u.addr;
521 }
522
523
524 // Explicitly throw a no memory exception.
525 // The collector calls this when it encounters an out-of-memory condition.
526 void _Jv_ThrowNoMemory()
527 {
528 throw no_memory;
529 }
530
531 #ifdef ENABLE_JVMPI
532 # define JVMPI_NOTIFY_ALLOC(klass,size,obj) \
533 if (__builtin_expect (_Jv_JVMPI_Notify_OBJECT_ALLOC != 0, false)) \
534 jvmpi_notify_alloc(klass,size,obj);
535 static void
536 jvmpi_notify_alloc(jclass klass, jint size, jobject obj)
537 {
538 // Service JVMPI allocation request.
539 JVMPI_Event event;
540
541 event.event_type = JVMPI_EVENT_OBJECT_ALLOC;
542 event.env_id = NULL;
543 event.u.obj_alloc.arena_id = 0;
544 event.u.obj_alloc.class_id = (jobjectID) klass;
545 event.u.obj_alloc.is_array = 0;
546 event.u.obj_alloc.size = size;
547 event.u.obj_alloc.obj_id = (jobjectID) obj;
548
549 // FIXME: This doesn't look right for the Boehm GC. A GC may
550 // already be in progress. _Jv_DisableGC () doesn't wait for it.
551 // More importantly, I don't see the need for disabling GC, since we
552 // blatantly have a pointer to obj on our stack, ensuring that the
553 // object can't be collected. Even for a nonconservative collector,
554 // it appears to me that this must be true, since we are about to
555 // return obj. Isn't this whole approach way too intrusive for
556 // a useful profiling interface? - HB
557 _Jv_DisableGC ();
558 (*_Jv_JVMPI_Notify_OBJECT_ALLOC) (&event);
559 _Jv_EnableGC ();
560 }
561 #else /* !ENABLE_JVMPI */
562 # define JVMPI_NOTIFY_ALLOC(klass,size,obj) /* do nothing */
563 #endif
564
565 // Allocate a new object of class KLASS.
566 // First a version that assumes that we have no finalizer, and that
567 // the class is already initialized.
568 // If we know that JVMPI is disabled, this can be replaced by a direct call
569 // to the allocator for the appropriate GC.
570 jobject
571 _Jv_AllocObjectNoInitNoFinalizer (jclass klass)
572 {
573 jint size = klass->size ();
574 jobject obj = (jobject) _Jv_AllocObj (size, klass);
575 JVMPI_NOTIFY_ALLOC (klass, size, obj);
576 return obj;
577 }
578
579 // And now a version that initializes if necessary.
580 jobject
581 _Jv_AllocObjectNoFinalizer (jclass klass)
582 {
583 if (_Jv_IsPhantomClass(klass) )
584 throw new java::lang::NoClassDefFoundError(klass->getName());
585
586 _Jv_InitClass (klass);
587 jint size = klass->size ();
588 jobject obj = (jobject) _Jv_AllocObj (size, klass);
589 JVMPI_NOTIFY_ALLOC (klass, size, obj);
590 return obj;
591 }
592
593 // And now the general version that registers a finalizer if necessary.
594 jobject
595 _Jv_AllocObject (jclass klass)
596 {
597 jobject obj = _Jv_AllocObjectNoFinalizer (klass);
598
599 // We assume that the compiler only generates calls to this routine
600 // if there really is an interesting finalizer.
601 // Unfortunately, we still have to the dynamic test, since there may
602 // be cni calls to this routine.
603 // Note that on IA64 get_finalizer() returns the starting address of the
604 // function, not a function pointer. Thus this still works.
605 if (klass->vtable->get_finalizer ()
606 != java::lang::Object::class$.vtable->get_finalizer ())
607 _Jv_RegisterFinalizer (obj, _Jv_FinalizeObject);
608 return obj;
609 }
610
611 // Allocate a String, including variable length storage.
612 jstring
613 _Jv_AllocString(jsize len)
614 {
615 using namespace java::lang;
616
617 jsize sz = sizeof(java::lang::String) + len * sizeof(jchar);
618
619 // We assert that for strings allocated this way, the data field
620 // will always point to the object itself. Thus there is no reason
621 // for the garbage collector to scan any of it.
622 // Furthermore, we're about to overwrite the string data, so
623 // initialization of the object is not an issue.
624
625 // String needs no initialization, and there is no finalizer, so
626 // we can go directly to the collector's allocator interface.
627 jstring obj = (jstring) _Jv_AllocPtrFreeObj(sz, &String::class$);
628
629 obj->data = obj;
630 obj->boffset = sizeof(java::lang::String);
631 obj->count = len;
632 obj->cachedHashCode = 0;
633
634 JVMPI_NOTIFY_ALLOC (&String::class$, sz, obj);
635
636 return obj;
637 }
638
639 // A version of the above that assumes the object contains no pointers,
640 // and requires no finalization. This can't happen if we need pointers
641 // to locks.
642 #ifdef JV_HASH_SYNCHRONIZATION
643 jobject
644 _Jv_AllocPtrFreeObject (jclass klass)
645 {
646 _Jv_InitClass (klass);
647 jint size = klass->size ();
648
649 jobject obj = (jobject) _Jv_AllocPtrFreeObj (size, klass);
650
651 JVMPI_NOTIFY_ALLOC (klass, size, obj);
652
653 return obj;
654 }
655 #endif /* JV_HASH_SYNCHRONIZATION */
656
657
658 // Allocate a new array of Java objects. Each object is of type
659 // `elementClass'. `init' is used to initialize each slot in the
660 // array.
661 jobjectArray
662 _Jv_NewObjectArray (jsize count, jclass elementClass, jobject init)
663 {
664 // Creating an array of an unresolved type is impossible. So we throw
665 // the NoClassDefFoundError.
666 if ( _Jv_IsPhantomClass(elementClass) )
667 throw new java::lang::NoClassDefFoundError(elementClass->getName());
668
669 if (__builtin_expect (count < 0, false))
670 throw new java::lang::NegativeArraySizeException;
671
672 JvAssert (! elementClass->isPrimitive ());
673
674 // Ensure that elements pointer is properly aligned.
675 jobjectArray obj = NULL;
676 size_t size = (size_t) elements (obj);
677 // Check for overflow.
678 if (__builtin_expect ((size_t) count >
679 (MAX_OBJECT_SIZE - 1 - size) / sizeof (jobject), false))
680 throw no_memory;
681
682 size += count * sizeof (jobject);
683
684 jclass klass = _Jv_GetArrayClass (elementClass,
685 elementClass->getClassLoaderInternal());
686
687 obj = (jobjectArray) _Jv_AllocArray (size, klass);
688 // Cast away const.
689 jsize *lp = const_cast<jsize *> (&obj->length);
690 *lp = count;
691 // We know the allocator returns zeroed memory. So don't bother
692 // zeroing it again.
693 if (init)
694 {
695 jobject *ptr = elements(obj);
696 while (--count >= 0)
697 *ptr++ = init;
698 }
699 return obj;
700 }
701
702 // Allocate a new array of primitives. ELTYPE is the type of the
703 // element, COUNT is the size of the array.
704 jobject
705 _Jv_NewPrimArray (jclass eltype, jint count)
706 {
707 int elsize = eltype->size();
708 if (__builtin_expect (count < 0, false))
709 throw new java::lang::NegativeArraySizeException;
710
711 JvAssert (eltype->isPrimitive ());
712 jobject dummy = NULL;
713 size_t size = (size_t) _Jv_GetArrayElementFromElementType (dummy, eltype);
714
715 // Check for overflow.
716 if (__builtin_expect ((size_t) count >
717 (MAX_OBJECT_SIZE - size) / elsize, false))
718 throw no_memory;
719
720 jclass klass = _Jv_GetArrayClass (eltype, 0);
721
722 # ifdef JV_HASH_SYNCHRONIZATION
723 // Since the vtable is always statically allocated,
724 // these are completely pointerfree! Make sure the GC doesn't touch them.
725 __JArray *arr =
726 (__JArray*) _Jv_AllocPtrFreeObj (size + elsize * count, klass);
727 memset((char *)arr + size, 0, elsize * count);
728 # else
729 __JArray *arr = (__JArray*) _Jv_AllocObj (size + elsize * count, klass);
730 // Note that we assume we are given zeroed memory by the allocator.
731 # endif
732 // Cast away const.
733 jsize *lp = const_cast<jsize *> (&arr->length);
734 *lp = count;
735
736 return arr;
737 }
738
739 jobject
740 _Jv_NewArray (jint type, jint size)
741 {
742 switch (type)
743 {
744 case 4: return JvNewBooleanArray (size);
745 case 5: return JvNewCharArray (size);
746 case 6: return JvNewFloatArray (size);
747 case 7: return JvNewDoubleArray (size);
748 case 8: return JvNewByteArray (size);
749 case 9: return JvNewShortArray (size);
750 case 10: return JvNewIntArray (size);
751 case 11: return JvNewLongArray (size);
752 }
753 throw new java::lang::InternalError
754 (JvNewStringLatin1 ("invalid type code in _Jv_NewArray"));
755 }
756
757 // Allocate a possibly multi-dimensional array but don't check that
758 // any array length is <0.
759 static jobject
760 _Jv_NewMultiArrayUnchecked (jclass type, jint dimensions, jint *sizes)
761 {
762 JvAssert (type->isArray());
763 jclass element_type = type->getComponentType();
764 jobject result;
765 if (element_type->isPrimitive())
766 result = _Jv_NewPrimArray (element_type, sizes[0]);
767 else
768 result = _Jv_NewObjectArray (sizes[0], element_type, NULL);
769
770 if (dimensions > 1)
771 {
772 JvAssert (! element_type->isPrimitive());
773 JvAssert (element_type->isArray());
774 jobject *contents = elements ((jobjectArray) result);
775 for (int i = 0; i < sizes[0]; ++i)
776 contents[i] = _Jv_NewMultiArrayUnchecked (element_type, dimensions - 1,
777 sizes + 1);
778 }
779
780 return result;
781 }
782
783 jobject
784 _Jv_NewMultiArray (jclass type, jint dimensions, jint *sizes)
785 {
786 for (int i = 0; i < dimensions; ++i)
787 if (sizes[i] < 0)
788 throw new java::lang::NegativeArraySizeException;
789
790 return _Jv_NewMultiArrayUnchecked (type, dimensions, sizes);
791 }
792
793 jobject
794 _Jv_NewMultiArray (jclass array_type, jint dimensions, ...)
795 {
796 // Creating an array of an unresolved type is impossible. So we throw
797 // the NoClassDefFoundError.
798 if (_Jv_IsPhantomClass(array_type))
799 throw new java::lang::NoClassDefFoundError(array_type->getName());
800
801 va_list args;
802 jint sizes[dimensions];
803 va_start (args, dimensions);
804 for (int i = 0; i < dimensions; ++i)
805 {
806 jint size = va_arg (args, jint);
807 if (size < 0)
808 throw new java::lang::NegativeArraySizeException;
809 sizes[i] = size;
810 }
811 va_end (args);
812
813 return _Jv_NewMultiArrayUnchecked (array_type, dimensions, sizes);
814 }
815
816 \f
817
818 // Ensure 8-byte alignment, for hash synchronization.
819 #define DECLARE_PRIM_TYPE(NAME) \
820 java::lang::Class _Jv_##NAME##Class __attribute__ ((aligned (8)));
821
822 DECLARE_PRIM_TYPE(byte)
823 DECLARE_PRIM_TYPE(short)
824 DECLARE_PRIM_TYPE(int)
825 DECLARE_PRIM_TYPE(long)
826 DECLARE_PRIM_TYPE(boolean)
827 DECLARE_PRIM_TYPE(char)
828 DECLARE_PRIM_TYPE(float)
829 DECLARE_PRIM_TYPE(double)
830 DECLARE_PRIM_TYPE(void)
831
832 void
833 _Jv_InitPrimClass (jclass cl, const char *cname, char sig, int len)
834 {
835 using namespace java::lang::reflect;
836
837 // We must set the vtable for the class; the Java constructor
838 // doesn't do this.
839 (*(_Jv_VTable **) cl) = java::lang::Class::class$.vtable;
840
841 // Initialize the fields we care about. We do this in the same
842 // order they are declared in Class.h.
843 cl->name = _Jv_makeUtf8Const ((char *) cname, -1);
844 cl->accflags = Modifier::PUBLIC | Modifier::FINAL | Modifier::ABSTRACT;
845 cl->method_count = sig;
846 cl->size_in_bytes = len;
847 cl->vtable = JV_PRIMITIVE_VTABLE;
848 cl->state = JV_STATE_DONE;
849 cl->depth = -1;
850 }
851
852 jclass
853 _Jv_FindClassFromSignature (char *sig, java::lang::ClassLoader *loader,
854 char **endp)
855 {
856 // First count arrays.
857 int array_count = 0;
858 while (*sig == '[')
859 {
860 ++sig;
861 ++array_count;
862 }
863
864 jclass result = NULL;
865 switch (*sig)
866 {
867 case 'B':
868 result = JvPrimClass (byte);
869 break;
870 case 'S':
871 result = JvPrimClass (short);
872 break;
873 case 'I':
874 result = JvPrimClass (int);
875 break;
876 case 'J':
877 result = JvPrimClass (long);
878 break;
879 case 'Z':
880 result = JvPrimClass (boolean);
881 break;
882 case 'C':
883 result = JvPrimClass (char);
884 break;
885 case 'F':
886 result = JvPrimClass (float);
887 break;
888 case 'D':
889 result = JvPrimClass (double);
890 break;
891 case 'V':
892 result = JvPrimClass (void);
893 break;
894 case 'L':
895 {
896 char *save = ++sig;
897 while (*sig && *sig != ';')
898 ++sig;
899 // Do nothing if signature appears to be malformed.
900 if (*sig == ';')
901 {
902 _Jv_Utf8Const *name = _Jv_makeUtf8Const (save, sig - save);
903 result = _Jv_FindClass (name, loader);
904 }
905 break;
906 }
907 default:
908 // Do nothing -- bad signature.
909 break;
910 }
911
912 if (endp)
913 {
914 // Not really the "end", but the last valid character that we
915 // looked at.
916 *endp = sig;
917 }
918
919 if (! result)
920 return NULL;
921
922 // Find arrays.
923 while (array_count-- > 0)
924 result = _Jv_GetArrayClass (result, loader);
925 return result;
926 }
927
928
929 jclass
930 _Jv_FindClassFromSignatureNoException (char *sig, java::lang::ClassLoader *loader,
931 char **endp)
932 {
933 jclass klass;
934
935 try
936 {
937 klass = _Jv_FindClassFromSignature(sig, loader, endp);
938 }
939 catch (java::lang::NoClassDefFoundError *ncdfe)
940 {
941 return NULL;
942 }
943 catch (java::lang::ClassNotFoundException *cnfe)
944 {
945 return NULL;
946 }
947
948 return klass;
949 }
950
951 JArray<jstring> *
952 JvConvertArgv (int argc, const char **argv)
953 {
954 if (argc < 0)
955 argc = 0;
956 jobjectArray ar = JvNewObjectArray(argc, &java::lang::String::class$, NULL);
957 jobject *ptr = elements(ar);
958 jbyteArray bytes = NULL;
959 for (int i = 0; i < argc; i++)
960 {
961 const char *arg = argv[i];
962 int len = strlen (arg);
963 if (bytes == NULL || bytes->length < len)
964 bytes = JvNewByteArray (len);
965 jbyte *bytePtr = elements (bytes);
966 // We assume jbyte == char.
967 memcpy (bytePtr, arg, len);
968
969 // Now convert using the default encoding.
970 *ptr++ = new java::lang::String (bytes, 0, len);
971 }
972 return (JArray<jstring>*) ar;
973 }
974
975 // FIXME: These variables are static so that they will be
976 // automatically scanned by the Boehm collector. This is needed
977 // because with qthreads the collector won't scan the initial stack --
978 // it will only scan the qthreads stacks.
979
980 // Command line arguments.
981 static JArray<jstring> *arg_vec;
982
983 // The primary thread.
984 static java::lang::Thread *main_thread;
985
986 #ifndef DISABLE_GETENV_PROPERTIES
987
988 static char *
989 next_property_key (char *s, size_t *length)
990 {
991 size_t l = 0;
992
993 JvAssert (s);
994
995 // Skip over whitespace
996 while (isspace (*s))
997 s++;
998
999 // If we've reached the end, return NULL. Also return NULL if for
1000 // some reason we've come across a malformed property string.
1001 if (*s == 0
1002 || *s == ':'
1003 || *s == '=')
1004 return NULL;
1005
1006 // Determine the length of the property key.
1007 while (s[l] != 0
1008 && ! isspace (s[l])
1009 && s[l] != ':'
1010 && s[l] != '=')
1011 {
1012 if (s[l] == '\\'
1013 && s[l+1] != 0)
1014 l++;
1015 l++;
1016 }
1017
1018 *length = l;
1019
1020 return s;
1021 }
1022
1023 static char *
1024 next_property_value (char *s, size_t *length)
1025 {
1026 size_t l = 0;
1027
1028 JvAssert (s);
1029
1030 while (isspace (*s))
1031 s++;
1032
1033 if (*s == ':'
1034 || *s == '=')
1035 s++;
1036
1037 while (isspace (*s))
1038 s++;
1039
1040 // Determine the length of the property value.
1041 while (s[l] != 0
1042 && ! isspace (s[l])
1043 && s[l] != ':'
1044 && s[l] != '=')
1045 {
1046 if (s[l] == '\\'
1047 && s[l+1] != 0)
1048 l += 2;
1049 else
1050 l++;
1051 }
1052
1053 *length = l;
1054
1055 return s;
1056 }
1057
1058 static void
1059 process_gcj_properties ()
1060 {
1061 char *props = getenv("GCJ_PROPERTIES");
1062
1063 if (NULL == props)
1064 return;
1065
1066 // Later on we will write \0s into this string. It is simplest to
1067 // just duplicate it here.
1068 props = strdup (props);
1069
1070 char *p = props;
1071 size_t length;
1072 size_t property_count = 0;
1073
1074 // Whip through props quickly in order to count the number of
1075 // property values.
1076 while (p && (p = next_property_key (p, &length)))
1077 {
1078 // Skip to the end of the key
1079 p += length;
1080
1081 p = next_property_value (p, &length);
1082 if (p)
1083 p += length;
1084
1085 property_count++;
1086 }
1087
1088 // Allocate an array of property value/key pairs.
1089 _Jv_Environment_Properties =
1090 (property_pair *) malloc (sizeof(property_pair)
1091 * (property_count + 1));
1092
1093 // Go through the properties again, initializing _Jv_Properties
1094 // along the way.
1095 p = props;
1096 property_count = 0;
1097 while (p && (p = next_property_key (p, &length)))
1098 {
1099 _Jv_Environment_Properties[property_count].key = p;
1100 _Jv_Environment_Properties[property_count].key_length = length;
1101
1102 // Skip to the end of the key
1103 p += length;
1104
1105 p = next_property_value (p, &length);
1106
1107 _Jv_Environment_Properties[property_count].value = p;
1108 _Jv_Environment_Properties[property_count].value_length = length;
1109
1110 if (p)
1111 p += length;
1112
1113 property_count++;
1114 }
1115 memset ((void *) &_Jv_Environment_Properties[property_count],
1116 0, sizeof (property_pair));
1117
1118 // Null terminate the strings.
1119 for (property_pair *prop = &_Jv_Environment_Properties[0];
1120 prop->key != NULL;
1121 prop++)
1122 {
1123 prop->key[prop->key_length] = 0;
1124 prop->value[prop->value_length] = 0;
1125 }
1126 }
1127 #endif // DISABLE_GETENV_PROPERTIES
1128
1129 namespace gcj
1130 {
1131 _Jv_Utf8Const *void_signature;
1132 _Jv_Utf8Const *clinit_name;
1133 _Jv_Utf8Const *init_name;
1134 _Jv_Utf8Const *finit_name;
1135
1136 bool runtimeInitialized = false;
1137
1138 // When true, print debugging information about class loading.
1139 bool verbose_class_flag;
1140
1141 // When true, enable the bytecode verifier and BC-ABI type verification.
1142 bool verifyClasses = true;
1143
1144 // Thread stack size specified by the -Xss runtime argument.
1145 size_t stack_size = 0;
1146
1147 // Start time of the VM
1148 jlong startTime = 0;
1149
1150 // Arguments passed to the VM
1151 JArray<jstring>* vmArgs;
1152
1153 // Currently loaded classes
1154 jint loadedClasses = 0;
1155
1156 // Unloaded classes
1157 jlong unloadedClasses = 0;
1158 }
1159
1160 // We accept all non-standard options accepted by Sun's java command,
1161 // for compatibility with existing application launch scripts.
1162 static jint
1163 parse_x_arg (char* option_string)
1164 {
1165 if (strlen (option_string) <= 0)
1166 return -1;
1167
1168 if (! strcmp (option_string, "int"))
1169 {
1170 // FIXME: this should cause the vm to never load shared objects
1171 }
1172 else if (! strcmp (option_string, "mixed"))
1173 {
1174 // FIXME: allow interpreted and native code
1175 }
1176 else if (! strcmp (option_string, "batch"))
1177 {
1178 // FIXME: disable background JIT'ing
1179 }
1180 else if (! strcmp (option_string, "debug"))
1181 {
1182 remoteDebug = true;
1183 }
1184 #ifdef INTERPRETER
1185 else if (! strncmp (option_string, "runjdwp:", 8))
1186 {
1187 if (strlen (option_string) > 8)
1188 jdwpOptions = &option_string[8];
1189 else
1190 {
1191 fprintf (stderr,
1192 "libgcj: argument required for JDWP options");
1193 return -1;
1194 }
1195 }
1196 #endif // INTERPRETER
1197 else if (! strncmp (option_string, "bootclasspath:", 14))
1198 {
1199 // FIXME: add a parse_bootclasspath_arg function
1200 }
1201 else if (! strncmp (option_string, "bootclasspath/a:", 16))
1202 {
1203 }
1204 else if (! strncmp (option_string, "bootclasspath/p:", 16))
1205 {
1206 }
1207 else if (! strcmp (option_string, "check:jni"))
1208 {
1209 // FIXME: enable strict JNI checking
1210 }
1211 else if (! strcmp (option_string, "future"))
1212 {
1213 // FIXME: enable strict class file format checks
1214 }
1215 else if (! strcmp (option_string, "noclassgc"))
1216 {
1217 // FIXME: disable garbage collection for classes
1218 }
1219 else if (! strcmp (option_string, "incgc"))
1220 {
1221 // FIXME: incremental garbage collection
1222 }
1223 else if (! strncmp (option_string, "loggc:", 6))
1224 {
1225 if (option_string[6] == '\0')
1226 {
1227 fprintf (stderr,
1228 "libgcj: filename argument expected for loggc option\n");
1229 return -1;
1230 }
1231 // FIXME: set gc logging filename
1232 }
1233 else if (! strncmp (option_string, "ms", 2))
1234 {
1235 // FIXME: ignore this option until PR 20699 is fixed.
1236 // _Jv_SetInitialHeapSize (option_string + 2);
1237 }
1238 else if (! strncmp (option_string, "mx", 2))
1239 _Jv_SetMaximumHeapSize (option_string + 2);
1240 else if (! strcmp (option_string, "prof"))
1241 {
1242 // FIXME: enable profiling of program running in vm
1243 }
1244 else if (! strncmp (option_string, "runhprof:", 9))
1245 {
1246 // FIXME: enable specific type of vm profiling. add a
1247 // parse_runhprof_arg function
1248 }
1249 else if (! strcmp (option_string, "rs"))
1250 {
1251 // FIXME: reduced system signal usage. disable thread dumps,
1252 // only terminate in response to user-initiated calls,
1253 // e.g. System.exit()
1254 }
1255 else if (! strncmp (option_string, "ss", 2))
1256 {
1257 _Jv_SetStackSize (option_string + 2);
1258 }
1259 else if (! strcmp (option_string, "X:+UseAltSigs"))
1260 {
1261 // FIXME: use signals other than SIGUSR1 and SIGUSR2
1262 }
1263 else if (! strcmp (option_string, "share:off"))
1264 {
1265 // FIXME: don't share class data
1266 }
1267 else if (! strcmp (option_string, "share:auto"))
1268 {
1269 // FIXME: share class data where possible
1270 }
1271 else if (! strcmp (option_string, "share:on"))
1272 {
1273 // FIXME: fail if impossible to share class data
1274 }
1275 else
1276 {
1277 // Unrecognized.
1278 return -1;
1279 }
1280 return 0;
1281 }
1282
1283 static jint
1284 parse_verbose_args (char* option_string,
1285 bool ignore_unrecognized)
1286 {
1287 size_t len = sizeof ("-verbose") - 1;
1288
1289 if (strlen (option_string) < len)
1290 return -1;
1291
1292 if (option_string[len] == ':'
1293 && option_string[len + 1] != '\0')
1294 {
1295 char* verbose_args = option_string + len + 1;
1296
1297 do
1298 {
1299 if (! strncmp (verbose_args,
1300 "gc", sizeof ("gc") - 1))
1301 {
1302 if (verbose_args[sizeof ("gc") - 1] == '\0'
1303 || verbose_args[sizeof ("gc") - 1] == ',')
1304 {
1305 // FIXME: we should add functions to boehm-gc that
1306 // toggle GC_print_stats, GC_PRINT_ADDRESS_MAP and
1307 // GC_print_back_height.
1308 verbose_args += sizeof ("gc") - 1;
1309 }
1310 else
1311 {
1312 verbose_arg_err:
1313 fprintf (stderr, "libgcj: unknown verbose option: %s\n",
1314 option_string);
1315 return -1;
1316 }
1317 }
1318 else if (! strncmp (verbose_args,
1319 "class",
1320 sizeof ("class") - 1))
1321 {
1322 if (verbose_args[sizeof ("class") - 1] == '\0'
1323 || verbose_args[sizeof ("class") - 1] == ',')
1324 {
1325 gcj::verbose_class_flag = true;
1326 verbose_args += sizeof ("class") - 1;
1327 }
1328 else
1329 goto verbose_arg_err;
1330 }
1331 else if (! strncmp (verbose_args, "jni",
1332 sizeof ("jni") - 1))
1333 {
1334 if (verbose_args[sizeof ("jni") - 1] == '\0'
1335 || verbose_args[sizeof ("jni") - 1] == ',')
1336 {
1337 // FIXME: enable JNI messages.
1338 verbose_args += sizeof ("jni") - 1;
1339 }
1340 else
1341 goto verbose_arg_err;
1342 }
1343 else if (ignore_unrecognized
1344 && verbose_args[0] == 'X')
1345 {
1346 // ignore unrecognized non-standard verbose option
1347 while (verbose_args[0] != '\0'
1348 && verbose_args[0] != ',')
1349 verbose_args++;
1350 }
1351 else if (verbose_args[0] == ',')
1352 {
1353 verbose_args++;
1354 }
1355 else
1356 goto verbose_arg_err;
1357
1358 if (verbose_args[0] == ',')
1359 verbose_args++;
1360 }
1361 while (verbose_args[0] != '\0');
1362 }
1363 else if (option_string[len] == 'g'
1364 && option_string[len + 1] == 'c'
1365 && option_string[len + 2] == '\0')
1366 {
1367 // FIXME: we should add functions to boehm-gc that
1368 // toggle GC_print_stats, GC_PRINT_ADDRESS_MAP and
1369 // GC_print_back_height.
1370 return 0;
1371 }
1372 else if (option_string[len] == '\0')
1373 {
1374 gcj::verbose_class_flag = true;
1375 return 0;
1376 }
1377 else
1378 {
1379 // unrecognized option beginning with -verbose
1380 return -1;
1381 }
1382 return 0;
1383 }
1384
1385 #ifdef INTERPRETER
1386 // This function loads the agent functions for JVMTI from the library indicated
1387 // by name. It returns a negative value on failure, the value of which
1388 // indicates where ltdl failed, it also prints an error message.
1389 static jint
1390 load_jvmti_agent (const char *name)
1391 {
1392 #ifdef USE_LTDL
1393 if (lt_dlinit ())
1394 {
1395 fprintf (stderr,
1396 "libgcj: Error in ltdl init while loading agent library.\n");
1397 return -1;
1398 }
1399
1400 lt_dlhandle lib = lt_dlopenext (name);
1401 if (!lib)
1402 {
1403 fprintf (stderr,
1404 "libgcj: Error opening agent library.\n");
1405 return -2;
1406 }
1407
1408 if (lib)
1409 {
1410 jvmti_agentonload
1411 = (jvmti_agent_onload_func *) lt_dlsym (lib, "Agent_OnLoad");
1412
1413 if (!jvmti_agentonload)
1414 {
1415 fprintf (stderr,
1416 "libgcj: Error finding agent function in library %s.\n",
1417 name);
1418 lt_dlclose (lib);
1419 lib = NULL;
1420 return -4;
1421 }
1422 else
1423 {
1424 jvmti_agentonunload
1425 = (jvmti_agent_onunload_func *) lt_dlsym (lib, "Agent_OnUnload");
1426
1427 return 0;
1428 }
1429 }
1430 else
1431 {
1432 fprintf (stderr, "libgcj: Library %s not found in library path.\n", name);
1433 return -3;
1434 }
1435
1436 #endif /* USE_LTDL */
1437
1438 // If LTDL cannot be used, return an error code indicating this.
1439 return -99;
1440 }
1441 #endif // INTERPRETER
1442
1443 static jint
1444 parse_init_args (JvVMInitArgs* vm_args)
1445 {
1446 // if _Jv_Compiler_Properties is non-NULL then it needs to be
1447 // re-allocated dynamically.
1448 if (_Jv_Compiler_Properties)
1449 {
1450 const char** props = _Jv_Compiler_Properties;
1451 _Jv_Compiler_Properties = NULL;
1452
1453 for (int i = 0; props[i]; i++)
1454 {
1455 _Jv_Compiler_Properties = (const char**) _Jv_Realloc
1456 (_Jv_Compiler_Properties,
1457 (_Jv_Properties_Count + 1) * sizeof (const char*));
1458 _Jv_Compiler_Properties[_Jv_Properties_Count++] = props[i];
1459 }
1460 }
1461
1462 if (vm_args == NULL)
1463 return 0;
1464
1465 for (int i = 0; i < vm_args->nOptions; ++i)
1466 {
1467 char* option_string = vm_args->options[i].optionString;
1468
1469 if (! strcmp (option_string, "vfprintf")
1470 || ! strcmp (option_string, "exit")
1471 || ! strcmp (option_string, "abort"))
1472 {
1473 // FIXME: we are required to recognize these, but for
1474 // now we don't handle them in any way.
1475 continue;
1476 }
1477 else if (! strncmp (option_string,
1478 "-verbose", sizeof ("-verbose") - 1))
1479 {
1480 jint result = parse_verbose_args (option_string,
1481 vm_args->ignoreUnrecognized);
1482 if (result < 0)
1483 return result;
1484 }
1485 else if (! strncmp (option_string, "-D", 2))
1486 {
1487 _Jv_Compiler_Properties = (const char**) _Jv_Realloc
1488 (_Jv_Compiler_Properties,
1489 (_Jv_Properties_Count + 1) * sizeof (char*));
1490
1491 _Jv_Compiler_Properties[_Jv_Properties_Count++] =
1492 strdup (option_string + 2);
1493
1494 continue;
1495 }
1496 #ifdef INTERPRETER
1497 else if (! strncmp (option_string, "-agentlib", sizeof ("-agentlib") - 1))
1498 {
1499 char *strPtr;
1500
1501 if (strlen(option_string) > (sizeof ("-agentlib:") - 1))
1502 strPtr = &option_string[sizeof ("-agentlib:") - 1];
1503 else
1504 {
1505 fprintf (stderr,
1506 "libgcj: Malformed agentlib argument %s: expected lib name\n",
1507 option_string);
1508 return -1;
1509 }
1510
1511 // These are optional arguments to pass to the agent library.
1512 jvmti_agent_opts = strchr (strPtr, '=');
1513
1514 if (! strncmp (strPtr, "jdwp", 4))
1515 {
1516 // We want to run JDWP here so set the correct variables.
1517 remoteDebug = true;
1518 jdwpOptions = ++jvmti_agent_opts;
1519 }
1520 else
1521 {
1522 jint nameLength;
1523
1524 if (jvmti_agent_opts == NULL)
1525 nameLength = strlen (strPtr);
1526 else
1527 {
1528 nameLength = jvmti_agent_opts - strPtr;
1529 jvmti_agent_opts++;
1530 }
1531
1532 char lib_name[nameLength + 3 + 1];
1533 strcpy (lib_name, "lib");
1534 strncat (lib_name, strPtr, nameLength);
1535
1536 jint result = load_jvmti_agent (lib_name);
1537
1538 if (result < 0)
1539 {
1540 return -1;
1541 }
1542
1543 // Mark JVMTI active
1544 JVMTI::enabled = true;
1545 }
1546
1547 continue;
1548 }
1549 else if (! strncmp (option_string, "-agentpath:",
1550 sizeof ("-agentpath:") - 1))
1551 {
1552 char *strPtr;
1553
1554 if (strlen(option_string) > 10)
1555 strPtr = &option_string[10];
1556 else
1557 {
1558 fprintf (stderr,
1559 "libgcj: Malformed agentlib argument %s: expected lib path\n",
1560 option_string);
1561 return -1;
1562 }
1563
1564 // These are optional arguments to pass to the agent library.
1565 jvmti_agent_opts = strchr (strPtr, '=');
1566
1567 jint nameLength;
1568
1569 if (jvmti_agent_opts == NULL)
1570 nameLength = strlen (strPtr);
1571 else
1572 {
1573 nameLength = jvmti_agent_opts - strPtr;
1574 jvmti_agent_opts++;
1575 }
1576
1577 char lib_name[nameLength + 3 + 1];
1578 strcpy (lib_name, "lib");
1579 strncat (lib_name, strPtr, nameLength);
1580 jint result = load_jvmti_agent (strPtr);
1581
1582 if (result < 0)
1583 {
1584 return -1;
1585 }
1586
1587 // Mark JVMTI active
1588 JVMTI::enabled = true;
1589 continue;
1590 }
1591 #endif // INTERPRETER
1592 else
1593 {
1594 int r = -1;
1595 if (option_string[0] == '_')
1596 r = parse_x_arg (option_string + 1);
1597 else if (! strncmp (option_string, "-X", 2))
1598 r = parse_x_arg (option_string + 2);
1599
1600 if (r == -1 && ! vm_args->ignoreUnrecognized)
1601 {
1602 fprintf (stderr, "libgcj: unknown option: %s\n", option_string);
1603 return -1;
1604 }
1605 }
1606 }
1607 return 0;
1608 }
1609
1610 jint
1611 _Jv_CreateJavaVM (JvVMInitArgs* vm_args)
1612 {
1613 using namespace gcj;
1614
1615 if (runtimeInitialized)
1616 return -1;
1617
1618 runtimeInitialized = true;
1619 startTime = _Jv_platform_gettimeofday();
1620
1621 jint result = parse_init_args (vm_args);
1622 if (result < 0)
1623 return -1;
1624
1625 PROCESS_GCJ_PROPERTIES;
1626
1627 /* Threads must be initialized before the GC, so that it inherits the
1628 signal mask. */
1629 _Jv_InitThreads ();
1630 _Jv_InitGC ();
1631 _Jv_InitializeSyncMutex ();
1632
1633 #ifdef INTERPRETER
1634 _Jv_InitInterpreter ();
1635 #endif
1636
1637 #ifdef HANDLE_SEGV
1638 INIT_SEGV;
1639 #endif
1640
1641 #ifdef HANDLE_FPE
1642 INIT_FPE;
1643 #endif
1644
1645 /* Initialize Utf8 constants declared in jvm.h. */
1646 void_signature = _Jv_makeUtf8Const ("()V", 3);
1647 clinit_name = _Jv_makeUtf8Const ("<clinit>", 8);
1648 init_name = _Jv_makeUtf8Const ("<init>", 6);
1649 finit_name = _Jv_makeUtf8Const ("finit$", 6);
1650
1651 /* Initialize built-in classes to represent primitive TYPEs. */
1652 _Jv_InitPrimClass (&_Jv_byteClass, "byte", 'B', 1);
1653 _Jv_InitPrimClass (&_Jv_shortClass, "short", 'S', 2);
1654 _Jv_InitPrimClass (&_Jv_intClass, "int", 'I', 4);
1655 _Jv_InitPrimClass (&_Jv_longClass, "long", 'J', 8);
1656 _Jv_InitPrimClass (&_Jv_booleanClass, "boolean", 'Z', 1);
1657 _Jv_InitPrimClass (&_Jv_charClass, "char", 'C', 2);
1658 _Jv_InitPrimClass (&_Jv_floatClass, "float", 'F', 4);
1659 _Jv_InitPrimClass (&_Jv_doubleClass, "double", 'D', 8);
1660 _Jv_InitPrimClass (&_Jv_voidClass, "void", 'V', 0);
1661
1662 // We have to initialize this fairly early, to avoid circular class
1663 // initialization. In particular we want to start the
1664 // initialization of ClassLoader before we start the initialization
1665 // of VMClassLoader.
1666 _Jv_InitClass (&java::lang::ClassLoader::class$);
1667
1668 // Set up the system class loader and the bootstrap class loader.
1669 gnu::gcj::runtime::ExtensionClassLoader::initialize();
1670 java::lang::VMClassLoader::initialize(JvNewStringLatin1(TOOLEXECLIBDIR));
1671
1672 _Jv_RegisterBootstrapPackages();
1673
1674 no_memory = new java::lang::OutOfMemoryError;
1675
1676 #ifdef USE_LTDL
1677 LTDL_SET_PRELOADED_SYMBOLS ();
1678 #endif
1679
1680 _Jv_platform_initialize ();
1681
1682 _Jv_JNI_Init ();
1683
1684 #ifdef INTERPRETER
1685 _Jv_JVMTI_Init ();
1686 #endif
1687
1688 _Jv_GCInitializeFinalizers (&::gnu::gcj::runtime::FinalizerThread::finalizerReady);
1689
1690 // Start the GC finalizer thread. A VirtualMachineError can be
1691 // thrown by the runtime if, say, threads aren't available.
1692 try
1693 {
1694 using namespace gnu::gcj::runtime;
1695 FinalizerThread *ft = new FinalizerThread ();
1696 ft->start ();
1697 }
1698 catch (java::lang::VirtualMachineError *ignore)
1699 {
1700 }
1701
1702 runtimeInitialized = true;
1703
1704 return 0;
1705 }
1706
1707 void
1708 _Jv_RunMain (JvVMInitArgs *vm_args, jclass klass, const char *name, int argc,
1709 const char **argv, bool is_jar)
1710 {
1711 #ifndef DISABLE_MAIN_ARGS
1712 _Jv_SetArgs (argc, argv);
1713 #endif
1714
1715 java::lang::Runtime *runtime = NULL;
1716
1717 try
1718 {
1719 if (_Jv_CreateJavaVM (vm_args) < 0)
1720 {
1721 fprintf (stderr, "libgcj: couldn't create virtual machine\n");
1722 exit (1);
1723 }
1724
1725 if (vm_args == NULL)
1726 gcj::vmArgs = JvConvertArgv(0, NULL);
1727 else
1728 {
1729 const char* vmArgs[vm_args->nOptions];
1730 const char** vmPtr = vmArgs;
1731 struct _Jv_VMOption* optionPtr = vm_args->options;
1732 for (int i = 0; i < vm_args->nOptions; ++i)
1733 *vmPtr++ = (*optionPtr++).optionString;
1734 gcj::vmArgs = JvConvertArgv(vm_args->nOptions, vmArgs);
1735 }
1736
1737 // Get the Runtime here. We want to initialize it before searching
1738 // for `main'; that way it will be set up if `main' is a JNI method.
1739 runtime = java::lang::Runtime::getRuntime ();
1740
1741 #ifdef DISABLE_MAIN_ARGS
1742 arg_vec = JvConvertArgv (0, 0);
1743 #else
1744 arg_vec = JvConvertArgv (argc - 1, argv + 1);
1745 #endif
1746
1747 using namespace gnu::java::lang;
1748 if (klass)
1749 main_thread = new MainThread (klass, arg_vec);
1750 else
1751 main_thread = new MainThread (JvNewStringUTF (name),
1752 arg_vec, is_jar);
1753 _Jv_AttachCurrentThread (main_thread);
1754
1755 #ifdef INTERPRETER
1756 // Start JVMTI if an agent function has been found.
1757 if (jvmti_agentonload)
1758 (*jvmti_agentonload) (_Jv_GetJavaVM (), jvmti_agent_opts, NULL);
1759
1760 // Start JDWP
1761 if (remoteDebug)
1762 {
1763 using namespace gnu::classpath::jdwp;
1764 VMVirtualMachine::initialize ();
1765 Jdwp *jdwp = new Jdwp ();
1766 jdwp->setDaemon (true);
1767 jdwp->configure (JvNewStringLatin1 (jdwpOptions));
1768 jdwp->start ();
1769
1770 // Wait for JDWP to initialize and start
1771 jdwp->join ();
1772 }
1773 // Send VMInit
1774 if (JVMTI_REQUESTED_EVENT (VMInit))
1775 _Jv_JVMTI_PostEvent (JVMTI_EVENT_VM_INIT, main_thread);
1776 #endif // INTERPRETER
1777 }
1778 catch (java::lang::Throwable *t)
1779 {
1780 java::lang::System::err->println (JvNewStringLatin1
1781 ("Exception during runtime initialization"));
1782 t->printStackTrace();
1783 if (runtime)
1784 java::lang::Runtime::exitNoChecksAccessor (1);
1785 // In case the runtime creation failed.
1786 ::exit (1);
1787 }
1788
1789 _Jv_ThreadRun (main_thread);
1790
1791 #ifdef INTERPRETER
1792 // Send VMDeath
1793 if (JVMTI_REQUESTED_EVENT (VMDeath))
1794 {
1795 java::lang::Thread *thread = java::lang::Thread::currentThread ();
1796 JNIEnv *jni_env = _Jv_GetCurrentJNIEnv ();
1797 _Jv_JVMTI_PostEvent (JVMTI_EVENT_VM_DEATH, thread, jni_env);
1798 }
1799
1800 // Run JVMTI AgentOnUnload if it exists and an agent is loaded.
1801 if (jvmti_agentonunload)
1802 (*jvmti_agentonunload) (_Jv_GetJavaVM ());
1803 #endif // INTERPRETER
1804
1805 // If we got here then something went wrong, as MainThread is not
1806 // supposed to terminate.
1807 ::exit (1);
1808 }
1809
1810 void
1811 _Jv_RunMain (jclass klass, const char *name, int argc, const char **argv,
1812 bool is_jar)
1813 {
1814 _Jv_RunMain (NULL, klass, name, argc, argv, is_jar);
1815 }
1816
1817 void
1818 JvRunMain (jclass klass, int argc, const char **argv)
1819 {
1820 _Jv_RunMain (klass, NULL, argc, argv, false);
1821 }
1822
1823 void
1824 JvRunMainName (const char *name, int argc, const char **argv)
1825 {
1826 _Jv_RunMain (NULL, name, argc, argv, false);
1827 }
1828
1829 \f
1830
1831 // Parse a string and return a heap size.
1832 static size_t
1833 parse_memory_size (const char *spec)
1834 {
1835 char *end;
1836 unsigned long val = strtoul (spec, &end, 10);
1837 if (*end == 'k' || *end == 'K')
1838 val *= 1024;
1839 else if (*end == 'm' || *end == 'M')
1840 val *= 1048576;
1841 return (size_t) val;
1842 }
1843
1844 // Set the initial heap size. This might be ignored by the GC layer.
1845 // This must be called before _Jv_RunMain.
1846 void
1847 _Jv_SetInitialHeapSize (const char *arg)
1848 {
1849 size_t size = parse_memory_size (arg);
1850 _Jv_GCSetInitialHeapSize (size);
1851 }
1852
1853 // Set the maximum heap size. This might be ignored by the GC layer.
1854 // This must be called before _Jv_RunMain.
1855 void
1856 _Jv_SetMaximumHeapSize (const char *arg)
1857 {
1858 size_t size = parse_memory_size (arg);
1859 _Jv_GCSetMaximumHeapSize (size);
1860 }
1861
1862 void
1863 _Jv_SetStackSize (const char *arg)
1864 {
1865 size_t size = parse_memory_size (arg);
1866 gcj::stack_size = size;
1867 }
1868
1869 void *
1870 _Jv_Malloc (jsize size)
1871 {
1872 if (__builtin_expect (size == 0, false))
1873 size = 1;
1874 void *ptr = malloc ((size_t) size);
1875 if (__builtin_expect (ptr == NULL, false))
1876 throw no_memory;
1877 return ptr;
1878 }
1879
1880 void *
1881 _Jv_Realloc (void *ptr, jsize size)
1882 {
1883 if (__builtin_expect (size == 0, false))
1884 size = 1;
1885 ptr = realloc (ptr, (size_t) size);
1886 if (__builtin_expect (ptr == NULL, false))
1887 throw no_memory;
1888 return ptr;
1889 }
1890
1891 void *
1892 _Jv_MallocUnchecked (jsize size)
1893 {
1894 if (__builtin_expect (size == 0, false))
1895 size = 1;
1896 return malloc ((size_t) size);
1897 }
1898
1899 void
1900 _Jv_Free (void* ptr)
1901 {
1902 return free (ptr);
1903 }
1904
1905 \f
1906
1907 // In theory, these routines can be #ifdef'd away on machines which
1908 // support divide overflow signals. However, we never know if some
1909 // code might have been compiled with "-fuse-divide-subroutine", so we
1910 // always include them in libgcj.
1911
1912 jint
1913 _Jv_divI (jint dividend, jint divisor)
1914 {
1915 if (__builtin_expect (divisor == 0, false))
1916 {
1917 java::lang::ArithmeticException *arithexception
1918 = new java::lang::ArithmeticException (JvNewStringLatin1 ("/ by zero"));
1919 throw arithexception;
1920 }
1921
1922 if (dividend == (jint) 0x80000000L && divisor == -1)
1923 return dividend;
1924
1925 return dividend / divisor;
1926 }
1927
1928 jint
1929 _Jv_remI (jint dividend, jint divisor)
1930 {
1931 if (__builtin_expect (divisor == 0, false))
1932 {
1933 java::lang::ArithmeticException *arithexception
1934 = new java::lang::ArithmeticException (JvNewStringLatin1 ("/ by zero"));
1935 throw arithexception;
1936 }
1937
1938 if (dividend == (jint) 0x80000000L && divisor == -1)
1939 return 0;
1940
1941 return dividend % divisor;
1942 }
1943
1944 jlong
1945 _Jv_divJ (jlong dividend, jlong divisor)
1946 {
1947 if (__builtin_expect (divisor == 0, false))
1948 {
1949 java::lang::ArithmeticException *arithexception
1950 = new java::lang::ArithmeticException (JvNewStringLatin1 ("/ by zero"));
1951 throw arithexception;
1952 }
1953
1954 if (dividend == (jlong) 0x8000000000000000LL && divisor == -1)
1955 return dividend;
1956
1957 return dividend / divisor;
1958 }
1959
1960 jlong
1961 _Jv_remJ (jlong dividend, jlong divisor)
1962 {
1963 if (__builtin_expect (divisor == 0, false))
1964 {
1965 java::lang::ArithmeticException *arithexception
1966 = new java::lang::ArithmeticException (JvNewStringLatin1 ("/ by zero"));
1967 throw arithexception;
1968 }
1969
1970 if (dividend == (jlong) 0x8000000000000000LL && divisor == -1)
1971 return 0;
1972
1973 return dividend % divisor;
1974 }
1975
1976 \f
1977
1978 // Return true if SELF_KLASS can access a field or method in
1979 // OTHER_KLASS. The field or method's access flags are specified in
1980 // FLAGS.
1981 jboolean
1982 _Jv_CheckAccess (jclass self_klass, jclass other_klass, jint flags)
1983 {
1984 using namespace java::lang::reflect;
1985 return ((self_klass == other_klass)
1986 || ((flags & Modifier::PUBLIC) != 0)
1987 || (((flags & Modifier::PROTECTED) != 0)
1988 && _Jv_IsAssignableFromSlow (self_klass, other_klass))
1989 || (((flags & Modifier::PRIVATE) == 0)
1990 && _Jv_ClassNameSamePackage (self_klass->name,
1991 other_klass->name)));
1992 }
1993
1994 // Prepend GCJ_VERSIONED_LIBDIR to a module search path stored in a C
1995 // char array, if the path is not already prefixed by
1996 // GCJ_VERSIONED_LIBDIR. Return a newly JvMalloc'd char buffer. The
1997 // result should be freed using JvFree.
1998 char*
1999 _Jv_PrependVersionedLibdir (char* libpath)
2000 {
2001 char* retval = 0;
2002
2003 if (libpath && libpath[0] != '\0')
2004 {
2005 if (! strncmp (libpath,
2006 GCJ_VERSIONED_LIBDIR,
2007 sizeof (GCJ_VERSIONED_LIBDIR) - 1))
2008 {
2009 // LD_LIBRARY_PATH is already prefixed with
2010 // GCJ_VERSIONED_LIBDIR.
2011 retval = (char*) _Jv_Malloc (strlen (libpath) + 1);
2012 strcpy (retval, libpath);
2013 }
2014 else
2015 {
2016 // LD_LIBRARY_PATH is not prefixed with
2017 // GCJ_VERSIONED_LIBDIR.
2018 char path_sep[2];
2019 path_sep[0] = (char) _Jv_platform_path_separator;
2020 path_sep[1] = '\0';
2021 jsize total = ((sizeof (GCJ_VERSIONED_LIBDIR) - 1)
2022 + 1 /* path separator */ + strlen (libpath) + 1);
2023 retval = (char*) _Jv_Malloc (total);
2024 strcpy (retval, GCJ_VERSIONED_LIBDIR);
2025 strcat (retval, path_sep);
2026 strcat (retval, libpath);
2027 }
2028 }
2029 else
2030 {
2031 // LD_LIBRARY_PATH was not specified or is empty.
2032 retval = (char*) _Jv_Malloc (sizeof (GCJ_VERSIONED_LIBDIR));
2033 strcpy (retval, GCJ_VERSIONED_LIBDIR);
2034 }
2035
2036 return retval;
2037 }