corda/classpath/java-lang.cpp

1244 lines
29 KiB
C++
Raw Normal View History

2014-04-21 02:14:48 +00:00
/* Copyright (c) 2008-2014, Avian Contributors
Permission to use, copy, modify, and/or distribute this software
for any purpose with or without fee is hereby granted, provided
that the above copyright notice and this permission notice appear
in all copies.
There is NO WARRANTY for this software. See license.txt for
details. */
2014-07-11 15:50:18 +00:00
2007-10-11 21:39:21 +00:00
#include "math.h"
#include "stdlib.h"
2007-07-28 16:10:13 +00:00
#include "time.h"
2007-07-07 23:47:35 +00:00
#include "string.h"
#include "stdio.h"
#include "jni.h"
#include "jni-util.h"
#include "errno.h"
#include "fcntl.h"
#include "ctype.h"
2007-06-25 01:34:07 +00:00
#ifdef PLATFORM_WINDOWS
2014-07-11 15:50:18 +00:00
#include "windows.h"
#include "winbase.h"
#include "io.h"
#include "tchar.h"
#include "float.h"
#include "sys/types.h"
#include "sys/timeb.h"
#define SO_PREFIX ""
#define SO_SUFFIX ".dll"
#ifdef _MSC_VER
#define snprintf sprintf_s
#define isnan _isnan
#define isfinite _finite
#define strtof strtod
#endif
#else // not PLATFORM_WINDOWS
#define SO_PREFIX "lib"
#ifdef __APPLE__
#define SO_SUFFIX ".jnilib"
#include <TargetConditionals.h>
#if !TARGET_IPHONE_SIMULATOR && !TARGET_OS_IPHONE
#include <CoreServices/CoreServices.h>
#endif
#else
#define SO_SUFFIX ".so"
#endif
#include "unistd.h"
#include "limits.h"
#include "signal.h"
#include "sys/time.h"
#include "sys/types.h"
#ifndef __ANDROID__
#include "sys/sysctl.h"
#endif
#include "sys/utsname.h"
#include "sys/wait.h"
#endif // not PLATFORM_WINDOWS
#ifndef WINAPI_FAMILY
2014-07-11 15:50:18 +00:00
#ifndef WINAPI_PARTITION_DESKTOP
#define WINAPI_PARTITION_DESKTOP 1
#endif
2014-07-11 15:50:18 +00:00
#ifndef WINAPI_FAMILY_PARTITION
#define WINAPI_FAMILY_PARTITION(x) (x)
#endif
2013-02-02 09:00:33 +00:00
#else
2014-07-11 15:50:18 +00:00
#if !WINAPI_FAMILY_PARTITION(WINAPI_PARTITION_DESKTOP)
2013-02-02 09:00:33 +00:00
2014-07-11 15:50:18 +00:00
#include "avian-interop.h"
2013-02-02 09:00:33 +00:00
2014-07-11 15:50:18 +00:00
#endif
#endif // WINAPI_FAMILY
2014-07-29 09:51:33 +00:00
#ifndef M_E
// in new C++-11 standard math.h doesn't have M_E, at least on MinGW, so define it manually
#define M_E 2.7182818284590452354
#endif // M_E
namespace {
void add(JNIEnv* e, jobjectArray array, unsigned index, const char* format, ...)
{
int size = 256;
while (true) {
va_list a;
va_start(a, format);
RUNTIME_ARRAY(char, buffer, size);
int r = vsnprintf(RUNTIME_ARRAY_BODY(buffer), size - 1, format, a);
va_end(a);
if (r >= 0 and r < size - 1) {
e->SetObjectArrayElement(
array, index++, e->NewStringUTF(RUNTIME_ARRAY_BODY(buffer)));
return;
}
size *= 2;
}
}
#ifdef PLATFORM_WINDOWS
2013-02-02 08:14:47 +00:00
void add(JNIEnv* e,
jobjectArray array,
unsigned index,
const WCHAR* format,
...)
{
int size = 256;
while (true) {
va_list a;
va_start(a, format);
RUNTIME_ARRAY(WCHAR, buffer, size);
int r = _vsnwprintf(RUNTIME_ARRAY_BODY(buffer), size - 1, format, a);
va_end(a);
if (r >= 0 and r < size - 1) {
e->SetObjectArrayElement(
array,
index++,
e->NewString(reinterpret_cast<jchar*>(RUNTIME_ARRAY_BODY(buffer)),
r));
return;
}
size *= 2;
}
}
2013-02-02 08:14:47 +00:00
#if !defined(WINAPI_FAMILY) || WINAPI_FAMILY_PARTITION(WINAPI_PARTITION_DESKTOP)
2014-07-11 15:50:18 +00:00
char* getErrorStr(DWORD err)
{
LPSTR errorStr = 0;
if (!FormatMessageA(FORMAT_MESSAGE_ALLOCATE_BUFFER
| FORMAT_MESSAGE_FROM_SYSTEM
| FORMAT_MESSAGE_IGNORE_INSERTS,
0,
err,
LANG_SYSTEM_DEFAULT,
(LPSTR)&errorStr,
0,
0)) {
char* errStr = (char*)malloc(9 * sizeof(char));
snprintf(errStr, 9, "%d", (int)err);
2013-02-02 08:14:47 +00:00
return errStr;
}
2014-07-11 15:50:18 +00:00
char* errStr = strdup(errorStr);
LocalFree(errorStr);
return errStr;
}
2013-02-02 08:14:47 +00:00
#else
2014-07-11 15:50:18 +00:00
char* getErrorStr(DWORD err)
{
LPSTR errorStr = (LPSTR)malloc(4096); // NOTE: something constant
if (!FormatMessageA(
FORMAT_MESSAGE_FROM_SYSTEM | FORMAT_MESSAGE_IGNORE_INSERTS,
0,
err,
LANG_SYSTEM_DEFAULT,
errorStr,
0,
0)) {
2013-02-02 08:14:47 +00:00
free(errorStr);
2014-07-11 15:50:18 +00:00
char* errStr = (char*)malloc(9 * sizeof(char));
snprintf(errStr, 9, "%d", (int)err);
return errStr;
}
2014-07-11 15:50:18 +00:00
char* errStr = strdup(errorStr);
free(errorStr);
return errStr;
}
2013-02-02 08:14:47 +00:00
#endif
#if !defined(WINAPI_FAMILY) || WINAPI_FAMILY_PARTITION(WINAPI_PARTITION_DESKTOP)
2014-07-11 15:50:18 +00:00
void makePipe(JNIEnv* e, HANDLE p[2])
{
SECURITY_ATTRIBUTES sa;
sa.nLength = sizeof(sa);
sa.bInheritHandle = 1;
sa.lpSecurityDescriptor = 0;
BOOL success = CreatePipe(p, p + 1, &sa, 0);
if (not success) {
throwNew(e, "java/io/IOException", getErrorStr(GetLastError()));
}
2014-07-11 15:50:18 +00:00
}
#endif
2014-07-11 15:50:18 +00:00
int descriptor(JNIEnv* e, HANDLE h)
{
int fd = _open_osfhandle(reinterpret_cast<intptr_t>(h), 0);
if (fd == -1) {
throwNewErrno(e, "java/io/IOException");
}
2014-07-11 15:50:18 +00:00
return fd;
}
#else
2014-07-11 15:50:18 +00:00
void makePipe(JNIEnv* e, int p[2])
{
if (pipe(p) != 0) {
throwNewErrno(e, "java/io/IOException");
}
2014-07-11 15:50:18 +00:00
}
void safeClose(int& fd)
{
if (fd != -1)
close(fd);
fd = -1;
}
void close(int p[2])
{
::close(p[0]);
::close(p[1]);
}
void clean(JNIEnv* e, jobjectArray command, char** p)
{
int i = 0;
for (char** x = p; *x; ++x, ++i) {
jstring element = (jstring)e->GetObjectArrayElement(command, i);
e->ReleaseStringUTFChars(element, *x);
}
2014-07-11 15:50:18 +00:00
free(p);
}
#endif
}
2014-07-11 15:50:18 +00:00
class Locale { // represents an ISO two-char language/country pair
static const unsigned FIELDLEN = 2;
static const unsigned FIELDSIZE = FIELDLEN + 1;
static const char* DEFAULT_LANGUAGE;
static const char* DEFAULT_REGION;
char language[FIELDSIZE];
char region[FIELDSIZE];
2014-07-11 15:50:18 +00:00
bool isLanguage(const char* language)
{
if (!language)
return false;
unsigned len = strlen(language);
2014-07-11 15:50:18 +00:00
if (len != FIELDLEN)
return false;
const char* p = language - 1;
2014-07-11 15:50:18 +00:00
while (islower(*++p))
;
if (*p != '\0')
return false;
return true;
}
2014-07-11 15:50:18 +00:00
bool isRegion(const char* region)
{
if (!region)
return false;
unsigned len = strlen(region);
2014-07-11 15:50:18 +00:00
if (len != FIELDLEN)
return false;
const char* p = region - 1;
2014-07-11 15:50:18 +00:00
while (isupper(*++p))
;
if (*p != '\0')
return false;
return true;
}
2014-07-11 15:50:18 +00:00
public:
Locale(const char* language = "")
{
Locale l(language, "");
*this = l;
}
2014-07-11 15:50:18 +00:00
Locale(const char* language, const char* region)
{
language = isLanguage(language) ? language : DEFAULT_LANGUAGE;
region = isRegion(region) ? region : DEFAULT_REGION;
memcpy(this->language, language, FIELDSIZE);
memcpy(this->region, region, FIELDSIZE);
}
2014-07-11 15:50:18 +00:00
Locale& operator=(const Locale& l)
{
memcpy(language, l.language, FIELDSIZE);
memcpy(region, l.region, FIELDSIZE);
return *this;
}
2014-07-11 15:50:18 +00:00
const char* getLanguage()
{
return reinterpret_cast<const char*>(language);
}
const char* getRegion()
{
return reinterpret_cast<const char*>(region);
}
};
const char* Locale::DEFAULT_LANGUAGE = "en";
const char* Locale::DEFAULT_REGION = "";
#ifdef PLATFORM_WINDOWS
2014-07-11 15:50:18 +00:00
void appendN(char** dest, char ch, size_t length)
{
for (size_t i = 0; i < length; i++) {
*((*dest)++) = ch;
}
}
2014-07-11 15:50:18 +00:00
bool needsEscape(const char* src, size_t length)
{
const char* end = src + length;
2014-07-11 15:50:18 +00:00
for (const char* ptr = src; ptr < end; ptr++) {
switch (*ptr) {
case ' ':
case '\t':
case '\n':
case '\v':
case '"':
return true;
}
}
return false;
}
2014-07-11 15:50:18 +00:00
void copyAndEscape(char** dest, const char* src, size_t length)
{
char* destp = *dest;
const char* end = src + length;
2014-07-11 15:50:18 +00:00
if (length != 0 && !needsEscape(src, length)) {
for (const char* ptr = src; ptr < end; ptr++) {
*(destp++) = *ptr;
}
} else {
*(destp++) = '"';
2014-07-11 15:50:18 +00:00
for (const char* ptr = src;; ptr++) {
unsigned numBackslashes = 0;
2014-07-11 15:50:18 +00:00
while (ptr < end && *ptr == '\\') {
ptr++;
numBackslashes++;
}
2014-07-11 15:50:18 +00:00
if (ptr == end) {
appendN(&destp, '\\', 2 * numBackslashes);
break;
2014-07-11 15:50:18 +00:00
} else if (*ptr == '"') {
appendN(&destp, '\\', 2 * numBackslashes + 1);
*(destp++) = *ptr;
} else {
appendN(&destp, '\\', numBackslashes);
*(destp++) = *ptr;
}
}
2014-07-11 15:50:18 +00:00
*(destp++) = '"';
}
*dest = destp;
}
2014-07-11 15:50:18 +00:00
extern "C" JNIEXPORT void JNICALL
Java_java_lang_Runtime_exec(JNIEnv* e,
jclass,
jobjectArray command,
jlongArray process)
{
#if !defined(WINAPI_FAMILY) || WINAPI_FAMILY_PARTITION(WINAPI_PARTITION_DESKTOP)
int size = 0;
2014-07-11 15:50:18 +00:00
for (int i = 0; i < e->GetArrayLength(command); ++i) {
jstring element = (jstring)e->GetObjectArrayElement(command, i);
if (element) {
// worst case, assuming every character is '"', and we escape all of them
size += 2 * e->GetStringUTFLength(element) + 3;
} else {
2014-07-11 15:50:18 +00:00
throwNew(e,
"java/lang/NullPointerException",
strdup("null string array element"));
}
}
2014-07-11 15:50:18 +00:00
RUNTIME_ARRAY(char, line, size);
char* linep = RUNTIME_ARRAY_BODY(line);
for (int i = 0; i < e->GetArrayLength(command); ++i) {
2014-07-11 15:50:18 +00:00
if (i)
*(linep++) = _T(' ');
jstring element = (jstring)e->GetObjectArrayElement(command, i);
const char* s = e->GetStringUTFChars(element, 0);
copyAndEscape(&linep, s, e->GetStringUTFLength(element));
e->ReleaseStringUTFChars(element, s);
}
*(linep++) = _T('\0');
2014-07-11 15:50:18 +00:00
HANDLE in[] = {0, 0};
HANDLE out[] = {0, 0};
HANDLE err[] = {0, 0};
makePipe(e, in);
SetHandleInformation(in[0], HANDLE_FLAG_INHERIT, 0);
jlong inDescriptor = static_cast<jlong>(descriptor(e, in[0]));
2014-07-11 15:50:18 +00:00
if (e->ExceptionCheck())
return;
e->SetLongArrayRegion(process, 2, 1, &inDescriptor);
makePipe(e, out);
SetHandleInformation(out[1], HANDLE_FLAG_INHERIT, 0);
jlong outDescriptor = static_cast<jlong>(descriptor(e, out[1]));
2014-07-11 15:50:18 +00:00
if (e->ExceptionCheck())
return;
e->SetLongArrayRegion(process, 3, 1, &outDescriptor);
makePipe(e, err);
SetHandleInformation(err[0], HANDLE_FLAG_INHERIT, 0);
jlong errDescriptor = static_cast<jlong>(descriptor(e, err[0]));
2014-07-11 15:50:18 +00:00
if (e->ExceptionCheck())
return;
e->SetLongArrayRegion(process, 4, 1, &errDescriptor);
2014-07-11 15:50:18 +00:00
PROCESS_INFORMATION pi;
ZeroMemory(&pi, sizeof(pi));
2014-07-11 15:50:18 +00:00
STARTUPINFO si;
ZeroMemory(&si, sizeof(si));
si.cb = sizeof(si);
si.dwFlags = STARTF_USESTDHANDLES;
si.hStdOutput = in[1];
si.hStdInput = out[0];
si.hStdError = err[1];
2014-07-11 15:50:18 +00:00
BOOL success = CreateProcess(0,
(LPSTR)RUNTIME_ARRAY_BODY(line),
0,
0,
1,
CREATE_NO_WINDOW | CREATE_UNICODE_ENVIRONMENT,
2014-07-11 15:50:18 +00:00
0,
0,
&si,
&pi);
CloseHandle(in[1]);
CloseHandle(out[0]);
CloseHandle(err[1]);
2014-07-11 15:50:18 +00:00
if (not success) {
throwNew(e, "java/io/IOException", getErrorStr(GetLastError()));
return;
}
2014-07-11 15:50:18 +00:00
jlong pid = reinterpret_cast<jlong>(pi.hProcess);
e->SetLongArrayRegion(process, 0, 1, &pid);
2014-07-11 15:50:18 +00:00
jlong tid = reinterpret_cast<jlong>(pi.hThread);
e->SetLongArrayRegion(process, 1, 1, &tid);
#else
throwNew(e, "java/io/Exception", strdup("Not supported on WinRT/WinPhone8"));
#endif
}
2014-07-11 15:50:18 +00:00
extern "C" JNIEXPORT jint JNICALL
Java_java_lang_Runtime_waitFor(JNIEnv* e, jclass, jlong pid, jlong tid)
{
#if !defined(WINAPI_FAMILY) || WINAPI_FAMILY_PARTITION(WINAPI_PARTITION_DESKTOP)
DWORD exitCode;
WaitForSingleObject(reinterpret_cast<HANDLE>(pid), INFINITE);
BOOL success = GetExitCodeProcess(reinterpret_cast<HANDLE>(pid), &exitCode);
2014-07-11 15:50:18 +00:00
if (not success) {
throwNew(e, "java/lang/Exception", getErrorStr(GetLastError()));
}
CloseHandle(reinterpret_cast<HANDLE>(pid));
CloseHandle(reinterpret_cast<HANDLE>(tid));
return exitCode;
#else
throwNew(e, "java/io/Exception", strdup("Not supported on WinRT/WinPhone8"));
return -1;
#endif
}
2011-11-03 18:30:51 +00:00
extern "C" JNIEXPORT void JNICALL
2014-07-11 15:50:18 +00:00
Java_java_lang_Runtime_kill(JNIEnv* e UNUSED, jclass, jlong pid)
{
#if !defined(WINAPI_FAMILY) || WINAPI_FAMILY_PARTITION(WINAPI_PARTITION_DESKTOP)
2011-11-03 18:30:51 +00:00
TerminateProcess(reinterpret_cast<HANDLE>(pid), 1);
#else
throwNew(e, "java/io/Exception", strdup("Not supported on WinRT/WinPhone8"));
#endif
2011-11-03 18:30:51 +00:00
}
2014-07-11 15:50:18 +00:00
Locale getLocale()
{
#if !defined(WINAPI_FAMILY) || WINAPI_FAMILY_PARTITION(WINAPI_PARTITION_DESKTOP)
const char* lang = "";
const char* reg = "";
unsigned langid = GetUserDefaultUILanguage();
unsigned prilang = langid & 0x3ff;
unsigned sublang = langid >> 10;
switch (prilang) {
2014-07-11 15:50:18 +00:00
case 0x004: {
lang = "zh";
switch (sublang) {
case 0x01:
reg = "CN";
break;
case 0x02:
reg = "TW";
break;
case 0x03:
reg = "HK";
break;
case 0x04:
reg = "SG";
break;
}
} break;
case 0x006:
lang = "da";
reg = "DK";
break;
case 0x007:
lang = "de";
reg = "DE";
break;
case 0x009: {
lang = "en";
switch (sublang) {
case 0x01:
reg = "US";
break;
case 0x02:
reg = "GB";
break;
case 0x03:
reg = "AU";
break;
case 0x04:
reg = "CA";
break;
case 0x05:
reg = "NZ";
break;
case 0x06:
reg = "IE";
break;
case 0x07:
reg = "ZA";
break;
case 0x10:
reg = "IN";
break;
}
} break;
case 0x00a: {
lang = "es";
switch (sublang) {
case 0x01:
case 0x03:
reg = "ES";
break;
case 0x02:
reg = "MX";
break;
}
} break;
case 0x00c: {
lang = "fr";
switch (sublang) {
case 0x01:
reg = "FR";
break;
case 0x02:
reg = "BE";
break;
case 0x03:
reg = "CA";
break;
}
} break;
case 0x010:
lang = "it";
reg = "IT";
break;
case 0x011:
lang = "ja";
reg = "JP";
break;
case 0x012:
lang = "ko";
reg = "KR";
break;
case 0x013: {
lang = "nl";
switch (sublang) {
case 0x01:
reg = "NL";
break;
case 0x02:
reg = "BE";
break;
}
} break;
case 0x014:
lang = "no";
reg = "NO";
break;
case 0x015:
lang = "pl";
reg = "PL";
break;
case 0x016: {
lang = "pt";
switch (sublang) {
case 0x01:
reg = "BR";
break;
case 0x02:
reg = "PT";
break;
}
} break;
case 0x018:
lang = "ro";
reg = "RO";
break;
case 0x019:
lang = "ru";
reg = "RU";
break;
case 0x01d:
lang = "sv";
reg = "SE";
break;
default:
lang = "en";
}
return Locale(lang, reg);
#else
2013-02-02 09:00:33 +00:00
std::wstring culture = AvianInterop::GetCurrentUICulture();
2014-07-11 15:50:18 +00:00
char* cultureName
= strdup(std::string(culture.begin(), culture.end()).c_str());
2013-02-02 09:00:33 +00:00
char* delimiter = strchr(cultureName, '-');
2014-07-11 15:50:18 +00:00
if (!delimiter) {
2013-02-02 09:00:33 +00:00
free(cultureName);
return Locale("en", "US");
}
const char* lang = cultureName;
const char* reg = delimiter + 1;
*delimiter = 0;
Locale locale(lang, reg);
free(cultureName);
return locale;
#endif
}
#else
2014-07-11 15:50:18 +00:00
extern "C" JNIEXPORT void JNICALL
Java_java_lang_Runtime_exec(JNIEnv* e,
jclass,
jobjectArray command,
jlongArray process)
{
2014-07-11 15:50:18 +00:00
char** argv = static_cast<char**>(
malloc((e->GetArrayLength(command) + 1) * sizeof(char*)));
int i;
2014-07-11 15:50:18 +00:00
for (i = 0; i < e->GetArrayLength(command); i++) {
jstring element = (jstring)e->GetObjectArrayElement(command, i);
char* s = const_cast<char*>(e->GetStringUTFChars(element, 0));
argv[i] = s;
}
argv[i] = 0;
2014-07-11 15:50:18 +00:00
int in[] = {-1, -1};
int out[] = {-1, -1};
int err[] = {-1, -1};
int msg[] = {-1, -1};
makePipe(e, in);
2014-07-11 15:50:18 +00:00
if (e->ExceptionCheck())
return;
jlong inDescriptor = static_cast<jlong>(in[0]);
e->SetLongArrayRegion(process, 2, 1, &inDescriptor);
makePipe(e, out);
2014-07-11 15:50:18 +00:00
if (e->ExceptionCheck())
return;
jlong outDescriptor = static_cast<jlong>(out[1]);
e->SetLongArrayRegion(process, 3, 1, &outDescriptor);
makePipe(e, err);
2014-07-11 15:50:18 +00:00
if (e->ExceptionCheck())
return;
jlong errDescriptor = static_cast<jlong>(err[0]);
e->SetLongArrayRegion(process, 4, 1, &errDescriptor);
makePipe(e, msg);
2014-07-11 15:50:18 +00:00
if (e->ExceptionCheck())
return;
if (fcntl(msg[1], F_SETFD, FD_CLOEXEC) != 0) {
throwNewErrno(e, "java/io/IOException");
return;
}
2014-07-11 15:50:18 +00:00
#ifdef __QNX__
// fork(2) doesn't work in multithreaded QNX programs. See
// http://www.qnx.com/developers/docs/6.4.1/neutrino/getting_started/s1_procs.html
pid_t pid = vfork();
#else
// We might be able to just use vfork on all UNIX-style systems, but
// the manual makes it sound dangerous due to the shared
// parent/child address space, so we use fork if we can.
pid_t pid = fork();
#endif
2014-07-11 15:50:18 +00:00
switch (pid) {
case -1: // error
throwNewErrno(e, "java/io/IOException");
return;
2014-07-11 15:50:18 +00:00
case 0: { // child
// Setup stdin, stdout and stderr
dup2(in[1], 1);
close(in);
dup2(out[0], 0);
close(out);
dup2(err[1], 2);
close(err);
close(msg[0]);
2014-07-11 15:50:18 +00:00
execvp(argv[0], argv);
2014-07-11 15:50:18 +00:00
// Error if here
int val = errno;
ssize_t rv UNUSED = write(msg[1], &val, sizeof(val));
exit(127);
} break;
2014-07-11 15:50:18 +00:00
default: { // parent
jlong JNIPid = static_cast<jlong>(pid);
e->SetLongArrayRegion(process, 0, 1, &JNIPid);
2014-07-11 15:50:18 +00:00
safeClose(in[1]);
safeClose(out[0]);
safeClose(err[1]);
safeClose(msg[1]);
2014-07-11 15:50:18 +00:00
int val;
int r = read(msg[0], &val, sizeof(val));
2014-07-11 15:50:18 +00:00
if (r == -1) {
throwNewErrno(e, "java/io/IOException");
return;
2014-07-11 15:50:18 +00:00
} else if (r) {
errno = val;
throwNewErrno(e, "java/io/IOException");
return;
}
} break;
}
2014-07-11 15:50:18 +00:00
safeClose(msg[0]);
clean(e, command, argv);
2014-07-11 15:50:18 +00:00
fcntl(in[0], F_SETFD, FD_CLOEXEC);
fcntl(out[1], F_SETFD, FD_CLOEXEC);
fcntl(err[0], F_SETFD, FD_CLOEXEC);
}
2014-07-11 15:50:18 +00:00
extern "C" JNIEXPORT jint JNICALL
Java_java_lang_Runtime_waitFor(JNIEnv*, jclass, jlong pid, jlong)
{
bool finished = false;
int status;
int exitCode;
2014-07-11 15:50:18 +00:00
while (!finished) {
waitpid(pid, &status, 0);
2014-07-11 15:50:18 +00:00
if (WIFEXITED(status)) {
finished = true;
exitCode = WEXITSTATUS(status);
2014-07-11 15:50:18 +00:00
} else if (WIFSIGNALED(status)) {
finished = true;
exitCode = -1;
}
}
2014-07-11 15:50:18 +00:00
return exitCode;
}
2011-11-03 18:30:51 +00:00
extern "C" JNIEXPORT void JNICALL
2014-07-11 15:50:18 +00:00
Java_java_lang_Runtime_kill(JNIEnv*, jclass, jlong pid)
{
2011-11-03 18:30:51 +00:00
kill((pid_t)pid, SIGTERM);
}
2014-07-11 15:50:18 +00:00
Locale getLocale()
{
Locale fallback;
const char* LANG = getenv("LANG");
2014-07-11 15:50:18 +00:00
if (!LANG || strcmp(LANG, "C") == 0)
return fallback;
int len = strlen(LANG);
2014-07-11 15:50:18 +00:00
char buf[len + 1]; // + 1 for the '\0' char
memcpy(buf, LANG, len + 1);
char* tracer = buf;
const char* reg;
2014-07-11 15:50:18 +00:00
while (*tracer && *tracer != '_')
++tracer;
if (!*tracer)
return fallback;
*tracer = '\0';
reg = ++tracer;
2014-07-11 15:50:18 +00:00
while (*tracer && *tracer != '.')
++tracer;
if (tracer == reg)
return fallback;
*tracer = '\0';
Locale locale(buf, reg);
return locale;
}
#endif
extern "C" JNIEXPORT jobjectArray JNICALL
Java_java_lang_System_getNativeProperties(JNIEnv* e, jclass)
2007-07-07 23:47:35 +00:00
{
jobjectArray array
= e->NewObjectArray(32, e->FindClass("java/lang/String"), 0);
unsigned index = 0;
#ifdef ARCH_x86_32
e->SetObjectArrayElement(array, index++, e->NewStringUTF("os.arch=x86"));
#elif defined ARCH_x86_64
e->SetObjectArrayElement(array, index++, e->NewStringUTF("os.arch=x86_64"));
#elif defined ARCH_arm
e->SetObjectArrayElement(array, index++, e->NewStringUTF("os.arch=arm"));
#endif
#ifdef PLATFORM_WINDOWS
e->SetObjectArrayElement(
array, index++, e->NewStringUTF("line.separator=\r\n"));
e->SetObjectArrayElement(
array, index++, e->NewStringUTF("file.separator=\\"));
e->SetObjectArrayElement(array, index++, e->NewStringUTF("path.separator=;"));
2014-07-11 15:50:18 +00:00
#if !defined(WINAPI_FAMILY) || WINAPI_FAMILY_PARTITION(WINAPI_PARTITION_DESKTOP)
e->SetObjectArrayElement(array, index++, e->NewStringUTF("os.name=Windows"));
2014-07-11 15:50:18 +00:00
#elif WINAPI_FAMILY_PARTITION(WINAPI_PARTITION_PHONE)
e->SetObjectArrayElement(
array, index++, e->NewStringUTF("os.name=Windows Phone"));
2014-07-11 15:50:18 +00:00
#else
e->SetObjectArrayElement(
array, index++, e->NewStringUTF("os.name=Windows RT"));
2014-07-11 15:50:18 +00:00
#endif
2014-07-11 15:50:18 +00:00
#if !defined(WINAPI_FAMILY) || WINAPI_FAMILY_PARTITION(WINAPI_PARTITION_DESKTOP)
{
OSVERSIONINFO OSversion;
OSversion.dwOSVersionInfoSize = sizeof(OSVERSIONINFO);
::GetVersionEx(&OSversion);
add(e,
array,
index++,
"os.version=%i.%i",
static_cast<int>(OSversion.dwMajorVersion),
static_cast<int>(OSversion.dwMinorVersion));
}
2014-07-11 15:50:18 +00:00
#else
// Currently there is no alternative on WinRT/WP8
e->SetObjectArrayElement(array, index++, e->NewStringUTF("os.version=8.0"));
2014-07-11 15:50:18 +00:00
#endif
2014-07-11 15:50:18 +00:00
#if !defined(WINAPI_FAMILY) || WINAPI_FAMILY_PARTITION(WINAPI_PARTITION_DESKTOP)
{
WCHAR buffer[MAX_PATH];
GetTempPathW(MAX_PATH, buffer);
add(e, array, index++, L"java.io.tmpdir=%ls", buffer);
}
2014-07-11 15:50:18 +00:00
#else
add(e,
array,
index++,
L"java.io.tmpdir=%ls",
AvianInterop::GetTemporaryFolder().c_str());
2014-07-11 15:50:18 +00:00
#endif
2014-07-11 15:50:18 +00:00
#if !defined(WINAPI_FAMILY) || WINAPI_FAMILY_PARTITION(WINAPI_PARTITION_DESKTOP)
{
WCHAR buffer[MAX_PATH];
GetCurrentDirectoryW(MAX_PATH, buffer);
add(e, array, index++, L"user.dir=%ls", buffer);
}
2014-07-11 15:50:18 +00:00
#else
add(e,
array,
index++,
L"user.dir=%ls",
AvianInterop::GetInstalledLocation().c_str());
2014-07-11 15:50:18 +00:00
#endif
2014-07-11 15:50:18 +00:00
#if !defined(WINAPI_FAMILY) || WINAPI_FAMILY_PARTITION(WINAPI_PARTITION_DESKTOP)
#ifdef _MSC_VER
{
WCHAR buffer[MAX_PATH];
size_t needed;
if (_wgetenv_s(&needed, buffer, MAX_PATH, L"USERPROFILE") == 0) {
add(e, array, index++, L"user.home=%ls", buffer);
}
}
#else
add(e, array, index++, L"user.home=%ls", _wgetenv(L"USERPROFILE"));
#endif
2014-07-11 15:50:18 +00:00
#else
add(e,
array,
index++,
L"user.home=%ls",
AvianInterop::GetDocumentsLibraryLocation().c_str());
2014-07-11 15:50:18 +00:00
#endif
#else // not Windows
e->SetObjectArrayElement(
array, index++, e->NewStringUTF("line.separator=\n"));
e->SetObjectArrayElement(array, index++, e->NewStringUTF("file.separator=/"));
e->SetObjectArrayElement(array, index++, e->NewStringUTF("path.separator=:"));
#ifdef __APPLE__
e->SetObjectArrayElement(array, index++, e->NewStringUTF("os.name=Mac OS X"));
2012-08-02 16:36:16 +00:00
#elif defined __FreeBSD__
e->SetObjectArrayElement(array, index++, e->NewStringUTF("os.name=FreeBSD"));
#else
e->SetObjectArrayElement(array, index++, e->NewStringUTF("os.name=Linux"));
#endif
{
struct utsname system_id;
uname(&system_id);
add(e, array, index++, "os.version=%s", system_id.release);
}
e->SetObjectArrayElement(
array, index++, e->NewStringUTF("java.io.tmpdir=/tmp"));
2007-07-07 23:47:35 +00:00
{
char buffer[PATH_MAX];
add(e, array, index++, "user.dir=%s", getcwd(buffer, PATH_MAX));
2007-10-24 15:44:51 +00:00
}
2014-05-01 17:26:27 +00:00
add(e, array, index++, "user.home=%s", getenv("HOME"));
#endif // not Windows
{
Locale locale = getLocale();
add(e, array, index++, "user.language=%s", locale.getLanguage());
add(e, array, index++, "user.region=%s", locale.getRegion());
2007-10-24 15:44:51 +00:00
}
return array;
2007-07-07 23:47:35 +00:00
}
2007-07-28 16:10:13 +00:00
// System.getEnvironment() implementation
2012-04-03 16:38:48 +00:00
// TODO: For Win32, replace usage of deprecated _environ and add Unicode
// support (neither of which is likely to be of great importance).
2012-04-05 16:04:53 +00:00
#ifdef AVIAN_IOS
namespace {
2014-07-11 15:50:18 +00:00
const char* environ[] = {0};
2012-04-05 16:04:53 +00:00
}
2012-04-06 22:52:24 +00:00
#elif defined __APPLE__
2014-07-11 15:50:18 +00:00
#include <crt_externs.h>
#define environ (*_NSGetEnviron())
#elif defined(WINAPI_FAMILY) \
&& !WINAPI_FAMILY_PARTITION(WINAPI_PARTITION_DESKTOP)
// WinRT/WP8 does not provide alternative for environment variables
2014-07-11 15:50:18 +00:00
char* environ[] = {0};
2012-04-05 16:04:53 +00:00
#else
extern char** environ;
2012-04-05 16:04:53 +00:00
#endif
extern "C" JNIEXPORT jobjectArray JNICALL
2014-07-11 15:50:18 +00:00
Java_java_lang_System_getEnvironment(JNIEnv* env, jclass)
{
int length;
2014-07-11 15:50:18 +00:00
for (length = 0; environ[length] != 0; ++length)
;
2014-07-11 15:50:18 +00:00
jobjectArray stringArray = env->NewObjectArray(
length, env->FindClass("java/lang/String"), env->NewStringUTF(""));
for (int i = 0; i < length; i++) {
2012-04-03 16:38:48 +00:00
jobject varString = env->NewStringUTF(environ[i]);
env->SetObjectArrayElement(stringArray, i, varString);
}
return stringArray;
}
2007-07-28 16:10:13 +00:00
extern "C" JNIEXPORT jlong JNICALL
2014-07-11 15:50:18 +00:00
Java_java_lang_System_currentTimeMillis(JNIEnv*, jclass)
2007-07-28 16:10:13 +00:00
{
#ifdef PLATFORM_WINDOWS
// We used to use _ftime here, but that only gives us 1-second
// resolution on Windows 7. _ftime_s might work better, but MinGW
// doesn't have it as of this writing. So we use this mess instead:
FILETIME time;
GetSystemTimeAsFileTime(&time);
2014-07-11 15:50:18 +00:00
return (((static_cast<jlong>(time.dwHighDateTime) << 32) | time.dwLowDateTime)
/ 10000) - 11644473600000LL;
2007-10-23 17:22:48 +00:00
#else
2014-07-11 15:50:18 +00:00
timeval tv = {0, 0};
2007-07-28 16:10:13 +00:00
gettimeofday(&tv, 0);
2014-07-11 15:50:18 +00:00
return (static_cast<jlong>(tv.tv_sec) * 1000)
+ (static_cast<jlong>(tv.tv_usec) / 1000);
2007-10-23 17:22:48 +00:00
#endif
2007-07-28 16:10:13 +00:00
}
extern "C" JNIEXPORT jstring JNICALL
2014-07-11 15:50:18 +00:00
Java_java_lang_System_doMapLibraryName(JNIEnv* e, jclass, jstring name)
{
jstring r = 0;
const char* chars = e->GetStringUTFChars(name, 0);
if (chars) {
unsigned nameLength = strlen(chars);
unsigned size = sizeof(SO_PREFIX) + nameLength + sizeof(SO_SUFFIX);
RUNTIME_ARRAY(char, buffer, size);
2014-07-11 15:50:18 +00:00
snprintf(RUNTIME_ARRAY_BODY(buffer), size, SO_PREFIX "%s" SO_SUFFIX, chars);
r = e->NewStringUTF(RUNTIME_ARRAY_BODY(buffer));
e->ReleaseStringUTFChars(name, chars);
}
return r;
}
extern "C" JNIEXPORT jboolean JNICALL
2014-07-11 15:50:18 +00:00
Java_java_lang_Double_isInfinite(JNIEnv*, jclass, jdouble val)
{
return !isfinite(val);
}
extern "C" JNIEXPORT jboolean JNICALL
2014-07-11 15:50:18 +00:00
Java_java_lang_Double_isNaN(JNIEnv*, jclass, jdouble val)
{
return isnan(val);
}
extern "C" JNIEXPORT jdouble JNICALL
2014-07-11 15:50:18 +00:00
Java_java_lang_Double_doubleFromString(JNIEnv* e,
jclass,
jstring s,
jintArray numDoublesRead)
{
const char* chars = e->GetStringUTFChars(s, 0);
double d = 0.0;
2008-02-28 17:03:24 +00:00
jint numRead = 0;
if (chars) {
char* lastRead;
d = strtod(chars, &lastRead);
if ((lastRead != chars) && ((chars + strlen(chars)) == lastRead)) {
numRead = 1;
}
e->ReleaseStringUTFChars(s, chars);
}
e->SetIntArrayRegion(numDoublesRead, 0, 1, &numRead);
return d;
}
extern "C" JNIEXPORT jboolean JNICALL
2014-07-11 15:50:18 +00:00
Java_java_lang_Float_isInfinite(JNIEnv*, jclass, jfloat val)
{
return !isfinite(val);
}
extern "C" JNIEXPORT jboolean JNICALL
2014-07-11 15:50:18 +00:00
Java_java_lang_Float_isNaN(JNIEnv*, jclass, jfloat val)
{
return isnan(val);
}
extern "C" JNIEXPORT jfloat JNICALL
2014-07-11 15:50:18 +00:00
Java_java_lang_Float_floatFromString(JNIEnv* e,
jclass,
jstring s,
jintArray numFloatsRead)
{
const char* chars = e->GetStringUTFChars(s, 0);
float f = 0.0;
2008-02-28 17:03:24 +00:00
jint numRead = 0;
if (chars) {
char* lastRead;
f = strtof(chars, &lastRead);
if ((lastRead != chars) && ((chars + strlen(chars)) == lastRead)) {
numRead = 1;
}
e->ReleaseStringUTFChars(s, chars);
}
e->SetIntArrayRegion(numFloatsRead, 0, 1, &numRead);
return f;
}
2007-10-12 18:53:56 +00:00
extern "C" JNIEXPORT jdouble JNICALL
2014-07-11 15:50:18 +00:00
Java_java_lang_Math_sin(JNIEnv*, jclass, jdouble val)
2007-10-12 18:53:56 +00:00
{
return sin(val);
}
2007-12-21 21:10:40 +00:00
extern "C" JNIEXPORT jdouble JNICALL
2014-07-11 15:50:18 +00:00
Java_java_lang_Math_cos(JNIEnv*, jclass, jdouble val)
2007-12-21 21:10:40 +00:00
{
return cos(val);
}
extern "C" JNIEXPORT jdouble JNICALL
2014-07-11 15:50:18 +00:00
Java_java_lang_Math_tan(JNIEnv*, jclass, jdouble val)
{
return tan(val);
}
2013-01-21 21:43:29 +00:00
extern "C" JNIEXPORT jdouble JNICALL
2014-07-11 15:50:18 +00:00
Java_java_lang_Math_asin(JNIEnv*, jclass, jdouble val)
2013-01-21 21:43:29 +00:00
{
return asin(val);
}
extern "C" JNIEXPORT jdouble JNICALL
2014-07-11 15:50:18 +00:00
Java_java_lang_Math_acos(JNIEnv*, jclass, jdouble val)
2013-01-21 21:43:29 +00:00
{
return acos(val);
}
extern "C" JNIEXPORT jdouble JNICALL
2014-07-11 15:50:18 +00:00
Java_java_lang_Math_atan(JNIEnv*, jclass, jdouble val)
2013-01-21 21:43:29 +00:00
{
return atan(val);
}
extern "C" JNIEXPORT jdouble JNICALL
2014-07-11 15:50:18 +00:00
Java_java_lang_Math_atan2(JNIEnv*, jclass, jdouble y, jdouble x)
{
return atan2(y, x);
}
2013-01-21 21:43:29 +00:00
extern "C" JNIEXPORT jdouble JNICALL
2014-07-11 15:50:18 +00:00
Java_java_lang_Math_sinh(JNIEnv*, jclass, jdouble val)
2013-01-21 21:43:29 +00:00
{
return sinh(val);
}
extern "C" JNIEXPORT jdouble JNICALL
2014-07-11 15:50:18 +00:00
Java_java_lang_Math_cosh(JNIEnv*, jclass, jdouble val)
2013-01-21 21:43:29 +00:00
{
return cosh(val);
}
extern "C" JNIEXPORT jdouble JNICALL
2014-07-11 15:50:18 +00:00
Java_java_lang_Math_tanh(JNIEnv*, jclass, jdouble val)
2013-01-21 21:43:29 +00:00
{
return tanh(val);
}
extern "C" JNIEXPORT jdouble JNICALL
2014-07-11 15:50:18 +00:00
Java_java_lang_Math_sqrt(JNIEnv*, jclass, jdouble val)
{
return sqrt(val);
}
extern "C" JNIEXPORT jdouble JNICALL
2014-07-11 15:50:18 +00:00
Java_java_lang_Math_pow(JNIEnv*, jclass, jdouble val, jdouble exp)
{
return pow(val, exp);
}
extern "C" JNIEXPORT jdouble JNICALL
2014-07-11 15:50:18 +00:00
Java_java_lang_Math_log(JNIEnv*, jclass, jdouble val)
{
return log(val);
}
extern "C" JNIEXPORT jdouble JNICALL
2014-07-11 15:50:18 +00:00
Java_java_lang_Math_floor(JNIEnv*, jclass, jdouble val)
{
return floor(val);
}
2007-10-12 18:53:56 +00:00
extern "C" JNIEXPORT jdouble JNICALL
2014-07-11 15:50:18 +00:00
Java_java_lang_Math_ceil(JNIEnv*, jclass, jdouble val)
2007-10-12 18:53:56 +00:00
{
return ceil(val);
}
extern "C" JNIEXPORT jdouble JNICALL
2014-07-11 15:50:18 +00:00
Java_java_lang_Math_exp(JNIEnv*, jclass, jdouble exp)
{
return pow(M_E, exp);
}
extern "C" JNIEXPORT jint JNICALL
2014-07-11 15:50:18 +00:00
Java_java_lang_Double_fillBufferWithDouble(JNIEnv* e,
jclass,
jdouble val,
jbyteArray buffer,
jint bufferSize)
{
jboolean isCopy;
jbyte* buf = e->GetByteArrayElements(buffer, &isCopy);
jint count = snprintf(reinterpret_cast<char*>(buf), bufferSize, "%g", val);
e->ReleaseByteArrayElements(buffer, buf, 0);
return count;
}