1999-04-07 10:42:40 -04:00
|
|
|
|
// natSystem.cc - Native code implementing System class.
|
|
|
|
|
|
|
|
|
|
/* Copyright (C) 1998, 1999 Cygnus Solutions
|
|
|
|
|
|
|
|
|
|
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>
|
|
|
|
|
|
|
|
|
|
#ifdef HAVE_GETPWUID_R
|
|
|
|
|
#define _POSIX_PTHREAD_SEMANTICS
|
|
|
|
|
#ifndef _REENTRANT
|
|
|
|
|
#define _REENTRANT
|
|
|
|
|
#endif
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
#include <string.h>
|
|
|
|
|
#include <time.h>
|
|
|
|
|
#include <stdlib.h>
|
|
|
|
|
|
|
|
|
|
#ifdef HAVE_UNISTD_H
|
|
|
|
|
#include <unistd.h>
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
#ifdef HAVE_SYS_TIME_H
|
|
|
|
|
#include <sys/time.h>
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
#ifdef HAVE_PWD_H
|
|
|
|
|
#include <pwd.h>
|
|
|
|
|
#endif
|
|
|
|
|
#include <errno.h>
|
|
|
|
|
|
|
|
|
|
#include <cni.h>
|
|
|
|
|
#include <jvm.h>
|
|
|
|
|
#include <java/lang/System.h>
|
|
|
|
|
#include <java/lang/Class.h>
|
|
|
|
|
#include <java/lang/ArrayStoreException.h>
|
|
|
|
|
#include <java/lang/ArrayIndexOutOfBoundsException.h>
|
|
|
|
|
#include <java/lang/NullPointerException.h>
|
|
|
|
|
#include <java/util/Properties.h>
|
|
|
|
|
#include <java/io/PrintStream.h>
|
|
|
|
|
#include <java/io/InputStream.h>
|
|
|
|
|
|
1999-04-12 08:34:41 -04:00
|
|
|
|
#define SystemClass _CL_Q34java4lang6System
|
|
|
|
|
extern java::lang::Class SystemClass;
|
|
|
|
|
|
1999-04-07 10:42:40 -04:00
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#if defined (ECOS)
|
|
|
|
|
extern "C" unsigned long long _clock (void);
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
java::lang::System::setErr (java::io::PrintStream *newErr)
|
|
|
|
|
{
|
|
|
|
|
checkSetIO ();
|
|
|
|
|
// This violates `final' semantics. Oh well.
|
|
|
|
|
err = newErr;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
java::lang::System::setIn (java::io::InputStream *newIn)
|
|
|
|
|
{
|
|
|
|
|
checkSetIO ();
|
|
|
|
|
// This violates `final' semantics. Oh well.
|
|
|
|
|
in = newIn;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
java::lang::System::setOut (java::io::PrintStream *newOut)
|
|
|
|
|
{
|
|
|
|
|
checkSetIO ();
|
|
|
|
|
// This violates `final' semantics. Oh well.
|
|
|
|
|
out = newOut;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
java::lang::System::arraycopy (jobject src, jint src_offset,
|
|
|
|
|
jobject dst, jint dst_offset,
|
|
|
|
|
jint count)
|
|
|
|
|
{
|
|
|
|
|
if (! src || ! dst)
|
|
|
|
|
_Jv_Throw (new NullPointerException);
|
|
|
|
|
|
|
|
|
|
jclass src_c = src->getClass();
|
|
|
|
|
jclass dst_c = dst->getClass();
|
|
|
|
|
jclass src_comp = src_c->getComponentType();
|
|
|
|
|
jclass dst_comp = dst_c->getComponentType();
|
|
|
|
|
|
|
|
|
|
if (! src_c->isArray() || ! dst_c->isArray()
|
|
|
|
|
|| src_comp->isPrimitive() != dst_comp->isPrimitive()
|
|
|
|
|
|| (src_comp->isPrimitive() && src_comp != dst_comp))
|
|
|
|
|
_Jv_Throw (new ArrayStoreException);
|
|
|
|
|
|
|
|
|
|
__JArray *src_a = (__JArray *) src;
|
|
|
|
|
__JArray *dst_a = (__JArray *) dst;
|
|
|
|
|
if (src_offset < 0 || dst_offset < 0 || count < 0
|
|
|
|
|
|| src_offset + count > src_a->length
|
|
|
|
|
|| dst_offset + count > dst_a->length)
|
|
|
|
|
_Jv_Throw (new ArrayIndexOutOfBoundsException);
|
|
|
|
|
|
|
|
|
|
// Do-nothing cases.
|
|
|
|
|
if ((src == dst && src_offset == dst_offset)
|
|
|
|
|
|| ! count)
|
|
|
|
|
return;
|
|
|
|
|
|
|
|
|
|
// If both are primitive, we can optimize trivially. If DST
|
|
|
|
|
// components are always assignable from SRC components, then we
|
|
|
|
|
// will never need to raise an error, and thus can do the
|
|
|
|
|
// optimization. If source and destinations are the same, then we
|
|
|
|
|
// know that the assignability premise always holds.
|
|
|
|
|
const bool prim = src_comp->isPrimitive();
|
|
|
|
|
if (prim || dst_comp->isAssignableFrom(src_comp) || src == dst)
|
|
|
|
|
{
|
1999-04-13 08:16:41 -04:00
|
|
|
|
const size_t size = (prim ? src_comp->size()
|
|
|
|
|
: sizeof elements((jobjectArray)src)[0]);
|
|
|
|
|
|
|
|
|
|
// In an ideal world we would do this via a virtual function in
|
|
|
|
|
// __JArray. However, we can't have virtual functions in
|
|
|
|
|
// __JArray due to the need to copy an array's virtual table in
|
|
|
|
|
// _Jv_FindArrayClass.
|
|
|
|
|
// We can't just pick a single subtype of __JArray to use due to
|
|
|
|
|
// alignment concerns.
|
|
|
|
|
char *src_elts = NULL;
|
|
|
|
|
if (! prim)
|
|
|
|
|
src_elts = (char *) elements ((jobjectArray) src);
|
|
|
|
|
else if (src_comp == JvPrimClass (byte))
|
|
|
|
|
src_elts = (char *) elements ((jbyteArray) src);
|
|
|
|
|
else if (src_comp == JvPrimClass (short))
|
|
|
|
|
src_elts = (char *) elements ((jshortArray) src);
|
|
|
|
|
else if (src_comp == JvPrimClass (int))
|
|
|
|
|
src_elts = (char *) elements ((jintArray) src);
|
|
|
|
|
else if (src_comp == JvPrimClass (long))
|
|
|
|
|
src_elts = (char *) elements ((jlongArray) src);
|
|
|
|
|
else if (src_comp == JvPrimClass (boolean))
|
|
|
|
|
src_elts = (char *) elements ((jbooleanArray) src);
|
|
|
|
|
else if (src_comp == JvPrimClass (char))
|
|
|
|
|
src_elts = (char *) elements ((jcharArray) src);
|
|
|
|
|
else if (src_comp == JvPrimClass (float))
|
|
|
|
|
src_elts = (char *) elements ((jfloatArray) src);
|
|
|
|
|
else if (src_comp == JvPrimClass (double))
|
|
|
|
|
src_elts = (char *) elements ((jdoubleArray) src);
|
|
|
|
|
src_elts += size * src_offset;
|
|
|
|
|
|
|
|
|
|
char *dst_elts = NULL;
|
|
|
|
|
if (! prim)
|
|
|
|
|
dst_elts = (char *) elements ((jobjectArray) dst);
|
|
|
|
|
else if (dst_comp == JvPrimClass (byte))
|
|
|
|
|
dst_elts = (char *) elements ((jbyteArray) dst);
|
|
|
|
|
else if (dst_comp == JvPrimClass (short))
|
|
|
|
|
dst_elts = (char *) elements ((jshortArray) dst);
|
|
|
|
|
else if (dst_comp == JvPrimClass (int))
|
|
|
|
|
dst_elts = (char *) elements ((jintArray) dst);
|
|
|
|
|
else if (dst_comp == JvPrimClass (long))
|
|
|
|
|
dst_elts = (char *) elements ((jlongArray) dst);
|
|
|
|
|
else if (dst_comp == JvPrimClass (boolean))
|
|
|
|
|
dst_elts = (char *) elements ((jbooleanArray) dst);
|
|
|
|
|
else if (dst_comp == JvPrimClass (char))
|
|
|
|
|
dst_elts = (char *) elements ((jcharArray) dst);
|
|
|
|
|
else if (dst_comp == JvPrimClass (float))
|
|
|
|
|
dst_elts = (char *) elements ((jfloatArray) dst);
|
|
|
|
|
else if (dst_comp == JvPrimClass (double))
|
|
|
|
|
dst_elts = (char *) elements ((jdoubleArray) dst);
|
|
|
|
|
dst_elts += size * dst_offset;
|
|
|
|
|
|
1999-04-07 10:42:40 -04:00
|
|
|
|
// We don't bother trying memcpy. It can't be worth the cost of
|
|
|
|
|
// the check.
|
|
|
|
|
memmove ((void *) dst_elts, (void *) src_elts, count * size);
|
|
|
|
|
}
|
|
|
|
|
else
|
|
|
|
|
{
|
|
|
|
|
jobject *src_elts = elements ((jobjectArray) src_a) + src_offset;
|
|
|
|
|
jobject *dst_elts = elements ((jobjectArray) dst_a) + dst_offset;
|
|
|
|
|
|
|
|
|
|
for (int i = 0; i < count; ++i)
|
|
|
|
|
{
|
|
|
|
|
if (*src_elts
|
|
|
|
|
&& ! dst_comp->isAssignableFrom((*src_elts)->getClass()))
|
|
|
|
|
_Jv_Throw (new ArrayStoreException);
|
|
|
|
|
*dst_elts++ = *src_elts++;
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
jlong
|
|
|
|
|
java::lang::System::currentTimeMillis (void)
|
|
|
|
|
{
|
|
|
|
|
jlong r;
|
|
|
|
|
|
|
|
|
|
#if defined (HAVE_GETTIMEOFDAY)
|
|
|
|
|
struct timeval tv;
|
|
|
|
|
gettimeofday (&tv, NULL);
|
|
|
|
|
r = (jlong) tv.tv_sec * 1000 + tv.tv_usec / 1000;
|
|
|
|
|
#elif defined (HAVE_TIME)
|
|
|
|
|
r = time (NULL) * 1000;
|
|
|
|
|
#elif defined (HAVE_FTIME)
|
|
|
|
|
struct timeb t;
|
|
|
|
|
ftime (&t);
|
|
|
|
|
r = t.time * 1000 + t.millitm;
|
|
|
|
|
#elif defined (ECOS)
|
|
|
|
|
r = _clock();
|
|
|
|
|
#else
|
|
|
|
|
// In the absence of any function, time remains forever fixed.
|
|
|
|
|
r = 23;
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
return r;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
jint
|
|
|
|
|
java::lang::System::identityHashCode (jobject obj)
|
|
|
|
|
{
|
|
|
|
|
return _Jv_HashCode (obj);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void
|
|
|
|
|
java::lang::System::init_properties (void)
|
|
|
|
|
{
|
1999-04-12 08:34:41 -04:00
|
|
|
|
{
|
|
|
|
|
// We only need to synchronize around this gatekeeper.
|
|
|
|
|
JvSynchronize sync (&SystemClass);
|
|
|
|
|
if (prop_init)
|
|
|
|
|
return;
|
|
|
|
|
prop_init = true;
|
|
|
|
|
}
|
1999-04-07 10:42:40 -04:00
|
|
|
|
|
|
|
|
|
properties = new java::util::Properties ();
|
|
|
|
|
// A convenience define.
|
|
|
|
|
#define SET(Prop,Val) \
|
|
|
|
|
properties->put(JvNewStringLatin1 (Prop), JvNewStringLatin1 (Val))
|
|
|
|
|
SET ("java.version", "FIXME");
|
|
|
|
|
SET ("java.vendor", "Cygnus Solutions");
|
|
|
|
|
SET ("java.vendor.url", "http://www.cygnus.com/");
|
|
|
|
|
// SET ("java.home", "FIXME");
|
|
|
|
|
// SET ("java.class.version", "FIXME");
|
|
|
|
|
// SET ("java.class.path", "FIXME");
|
|
|
|
|
SET ("os.name", "FIXME");
|
|
|
|
|
SET ("os.arch", "FIXME");
|
|
|
|
|
SET ("os.version", "FIXME");
|
|
|
|
|
SET ("file.encoding", "8859_1"); // FIXME
|
|
|
|
|
#ifdef WIN32
|
|
|
|
|
SET ("file.separator", "\\");
|
|
|
|
|
SET ("path.separator", ";");
|
|
|
|
|
SET ("line.separator", "\r\n");
|
|
|
|
|
#else
|
|
|
|
|
// Unix.
|
|
|
|
|
SET ("file.separator", "/");
|
|
|
|
|
SET ("path.separator", ":");
|
|
|
|
|
SET ("line.separator", "\n");
|
|
|
|
|
#endif
|
|
|
|
|
|
|
|
|
|
#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_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
|
|
|
|
|
struct passwd *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 */
|
|
|
|
|
|
|
|
|
|
#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
|
|
|
|
|
}
|