1999-04-07 22:42:40 +08:00
|
|
|
|
// natRuntime.cc - Implementation of native side of Runtime class.
|
|
|
|
|
|
2002-04-11 04:36:04 +08:00
|
|
|
|
/* Copyright (C) 1998, 1999, 2000, 2001, 2002 Free Software Foundation
|
1999-04-07 22:42:40 +08:00
|
|
|
|
|
|
|
|
|
This file is part of libgcj.
|
|
|
|
|
|
|
|
|
|
This software is copyrighted work licensed under the terms of the
|
|
|
|
|
Libgcj License. Please consult the file "LIBGCJ_LICENSE" for
|
|
|
|
|
details. */
|
|
|
|
|
|
|
|
|
|
#include <config.h>
|
|
|
|
|
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
|
1999-09-11 06:03:10 +08:00
|
|
|
|
#include <gcj/cni.h>
|
1999-04-07 22:42:40 +08:00
|
|
|
|
#include <jvm.h>
|
2002-05-14 04:10:37 +08:00
|
|
|
|
#include <java-props.h>
|
|
|
|
|
#include <java/lang/Long.h>
|
1999-04-07 22:42:40 +08:00
|
|
|
|
#include <java/lang/Runtime.h>
|
1999-05-05 22:19:24 +08:00
|
|
|
|
#include <java/lang/UnknownError.h>
|
|
|
|
|
#include <java/lang/UnsatisfiedLinkError.h>
|
2000-09-05 00:55:48 +08:00
|
|
|
|
#include <gnu/gcj/runtime/FileDeleter.h>
|
2001-10-11 06:25:43 +08:00
|
|
|
|
#include <gnu/gcj/runtime/FinalizerThread.h>
|
2002-08-14 09:07:59 +08:00
|
|
|
|
#include <java/io/File.h>
|
2002-05-14 04:10:37 +08:00
|
|
|
|
#include <java/util/Properties.h>
|
|
|
|
|
#include <java/util/TimeZone.h>
|
|
|
|
|
#include <java/lang/StringBuffer.h>
|
|
|
|
|
#include <java/lang/Process.h>
|
|
|
|
|
#include <java/lang/ConcreteProcess.h>
|
2002-12-10 09:39:32 +08:00
|
|
|
|
#include <java/lang/ClassLoader.h>
|
|
|
|
|
#include <gnu/gcj/runtime/StackTrace.h>
|
2003-01-15 03:26:21 +08:00
|
|
|
|
#include <java/lang/ArrayIndexOutOfBoundsException.h>
|
1999-05-05 22:19:24 +08:00
|
|
|
|
|
2000-02-08 08:26:58 +08:00
|
|
|
|
#include <jni.h>
|
|
|
|
|
|
2002-05-14 04:10:37 +08:00
|
|
|
|
#include "platform.h"
|
|
|
|
|
|
|
|
|
|
#ifdef HAVE_PWD_H
|
|
|
|
|
#include <pwd.h>
|
|
|
|
|
#endif
|
|
|
|
|
#include <errno.h>
|
|
|
|
|
|
|
|
|
|
#ifdef HAVE_UNAME
|
|
|
|
|
#include <sys/utsname.h>
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
#ifdef HAVE_LOCALE_H
|
|
|
|
|
#include <locale.h>
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
#ifdef HAVE_LANGINFO_H
|
|
|
|
|
#include <langinfo.h>
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
1999-05-05 22:19:24 +08:00
|
|
|
|
#ifdef USE_LTDL
|
|
|
|
|
#include <ltdl.h>
|
2000-01-18 03:22:20 +08:00
|
|
|
|
|
|
|
|
|
/* FIXME: we don't always need this. The next libtool will let us use
|
|
|
|
|
AC_LTDL_PREOPEN to see if we do. */
|
2000-09-14 15:56:28 +08:00
|
|
|
|
extern const lt_dlsymlist lt_preloaded_symbols[1] = { { 0, 0 } };
|
2000-02-05 04:49:27 +08:00
|
|
|
|
|
|
|
|
|
// We keep track of all the libraries loaded by this application. For
|
|
|
|
|
// now we use them to look up symbols for JNI. `libraries_size' holds
|
|
|
|
|
// the total size of the buffer. `libraries_count' is the number of
|
|
|
|
|
// items which are in use.
|
|
|
|
|
static int libraries_size;
|
|
|
|
|
static int libraries_count;
|
|
|
|
|
static lt_dlhandle *libraries;
|
|
|
|
|
|
|
|
|
|
static void
|
|
|
|
|
add_library (lt_dlhandle lib)
|
|
|
|
|
{
|
|
|
|
|
if (libraries_count == libraries_size)
|
|
|
|
|
{
|
|
|
|
|
int ns = libraries_size * 2;
|
|
|
|
|
if (ns == 0)
|
|
|
|
|
ns = 10;
|
|
|
|
|
lt_dlhandle *n = (lt_dlhandle *) _Jv_Malloc (ns * sizeof (lt_dlhandle));
|
|
|
|
|
if (libraries)
|
|
|
|
|
{
|
|
|
|
|
memcpy (n, libraries, libraries_size * sizeof (lt_dlhandle));
|
|
|
|
|
_Jv_Free (libraries);
|
|
|
|
|
}
|
|
|
|
|
libraries = n;
|
|
|
|
|
libraries_size = ns;
|
|
|
|
|
for (int i = libraries_count; i < libraries_size; ++i)
|
|
|
|
|
libraries[i] = NULL;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
libraries[libraries_count++] = lib;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void *
|
|
|
|
|
_Jv_FindSymbolInExecutable (const char *symname)
|
|
|
|
|
{
|
|
|
|
|
for (int i = 0; i < libraries_count; ++i)
|
|
|
|
|
{
|
|
|
|
|
void *r = lt_dlsym (libraries[i], symname);
|
|
|
|
|
if (r)
|
|
|
|
|
return r;
|
|
|
|
|
}
|
|
|
|
|
|
2001-06-16 07:22:02 +08:00
|
|
|
|
return NULL;
|
2000-02-05 04:49:27 +08:00
|
|
|
|
}
|
|
|
|
|
|
2002-10-25 11:28:00 +08:00
|
|
|
|
void
|
|
|
|
|
_Jv_SetDLLSearchPath (const char *path)
|
|
|
|
|
{
|
|
|
|
|
lt_dlsetsearchpath (path);
|
|
|
|
|
}
|
|
|
|
|
|
2001-06-03 03:40:53 +08:00
|
|
|
|
#else
|
|
|
|
|
|
|
|
|
|
void *
|
2002-10-25 11:28:00 +08:00
|
|
|
|
_Jv_FindSymbolInExecutable (const char *)
|
2001-06-03 03:40:53 +08:00
|
|
|
|
{
|
|
|
|
|
return NULL;
|
|
|
|
|
}
|
|
|
|
|
|
2002-10-25 11:28:00 +08:00
|
|
|
|
void
|
|
|
|
|
_Jv_SetDLLSearchPath (const char *)
|
|
|
|
|
{
|
|
|
|
|
// Nothing.
|
|
|
|
|
}
|
|
|
|
|
|
2000-02-05 04:49:27 +08:00
|
|
|
|
#endif /* USE_LTDL */
|
1999-04-07 22:42:40 +08:00
|
|
|
|
|
2002-05-14 04:10:37 +08:00
|
|
|
|
|
1999-04-07 22:42:40 +08:00
|
|
|
|
|
2001-03-12 15:40:17 +08:00
|
|
|
|
void
|
2002-05-14 04:10:37 +08:00
|
|
|
|
java::lang::Runtime::exitInternal (jint status)
|
2001-03-12 15:40:17 +08:00
|
|
|
|
{
|
1999-04-07 22:42:40 +08:00
|
|
|
|
// Make status right for Unix. This is perhaps strange.
|
|
|
|
|
if (status < 0 || status > 255)
|
|
|
|
|
status = 255;
|
|
|
|
|
|
2002-05-14 04:10:37 +08:00
|
|
|
|
if (finalizeOnExit)
|
1999-04-07 22:42:40 +08:00
|
|
|
|
_Jv_RunAllFinalizers ();
|
|
|
|
|
|
2000-09-05 00:55:48 +08:00
|
|
|
|
// Delete all files registered with File.deleteOnExit()
|
|
|
|
|
gnu::gcj::runtime::FileDeleter::deleteOnExitNow ();
|
|
|
|
|
|
1999-04-07 22:42:40 +08:00
|
|
|
|
::exit (status);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
jlong
|
|
|
|
|
java::lang::Runtime::freeMemory (void)
|
|
|
|
|
{
|
|
|
|
|
return _Jv_GCFreeMemory ();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
java::lang::Runtime::gc (void)
|
|
|
|
|
{
|
|
|
|
|
_Jv_RunGC ();
|
|
|
|
|
}
|
|
|
|
|
|
1999-05-05 22:19:24 +08:00
|
|
|
|
void
|
2000-02-08 08:26:58 +08:00
|
|
|
|
java::lang::Runtime::_load (jstring path, jboolean do_search)
|
1999-05-05 22:19:24 +08:00
|
|
|
|
{
|
|
|
|
|
JvSynchronize sync (this);
|
|
|
|
|
using namespace java::lang;
|
|
|
|
|
#ifdef USE_LTDL
|
2000-01-18 03:22:20 +08:00
|
|
|
|
jint len = _Jv_GetStringUTFLength (path);
|
2002-12-10 09:39:32 +08:00
|
|
|
|
char buf[len + 1 + strlen (_Jv_platform_solib_prefix)
|
|
|
|
|
+ strlen (_Jv_platform_solib_suffix)];
|
2000-02-15 15:53:11 +08:00
|
|
|
|
int offset = 0;
|
|
|
|
|
if (do_search)
|
|
|
|
|
{
|
2002-12-10 09:39:32 +08:00
|
|
|
|
strcpy (buf, _Jv_platform_solib_prefix);
|
|
|
|
|
offset = strlen (_Jv_platform_solib_prefix);
|
2000-02-15 15:53:11 +08:00
|
|
|
|
}
|
|
|
|
|
jsize total = JvGetStringUTFRegion (path, 0, path->length(), &buf[offset]);
|
|
|
|
|
buf[offset + total] = '\0';
|
2002-12-10 09:39:32 +08:00
|
|
|
|
|
|
|
|
|
char *lib_name = buf;
|
|
|
|
|
|
|
|
|
|
if (do_search)
|
|
|
|
|
{
|
|
|
|
|
ClassLoader *sys = ClassLoader::getSystemClassLoader();
|
|
|
|
|
ClassLoader *look = NULL;
|
|
|
|
|
gnu::gcj::runtime::StackTrace *t = new gnu::gcj::runtime::StackTrace(10);
|
2003-01-15 03:26:21 +08:00
|
|
|
|
try
|
|
|
|
|
{
|
|
|
|
|
for (int i = 0; i < 10; ++i)
|
2002-12-10 09:39:32 +08:00
|
|
|
|
{
|
2003-01-15 03:26:21 +08:00
|
|
|
|
jclass klass = t->classAt(i);
|
|
|
|
|
if (klass != NULL)
|
2002-12-10 09:39:32 +08:00
|
|
|
|
{
|
2003-01-15 03:26:21 +08:00
|
|
|
|
ClassLoader *loader = klass->getClassLoaderInternal();
|
|
|
|
|
if (loader != NULL && loader != sys)
|
|
|
|
|
{
|
|
|
|
|
look = loader;
|
|
|
|
|
break;
|
|
|
|
|
}
|
2002-12-10 09:39:32 +08:00
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
2003-01-15 03:26:21 +08:00
|
|
|
|
catch (::java::lang::ArrayIndexOutOfBoundsException *e)
|
|
|
|
|
{
|
|
|
|
|
}
|
|
|
|
|
|
2002-12-10 09:39:32 +08:00
|
|
|
|
if (look != NULL)
|
|
|
|
|
{
|
|
|
|
|
// Don't include solib prefix in string passed to
|
|
|
|
|
// findLibrary.
|
|
|
|
|
jstring name = look->findLibrary(JvNewStringUTF(&buf[offset]));
|
|
|
|
|
if (name != NULL)
|
|
|
|
|
{
|
|
|
|
|
len = _Jv_GetStringUTFLength (name);
|
|
|
|
|
lib_name = (char *) _Jv_AllocBytes(len + 1);
|
|
|
|
|
total = JvGetStringUTFRegion (name, 0,
|
|
|
|
|
name->length(), lib_name);
|
|
|
|
|
lib_name[total] = '\0';
|
|
|
|
|
// Don't append suffixes any more; we have the full file
|
|
|
|
|
// name.
|
|
|
|
|
do_search = false;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2001-10-16 16:35:17 +08:00
|
|
|
|
lt_dlhandle h;
|
1999-05-05 22:19:24 +08:00
|
|
|
|
// FIXME: make sure path is absolute.
|
2001-10-16 16:35:17 +08:00
|
|
|
|
{
|
|
|
|
|
// Synchronize on java.lang.Class. This is to protect the class chain from
|
|
|
|
|
// concurrent modification by class registration calls which may be run
|
|
|
|
|
// during the dlopen().
|
|
|
|
|
JvSynchronize sync (&java::lang::Class::class$);
|
2002-12-10 09:39:32 +08:00
|
|
|
|
h = do_search ? lt_dlopenext (lib_name) : lt_dlopen (lib_name);
|
2001-10-16 16:35:17 +08:00
|
|
|
|
}
|
1999-05-05 22:19:24 +08:00
|
|
|
|
if (h == NULL)
|
|
|
|
|
{
|
|
|
|
|
const char *msg = lt_dlerror ();
|
2000-02-15 16:51:04 +08:00
|
|
|
|
jstring str = path->concat (JvNewStringLatin1 (": "));
|
|
|
|
|
str = str->concat (JvNewStringLatin1 (msg));
|
exception.cc (java_eh_info): Make value type jthrowable.
* exception.cc (java_eh_info): Make value type jthrowable.
(_Jv_type_matcher): Remove now unneeded cast.
(_Jv_Throw): Make argument type jthrowable. Munge name
for SJLJ_EXCEPTIONS here ...
* gcj/cni.h: ... not here.
(JvThrow): Remove.
* gcj/javaprims.h (_Jv_Throw, _Jv_Sjlj_Throw): Update declarations.
* defineclass.cc, interpret.cc, jni.cc, posix-threads.cc,
prims.cc, resolve.cc, gnu/gcj/runtime/natFirstThread.cc,
gnu/gcj/xlib/natDrawable.cc, gnu/gcj/xlib/natFont.cc,
gnu/gcj/xlib/natWMSizeHints.cc, gnu/gcj/xlib/natWindowAttributes.cc,
gnu/gcj/xlib/natXImage.cc, java/io/natFile.cc,
java/io/natFileDescriptorEcos.cc, java/io/natFileDescriptorPosix.cc,
java/io/natFileDescriptorWin32.cc, java/io/natFileWin32.cc,
java/lang/natClass.cc, java/lang/natClassLoader.cc,
java/lang/natDouble.cc, java/lang/natObject.cc,
java/lang/natPosixProcess.cc, java/lang/natRuntime.cc,
java/lang/natString.cc, java/lang/natSystem.cc,
java/lang/natThread.cc, java/lang/reflect/natArray.cc,
java/lang/reflect/natConstructor.cc, java/lang/reflect/natField.cc,
java/lang/reflect/natMethod.cc, java/util/zip/natDeflater.cc,
java/util/zip/natInflater.cc:
Use throw, not JvThrow or _Jv_Throw.
From-SVN: r40838
2001-03-26 15:05:32 +08:00
|
|
|
|
throw new UnsatisfiedLinkError (str);
|
1999-05-05 22:19:24 +08:00
|
|
|
|
}
|
|
|
|
|
|
2000-02-10 08:17:10 +08:00
|
|
|
|
add_library (h);
|
|
|
|
|
|
2000-02-08 08:26:58 +08:00
|
|
|
|
void *onload = lt_dlsym (h, "JNI_OnLoad");
|
|
|
|
|
if (onload != NULL)
|
1999-05-05 22:19:24 +08:00
|
|
|
|
{
|
2000-02-19 05:22:06 +08:00
|
|
|
|
JavaVM *vm = _Jv_GetJavaVM ();
|
|
|
|
|
if (vm == NULL)
|
|
|
|
|
{
|
|
|
|
|
// FIXME: what?
|
|
|
|
|
return;
|
|
|
|
|
}
|
|
|
|
|
jint vers = ((jint (*) (JavaVM *, void *)) onload) (vm, NULL);
|
2002-04-11 04:36:04 +08:00
|
|
|
|
if (vers != JNI_VERSION_1_1 && vers != JNI_VERSION_1_2
|
|
|
|
|
&& vers != JNI_VERSION_1_4)
|
2000-02-08 08:26:58 +08:00
|
|
|
|
{
|
|
|
|
|
// FIXME: unload the library.
|
exception.cc (java_eh_info): Make value type jthrowable.
* exception.cc (java_eh_info): Make value type jthrowable.
(_Jv_type_matcher): Remove now unneeded cast.
(_Jv_Throw): Make argument type jthrowable. Munge name
for SJLJ_EXCEPTIONS here ...
* gcj/cni.h: ... not here.
(JvThrow): Remove.
* gcj/javaprims.h (_Jv_Throw, _Jv_Sjlj_Throw): Update declarations.
* defineclass.cc, interpret.cc, jni.cc, posix-threads.cc,
prims.cc, resolve.cc, gnu/gcj/runtime/natFirstThread.cc,
gnu/gcj/xlib/natDrawable.cc, gnu/gcj/xlib/natFont.cc,
gnu/gcj/xlib/natWMSizeHints.cc, gnu/gcj/xlib/natWindowAttributes.cc,
gnu/gcj/xlib/natXImage.cc, java/io/natFile.cc,
java/io/natFileDescriptorEcos.cc, java/io/natFileDescriptorPosix.cc,
java/io/natFileDescriptorWin32.cc, java/io/natFileWin32.cc,
java/lang/natClass.cc, java/lang/natClassLoader.cc,
java/lang/natDouble.cc, java/lang/natObject.cc,
java/lang/natPosixProcess.cc, java/lang/natRuntime.cc,
java/lang/natString.cc, java/lang/natSystem.cc,
java/lang/natThread.cc, java/lang/reflect/natArray.cc,
java/lang/reflect/natConstructor.cc, java/lang/reflect/natField.cc,
java/lang/reflect/natMethod.cc, java/util/zip/natDeflater.cc,
java/util/zip/natInflater.cc:
Use throw, not JvThrow or _Jv_Throw.
From-SVN: r40838
2001-03-26 15:05:32 +08:00
|
|
|
|
throw new UnsatisfiedLinkError (JvNewStringLatin1 ("unrecognized version from JNI_OnLoad"));
|
2000-02-08 08:26:58 +08:00
|
|
|
|
}
|
1999-05-05 22:19:24 +08:00
|
|
|
|
}
|
|
|
|
|
#else
|
exception.cc (java_eh_info): Make value type jthrowable.
* exception.cc (java_eh_info): Make value type jthrowable.
(_Jv_type_matcher): Remove now unneeded cast.
(_Jv_Throw): Make argument type jthrowable. Munge name
for SJLJ_EXCEPTIONS here ...
* gcj/cni.h: ... not here.
(JvThrow): Remove.
* gcj/javaprims.h (_Jv_Throw, _Jv_Sjlj_Throw): Update declarations.
* defineclass.cc, interpret.cc, jni.cc, posix-threads.cc,
prims.cc, resolve.cc, gnu/gcj/runtime/natFirstThread.cc,
gnu/gcj/xlib/natDrawable.cc, gnu/gcj/xlib/natFont.cc,
gnu/gcj/xlib/natWMSizeHints.cc, gnu/gcj/xlib/natWindowAttributes.cc,
gnu/gcj/xlib/natXImage.cc, java/io/natFile.cc,
java/io/natFileDescriptorEcos.cc, java/io/natFileDescriptorPosix.cc,
java/io/natFileDescriptorWin32.cc, java/io/natFileWin32.cc,
java/lang/natClass.cc, java/lang/natClassLoader.cc,
java/lang/natDouble.cc, java/lang/natObject.cc,
java/lang/natPosixProcess.cc, java/lang/natRuntime.cc,
java/lang/natString.cc, java/lang/natSystem.cc,
java/lang/natThread.cc, java/lang/reflect/natArray.cc,
java/lang/reflect/natConstructor.cc, java/lang/reflect/natField.cc,
java/lang/reflect/natMethod.cc, java/util/zip/natDeflater.cc,
java/util/zip/natInflater.cc:
Use throw, not JvThrow or _Jv_Throw.
From-SVN: r40838
2001-03-26 15:05:32 +08:00
|
|
|
|
throw new UnknownError
|
|
|
|
|
(JvNewStringLatin1 (do_search
|
|
|
|
|
? "Runtime.loadLibrary not implemented"
|
|
|
|
|
: "Runtime.load not implemented"));
|
1999-05-05 22:19:24 +08:00
|
|
|
|
#endif /* USE_LTDL */
|
|
|
|
|
}
|
|
|
|
|
|
2000-01-18 03:22:20 +08:00
|
|
|
|
jboolean
|
|
|
|
|
java::lang::Runtime::loadLibraryInternal (jstring lib)
|
|
|
|
|
{
|
|
|
|
|
JvSynchronize sync (this);
|
|
|
|
|
using namespace java::lang;
|
|
|
|
|
#ifdef USE_LTDL
|
|
|
|
|
jint len = _Jv_GetStringUTFLength (lib);
|
|
|
|
|
char buf[len + 1];
|
|
|
|
|
jsize total = JvGetStringUTFRegion (lib, 0, lib->length(), buf);
|
|
|
|
|
buf[total] = '\0';
|
|
|
|
|
// FIXME: make sure path is absolute.
|
|
|
|
|
lt_dlhandle h = lt_dlopenext (buf);
|
2000-02-10 08:17:10 +08:00
|
|
|
|
if (h != NULL)
|
|
|
|
|
add_library (h);
|
2000-01-18 03:22:20 +08:00
|
|
|
|
return h != NULL;
|
|
|
|
|
#else
|
|
|
|
|
return false;
|
|
|
|
|
#endif /* USE_LTDL */
|
|
|
|
|
}
|
|
|
|
|
|
1999-05-05 22:19:24 +08:00
|
|
|
|
void
|
|
|
|
|
java::lang::Runtime::init (void)
|
|
|
|
|
{
|
|
|
|
|
#ifdef USE_LTDL
|
|
|
|
|
lt_dlinit ();
|
2001-06-16 07:22:02 +08:00
|
|
|
|
lt_dlhandle self = lt_dlopen (NULL);
|
|
|
|
|
if (self != NULL)
|
|
|
|
|
add_library (self);
|
1999-05-05 22:19:24 +08:00
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
1999-04-07 22:42:40 +08:00
|
|
|
|
void
|
|
|
|
|
java::lang::Runtime::runFinalization (void)
|
|
|
|
|
{
|
2001-10-11 06:25:43 +08:00
|
|
|
|
gnu::gcj::runtime::FinalizerThread::finalizerReady ();
|
1999-04-07 22:42:40 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
jlong
|
|
|
|
|
java::lang::Runtime::totalMemory (void)
|
|
|
|
|
{
|
|
|
|
|
return _Jv_GCTotalMemory ();
|
|
|
|
|
}
|
|
|
|
|
|
2002-05-14 04:10:37 +08:00
|
|
|
|
jlong
|
|
|
|
|
java::lang::Runtime::maxMemory (void)
|
|
|
|
|
{
|
|
|
|
|
// We don't have a maximum. FIXME: we might if we ask the GC for
|
|
|
|
|
// one.
|
|
|
|
|
return Long::MAX_VALUE;
|
|
|
|
|
}
|
|
|
|
|
|
1999-04-07 22:42:40 +08:00
|
|
|
|
void
|
|
|
|
|
java::lang::Runtime::traceInstructions (jboolean)
|
|
|
|
|
{
|
|
|
|
|
// Do nothing.
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
java::lang::Runtime::traceMethodCalls (jboolean)
|
|
|
|
|
{
|
|
|
|
|
// Do nothing.
|
|
|
|
|
}
|
2002-05-14 04:10:37 +08:00
|
|
|
|
|
|
|
|
|
#if ! defined (DEFAULT_FILE_ENCODING) && defined (HAVE_ICONV) \
|
|
|
|
|
&& defined (HAVE_NL_LANGINFO)
|
|
|
|
|
|
|
|
|
|
static char *
|
|
|
|
|
file_encoding ()
|
|
|
|
|
{
|
|
|
|
|
setlocale (LC_CTYPE, "");
|
|
|
|
|
char *e = nl_langinfo (CODESET);
|
|
|
|
|
if (e == NULL || *e == '\0')
|
|
|
|
|
e = "8859_1";
|
|
|
|
|
return e;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#define DEFAULT_FILE_ENCODING file_encoding ()
|
|
|
|
|
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
#ifndef DEFAULT_FILE_ENCODING
|
|
|
|
|
#define DEFAULT_FILE_ENCODING "8859_1"
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
static char *default_file_encoding = DEFAULT_FILE_ENCODING;
|
|
|
|
|
|
|
|
|
|
#if HAVE_GETPWUID_R
|
|
|
|
|
/* Use overload resolution to find out the signature of getpwuid_r. */
|
|
|
|
|
|
|
|
|
|
/* This is Posix getpwuid_r. */
|
|
|
|
|
template <typename T_uid, typename T_passwd, typename T_buf, typename T_len>
|
|
|
|
|
static inline int
|
|
|
|
|
getpwuid_adaptor(int (*getpwuid_r)(T_uid user_id, T_passwd *pwd_r,
|
|
|
|
|
T_buf *buf_r, T_len len_r,
|
|
|
|
|
T_passwd **pwd_entry_ptr),
|
|
|
|
|
uid_t user_id, struct passwd *pwd_r,
|
|
|
|
|
char *buf_r, size_t len_r, struct passwd **pwd_entry)
|
|
|
|
|
{
|
|
|
|
|
return getpwuid_r (user_id, pwd_r, buf_r, len_r, pwd_entry);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* This is used on HPUX 10.20 */
|
|
|
|
|
template <typename T_uid, typename T_passwd, typename T_buf, typename T_len>
|
|
|
|
|
static inline int
|
|
|
|
|
getpwuid_adaptor(int (*getpwuid_r)(T_uid user_id, T_passwd *pwd_r,
|
|
|
|
|
T_buf *buf_r, T_len len_r),
|
|
|
|
|
uid_t user_id, struct passwd *pwd_r,
|
|
|
|
|
char *buf_r, size_t len_r, struct passwd **pwd_entry)
|
|
|
|
|
{
|
|
|
|
|
return getpwuid_r (user_id, pwd_r, buf_r, len_r);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/* This is used on IRIX 5.2. */
|
|
|
|
|
template <typename T_uid, typename T_passwd, typename T_buf, typename T_len>
|
|
|
|
|
static inline int
|
|
|
|
|
getpwuid_adaptor(T_passwd * (*getpwuid_r)(T_uid user_id, T_passwd *pwd_r,
|
|
|
|
|
T_buf *buf_r, T_len len_r),
|
|
|
|
|
uid_t user_id, struct passwd *pwd_r,
|
|
|
|
|
char *buf_r, size_t len_r, struct passwd **pwd_entry)
|
|
|
|
|
{
|
|
|
|
|
*pwd_entry = getpwuid_r (user_id, pwd_r, buf_r, len_r);
|
|
|
|
|
return (*pwd_entry == NULL) ? errno : 0;
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
java::lang::Runtime::insertSystemProperties (java::util::Properties *newprops)
|
|
|
|
|
{
|
|
|
|
|
// A convenience define.
|
|
|
|
|
#define SET(Prop,Val) \
|
|
|
|
|
newprops->put(JvNewStringLatin1 (Prop), JvNewStringLatin1 (Val))
|
|
|
|
|
|
|
|
|
|
// A mixture of the Java Product Versioning Specification
|
|
|
|
|
// (introduced in 1.2), and earlier versioning properties.
|
|
|
|
|
SET ("java.version", GCJVERSION);
|
|
|
|
|
SET ("java.vendor", "Free Software Foundation, Inc.");
|
|
|
|
|
SET ("java.vendor.url", "http://gcc.gnu.org/java/");
|
|
|
|
|
SET ("java.class.version", "46.0");
|
|
|
|
|
SET ("java.vm.specification.version", "1.0");
|
|
|
|
|
SET ("java.vm.specification.name", "Java(tm) Virtual Machine Specification");
|
|
|
|
|
SET ("java.vm.specification.vendor", "Sun Microsystems Inc.");
|
|
|
|
|
SET ("java.vm.version", __VERSION__);
|
|
|
|
|
SET ("java.vm.vendor", "Free Software Foundation, Inc.");
|
|
|
|
|
SET ("java.vm.name", "GNU libgcj");
|
|
|
|
|
SET ("java.specification.version", "1.3");
|
|
|
|
|
SET ("java.specification.name", "Java(tm) Platform API Specification");
|
|
|
|
|
SET ("java.specification.vendor", "Sun Microsystems Inc.");
|
|
|
|
|
|
|
|
|
|
char value[100];
|
|
|
|
|
#define NAME "GNU libgcj "
|
|
|
|
|
strcpy (value, NAME);
|
|
|
|
|
strncpy (value + sizeof (NAME) - 1, __VERSION__,
|
|
|
|
|
sizeof(value) - sizeof(NAME));
|
|
|
|
|
value[sizeof (value) - 1] = '\0';
|
|
|
|
|
jstring version = JvNewStringLatin1 (value);
|
|
|
|
|
newprops->put (JvNewStringLatin1 ("java.fullversion"), version);
|
|
|
|
|
newprops->put (JvNewStringLatin1 ("java.vm.info"), version);
|
|
|
|
|
|
|
|
|
|
// This definition is rather arbitrary: we choose $(prefix). In
|
|
|
|
|
// part we do this because most people specify only --prefix and
|
|
|
|
|
// nothing else when installing gcj. Plus, people are free to
|
|
|
|
|
// redefine `java.home' with `-D' if necessary.
|
|
|
|
|
SET ("java.home", PREFIX);
|
2002-11-15 02:28:07 +08:00
|
|
|
|
SET ("gnu.classpath.home", PREFIX);
|
|
|
|
|
|
2002-05-14 04:10:37 +08:00
|
|
|
|
SET ("file.encoding", default_file_encoding);
|
|
|
|
|
|
|
|
|
|
#ifdef HAVE_UNAME
|
|
|
|
|
struct utsname u;
|
|
|
|
|
if (! uname (&u))
|
|
|
|
|
{
|
|
|
|
|
SET ("os.name", u.sysname);
|
|
|
|
|
SET ("os.arch", u.machine);
|
|
|
|
|
SET ("os.version", u.release);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
SET ("os.name", "unknown");
|
|
|
|
|
SET ("os.arch", "unknown");
|
|
|
|
|
SET ("os.version", "unknown");
|
|
|
|
|
}
|
|
|
|
|
#endif /* HAVE_UNAME */
|
|
|
|
|
|
|
|
|
|
#ifndef NO_GETUID
|
|
|
|
|
#ifdef HAVE_PWD_H
|
|
|
|
|
uid_t user_id = getuid ();
|
|
|
|
|
struct passwd *pwd_entry;
|
|
|
|
|
|
|
|
|
|
#ifdef HAVE_GETPWUID_R
|
|
|
|
|
struct passwd pwd_r;
|
|
|
|
|
size_t len_r = 200;
|
|
|
|
|
char *buf_r = (char *) _Jv_AllocBytes (len_r);
|
|
|
|
|
|
|
|
|
|
while (buf_r != NULL)
|
|
|
|
|
{
|
|
|
|
|
int r = getpwuid_adaptor (getpwuid_r, user_id, &pwd_r,
|
|
|
|
|
buf_r, len_r, &pwd_entry);
|
|
|
|
|
if (r == 0)
|
|
|
|
|
break;
|
|
|
|
|
else if (r != ERANGE)
|
|
|
|
|
{
|
|
|
|
|
pwd_entry = NULL;
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
len_r *= 2;
|
|
|
|
|
buf_r = (char *) _Jv_AllocBytes (len_r);
|
|
|
|
|
}
|
|
|
|
|
#else
|
|
|
|
|
pwd_entry = getpwuid (user_id);
|
|
|
|
|
#endif /* HAVE_GETPWUID_R */
|
|
|
|
|
|
|
|
|
|
if (pwd_entry != NULL)
|
|
|
|
|
{
|
|
|
|
|
SET ("user.name", pwd_entry->pw_name);
|
|
|
|
|
SET ("user.home", pwd_entry->pw_dir);
|
|
|
|
|
}
|
|
|
|
|
#endif /* HAVE_PWD_H */
|
|
|
|
|
#endif /* NO_GETUID */
|
|
|
|
|
|
|
|
|
|
#ifdef HAVE_GETCWD
|
|
|
|
|
#ifdef HAVE_UNISTD_H
|
|
|
|
|
/* Use getcwd to set "user.dir". */
|
|
|
|
|
int buflen = 250;
|
|
|
|
|
char *buffer = (char *) malloc (buflen);
|
|
|
|
|
while (buffer != NULL)
|
|
|
|
|
{
|
|
|
|
|
if (getcwd (buffer, buflen) != NULL)
|
|
|
|
|
{
|
|
|
|
|
SET ("user.dir", buffer);
|
|
|
|
|
break;
|
|
|
|
|
}
|
|
|
|
|
if (errno != ERANGE)
|
|
|
|
|
break;
|
|
|
|
|
buflen = 2 * buflen;
|
|
|
|
|
buffer = (char *) realloc (buffer, buflen);
|
|
|
|
|
}
|
|
|
|
|
if (buffer != NULL)
|
|
|
|
|
free (buffer);
|
|
|
|
|
#endif /* HAVE_UNISTD_H */
|
|
|
|
|
#endif /* HAVE_GETCWD */
|
|
|
|
|
|
|
|
|
|
// Set user locale properties based on setlocale()
|
|
|
|
|
#if defined (HAVE_SETLOCALE) && defined (HAVE_LC_MESSAGES)
|
|
|
|
|
// We let the user choose the locale. However, since Java differs
|
|
|
|
|
// from POSIX, we arbitrarily pick LC_MESSAGES as determining the
|
|
|
|
|
// Java locale. We can't use LC_ALL because it might return a full
|
|
|
|
|
// list of all the settings. If we don't have LC_MESSAGES then we
|
|
|
|
|
// just default to `en_US'.
|
|
|
|
|
setlocale (LC_ALL, "");
|
|
|
|
|
char *locale = setlocale (LC_MESSAGES, "");
|
|
|
|
|
if (locale && strlen (locale) >= 2)
|
|
|
|
|
{
|
|
|
|
|
char buf[3];
|
|
|
|
|
buf[2] = '\0';
|
|
|
|
|
// copy the first two chars to user.language
|
|
|
|
|
strncpy (buf, locale, 2);
|
|
|
|
|
SET ("user.language", buf);
|
|
|
|
|
// if the next char is a '_', copy the two after that to user.region
|
|
|
|
|
locale += 2;
|
|
|
|
|
if (locale[0] == '_')
|
|
|
|
|
{
|
|
|
|
|
locale++;
|
|
|
|
|
strncpy (buf, locale, 2);
|
|
|
|
|
SET ("user.region", buf);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
#endif /* HAVE_SETLOCALE and HAVE_LC_MESSAGES */
|
|
|
|
|
{
|
|
|
|
|
SET ("user.language", "en");
|
|
|
|
|
SET ("user.region", "US");
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Set some properties according to whatever was compiled in with
|
|
|
|
|
// `-D'.
|
|
|
|
|
for (int i = 0; _Jv_Compiler_Properties[i]; ++i)
|
|
|
|
|
{
|
|
|
|
|
const char *s, *p;
|
|
|
|
|
// Find the `='.
|
|
|
|
|
for (s = p = _Jv_Compiler_Properties[i]; *s && *s != '='; ++s)
|
|
|
|
|
;
|
|
|
|
|
jstring name = JvNewStringLatin1 (p, s - p);
|
|
|
|
|
jstring val = JvNewStringLatin1 (*s == '=' ? s + 1 : s);
|
|
|
|
|
newprops->put (name, val);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Set the system properties from the user's environment.
|
|
|
|
|
#ifndef DISABLE_GETENV_PROPERTIES
|
|
|
|
|
if (_Jv_Environment_Properties)
|
|
|
|
|
{
|
|
|
|
|
size_t i = 0;
|
|
|
|
|
|
|
|
|
|
while (_Jv_Environment_Properties[i].key)
|
|
|
|
|
{
|
|
|
|
|
SET (_Jv_Environment_Properties[i].key,
|
|
|
|
|
_Jv_Environment_Properties[i].value);
|
|
|
|
|
i++;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
if (_Jv_Jar_Class_Path)
|
|
|
|
|
newprops->put(JvNewStringLatin1 ("java.class.path"),
|
|
|
|
|
JvNewStringLatin1 (_Jv_Jar_Class_Path));
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
// FIXME: find libgcj.zip and append its path?
|
|
|
|
|
char *classpath = ::getenv("CLASSPATH");
|
|
|
|
|
jstring cp = newprops->getProperty (JvNewStringLatin1("java.class.path"));
|
|
|
|
|
java::lang::StringBuffer *sb = new java::lang::StringBuffer ();
|
|
|
|
|
|
|
|
|
|
if (classpath)
|
|
|
|
|
{
|
|
|
|
|
sb->append (JvNewStringLatin1 (classpath));
|
|
|
|
|
#ifdef WIN32
|
|
|
|
|
sb->append ((jchar) ';');
|
|
|
|
|
#else
|
|
|
|
|
sb->append ((jchar) ':');
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
if (cp != NULL)
|
|
|
|
|
sb->append (cp);
|
|
|
|
|
else
|
|
|
|
|
sb->append ((jchar) '.');
|
|
|
|
|
|
|
|
|
|
newprops->put(JvNewStringLatin1 ("java.class.path"),
|
|
|
|
|
sb->toString ());
|
|
|
|
|
}
|
|
|
|
|
|
2002-10-21 07:47:43 +08:00
|
|
|
|
// The name used to invoke this process (argv[0] in C).
|
|
|
|
|
SET ("gnu.gcj.progname", _Jv_ThisExecutable());
|
|
|
|
|
|
2002-05-14 04:10:37 +08:00
|
|
|
|
// Allow platform specific settings and overrides.
|
|
|
|
|
_Jv_platform_initProperties (newprops);
|
2002-10-25 11:28:00 +08:00
|
|
|
|
|
|
|
|
|
// If java.library.path is set, tell libltdl so we search the new
|
|
|
|
|
// directories as well. FIXME: does this work properly on Windows?
|
|
|
|
|
String *path = newprops->getProperty(JvNewStringLatin1("java.library.path"));
|
|
|
|
|
if (path)
|
|
|
|
|
{
|
|
|
|
|
char *val = (char *) _Jv_Malloc (JvGetStringUTFLength (path) + 1);
|
|
|
|
|
jsize total = JvGetStringUTFRegion (path, 0, path->length(), val);
|
|
|
|
|
val[total] = '\0';
|
|
|
|
|
_Jv_SetDLLSearchPath (val);
|
|
|
|
|
_Jv_Free (val);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
// Set a value for user code to see.
|
|
|
|
|
// FIXME: JDK sets this to the actual path used, including
|
|
|
|
|
// LD_LIBRARY_PATH, etc.
|
|
|
|
|
SET ("java.library.path", "");
|
|
|
|
|
}
|
2002-05-14 04:10:37 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
java::lang::Process *
|
|
|
|
|
java::lang::Runtime::execInternal (jstringArray cmd,
|
2002-08-14 09:07:59 +08:00
|
|
|
|
jstringArray env,
|
|
|
|
|
java::io::File *dir)
|
2002-05-14 04:10:37 +08:00
|
|
|
|
{
|
2002-08-14 09:07:59 +08:00
|
|
|
|
return new java::lang::ConcreteProcess (cmd, env, dir);
|
2002-05-14 04:10:37 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
jint
|
|
|
|
|
java::lang::Runtime::availableProcessors (void)
|
|
|
|
|
{
|
|
|
|
|
// FIXME: find the real value.
|
|
|
|
|
return 1;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
jstring
|
|
|
|
|
java::lang::Runtime::nativeGetLibname (jstring pathname, jstring libname)
|
|
|
|
|
{
|
|
|
|
|
java::lang::StringBuffer *sb = new java::lang::StringBuffer ();
|
|
|
|
|
sb->append(pathname);
|
|
|
|
|
if (pathname->length() > 0)
|
|
|
|
|
{
|
|
|
|
|
// FIXME: use platform function here.
|
|
|
|
|
#ifdef WIN32
|
|
|
|
|
sb->append ((jchar) '\\');
|
|
|
|
|
#else
|
|
|
|
|
sb->append ((jchar) '/');
|
|
|
|
|
#endif
|
|
|
|
|
}
|
|
|
|
|
|
2002-12-10 09:39:32 +08:00
|
|
|
|
sb->append (JvNewStringLatin1 (_Jv_platform_solib_prefix));
|
2002-05-14 04:10:37 +08:00
|
|
|
|
sb->append(libname);
|
2002-12-10 09:39:32 +08:00
|
|
|
|
sb->append (JvNewStringLatin1 (_Jv_platform_solib_suffix));
|
2002-05-14 04:10:37 +08:00
|
|
|
|
|
|
|
|
|
return sb->toString();
|
|
|
|
|
}
|