various fixes to get avian-swt-examples working with the Android class library

This commit is contained in:
Joel Dice 2013-02-26 16:24:02 -07:00
parent 025b628894
commit e6a46fc014
4 changed files with 84 additions and 36 deletions

View File

@ -169,9 +169,11 @@ ifneq ($(android),)
-D_FILE_OFFSET_BITS=64 \ -D_FILE_OFFSET_BITS=64 \
-g3 \ -g3 \
-Werror \ -Werror \
-fPIC -fPIC \
classpath-lflags := $(android)/icu4c/lib/libicuuc.a \ -fvisibility=hidden
classpath-lflags := \
$(android)/icu4c/lib/libicui18n.a \ $(android)/icu4c/lib/libicui18n.a \
$(android)/icu4c/lib/libicuuc.a \
$(android)/icu4c/lib/libicudata.a \ $(android)/icu4c/lib/libicudata.a \
$(android)/fdlibm/libfdm.a \ $(android)/fdlibm/libfdm.a \
$(android)/expat/.libs/libexpat.a \ $(android)/expat/.libs/libexpat.a \
@ -1413,9 +1415,10 @@ $(classpath-build)/%.class: $(classpath-src)/%.java
$(classpath-dep): $(classpath-sources) $(classpath-jar-dep) $(classpath-dep): $(classpath-sources) $(classpath-jar-dep)
@echo "compiling classpath classes" @echo "compiling classpath classes"
@mkdir -p $(classpath-build) @mkdir -p $(classpath-build)
classes="$(shell $(MAKE) -s --no-print-directory build=$(build) \
$(classpath-classes))"; if [ -n "$${classes}" ]; then \
$(javac) -d $(classpath-build) -bootclasspath $(boot-classpath) \ $(javac) -d $(classpath-build) -bootclasspath $(boot-classpath) \
$(shell $(MAKE) -s --no-print-directory build=$(build) \ $${classes}; fi
$(classpath-classes))
@touch $(@) @touch $(@)
$(build)/android-src/%.cpp: $(luni-native)/%.cpp $(build)/android-src/%.cpp: $(luni-native)/%.cpp

View File

@ -235,6 +235,8 @@ class MyClasspath : public Classpath {
t->m->processor->invoke t->m->processor->invoke
(t, constructor, thread, group, 0, NormalPriority, false); (t, constructor, thread, group, 0, NormalPriority, false);
set(t, thread, ThreadContextClassLoader, root(t, Machine::AppLoader));
initVmThread(t, thread); initVmThread(t, thread);
return thread; return thread;
@ -325,6 +327,10 @@ class MyClasspath : public Classpath {
virtual void virtual void
preBoot(Thread* t) preBoot(Thread* t)
{ {
// Android's System.initSystemProperties throws an NPE if
// LD_LIBRARY_PATH is not set as of this writing:
setenv("LD_LIBRARY_PATH", "", false);
{ object c = resolveClass { object c = resolveClass
(t, root(t, Machine::BootLoader), "java/lang/Runtime", false); (t, root(t, Machine::BootLoader), "java/lang/Runtime", false);
@ -531,6 +537,8 @@ jniRegisterNativeMethods(JNIEnv* e, const char* className,
if (c) { if (c) {
e->vtable->RegisterNatives(e, c, methods, methodCount); e->vtable->RegisterNatives(e, c, methods, methodCount);
} else {
e->vtable->ExceptionClear(e);
} }
return 0; return 0;
@ -693,7 +701,8 @@ extern "C" JNIEXPORT int64_t JNICALL
Avian_java_lang_String_equals Avian_java_lang_String_equals
(Thread* t, object, uintptr_t* arguments) (Thread* t, object, uintptr_t* arguments)
{ {
return stringEqual(t, reinterpret_cast<object>(arguments[0]), return arguments[1] and stringEqual
(t, reinterpret_cast<object>(arguments[0]),
reinterpret_cast<object>(arguments[1])); reinterpret_cast<object>(arguments[1]));
} }
@ -917,8 +926,14 @@ extern "C" JNIEXPORT int64_t JNICALL
Avian_dalvik_system_VMRuntime_properties Avian_dalvik_system_VMRuntime_properties
(Thread* t, object, uintptr_t*) (Thread* t, object, uintptr_t*)
{ {
return reinterpret_cast<uintptr_t> object array = makeObjectArray(t, type(t, Machine::StringType), 1);
(makeObjectArray(t, type(t, Machine::StringType), 0)); PROTECT(t, array);
object property = makeString(t, "java.protocol.handler.pkgs=avian");
set(t, array, ArrayBody, property);
return reinterpret_cast<uintptr_t>(array);
} }
extern "C" JNIEXPORT void JNICALL extern "C" JNIEXPORT void JNICALL
@ -928,6 +943,24 @@ Avian_java_lang_Runtime_gc
collect(t, Heap::MajorCollection); collect(t, Heap::MajorCollection);
} }
extern "C" JNIEXPORT int64_t JNICALL
Avian_java_lang_Runtime_nativeLoad
(Thread* t, object, uintptr_t* arguments)
{
object name = reinterpret_cast<object>(arguments[0]);
PROTECT(t, name);
unsigned length = stringLength(t, name);
THREAD_RUNTIME_ARRAY(t, char, n, length + 1);
stringChars(t, name, RUNTIME_ARRAY_BODY(n));
if (loadLibrary(t, "", RUNTIME_ARRAY_BODY(n), false, true)) {
return 0;
} else {
return reinterpret_cast<uintptr_t>(name);
}
}
extern "C" JNIEXPORT void JNICALL extern "C" JNIEXPORT void JNICALL
Avian_java_lang_System_arraycopy Avian_java_lang_System_arraycopy
(Thread* t, object, uintptr_t* arguments) (Thread* t, object, uintptr_t* arguments)
@ -1085,27 +1118,19 @@ Avian_java_lang_Math_max
} }
extern "C" JNIEXPORT int64_t JNICALL extern "C" JNIEXPORT int64_t JNICALL
Avian_java_lang_Math_floor Avian_java_lang_Math_cos
(Thread*, object, uintptr_t* arguments) (Thread*, object, uintptr_t* arguments)
{ {
int64_t v; memcpy(&v, arguments, 8); int64_t v; memcpy(&v, arguments, 8);
return doubleToBits(floor(bitsToDouble(v))); return doubleToBits(cos(bitsToDouble(v)));
} }
extern "C" JNIEXPORT int64_t JNICALL extern "C" JNIEXPORT int64_t JNICALL
Avian_java_lang_Math_ceil Avian_java_lang_Math_sin
(Thread*, object, uintptr_t* arguments) (Thread*, object, uintptr_t* arguments)
{ {
int64_t v; memcpy(&v, arguments, 8); int64_t v; memcpy(&v, arguments, 8);
return doubleToBits(ceil(bitsToDouble(v))); return doubleToBits(sin(bitsToDouble(v)));
}
extern "C" JNIEXPORT int64_t JNICALL
Avian_java_lang_Math_log
(Thread*, object, uintptr_t* arguments)
{
int64_t v; memcpy(&v, arguments, 8);
return doubleToBits(log(bitsToDouble(v)));
} }
extern "C" JNIEXPORT int64_t JNICALL extern "C" JNIEXPORT int64_t JNICALL
@ -1213,8 +1238,12 @@ Avian_java_lang_Class_getNameNative
object name = className object name = className
(t, jclassVmClass(t, reinterpret_cast<object>(arguments[0]))); (t, jclassVmClass(t, reinterpret_cast<object>(arguments[0])));
THREAD_RUNTIME_ARRAY(t, char, s, byteArrayLength(t, name));
replace('/', '.', RUNTIME_ARRAY_BODY(s),
reinterpret_cast<char*>(&byteArrayBody(t, name, 0)));
return reinterpret_cast<uintptr_t> return reinterpret_cast<uintptr_t>
(t->m->classpath->makeString(t, name, 0, byteArrayLength(t, name) - 1)); (makeString(t, "%s", RUNTIME_ARRAY_BODY(s)));
} }
extern "C" JNIEXPORT int64_t JNICALL extern "C" JNIEXPORT int64_t JNICALL
@ -1605,17 +1634,6 @@ Avian_libcore_io_Memory_peekByte
return *reinterpret_cast<int8_t*>(address); return *reinterpret_cast<int8_t*>(address);
} }
extern "C" JNIEXPORT void JNICALL
Avian_libcore_io_Memory_peekByteArray
(Thread* t, object, uintptr_t* arguments)
{
int64_t address; memcpy(&address, arguments, 8);
object array = reinterpret_cast<object>(arguments[2]);
memcpy(&byteArrayBody(t, array, arguments[3]),
reinterpret_cast<int8_t*>(address),
arguments[4]);
}
extern "C" JNIEXPORT int64_t JNICALL extern "C" JNIEXPORT int64_t JNICALL
Avian_java_lang_System_nanoTime Avian_java_lang_System_nanoTime
(Thread* t, object, uintptr_t*) (Thread* t, object, uintptr_t*)
@ -1636,3 +1654,19 @@ Avian_java_lang_System_identityHashCode
{ {
return objectHash(t, reinterpret_cast<object>(arguments[0])); return objectHash(t, reinterpret_cast<object>(arguments[0]));
} }
extern "C" JNIEXPORT int64_t JNICALL
Avian_java_lang_System_mapLibraryName
(Thread* t, object, uintptr_t* arguments)
{
object original = reinterpret_cast<object>(arguments[0]);
unsigned originalLength = stringUTFLength(t, original);
THREAD_RUNTIME_ARRAY(t, char, originalChars, originalLength);
stringUTFChars
(t, original, RUNTIME_ARRAY_BODY(originalChars), originalLength);
return reinterpret_cast<uintptr_t>
(makeString(t, "%s%.*s%s", t->m->system->libraryPrefix(), originalLength,
RUNTIME_ARRAY_BODY(originalChars),
t->m->system->librarySuffix()));
}

View File

@ -3254,13 +3254,21 @@ registerNatives(Thread* t, uintptr_t* arguments)
for (int i = 0; i < methodCount; ++i) { for (int i = 0; i < methodCount; ++i) {
if (methods[i].function) { if (methods[i].function) {
// Android's class library sometimes prepends a mysterious "!"
// to the method signature, which we happily ignore:
const char* sig = methods[i].signature;
if (*sig == '!') ++ sig;
object method = findMethodOrNull object method = findMethodOrNull
(t, jclassVmClass(t, *c), methods[i].name, methods[i].signature); (t, jclassVmClass(t, *c), methods[i].name, sig);
if (method == 0 or (methodFlags(t, method) & ACC_NATIVE) == 0) { if (method == 0 or (methodFlags(t, method) & ACC_NATIVE) == 0) {
// The JNI spec says we must throw a NoSuchMethodError in this // The JNI spec says we must throw a NoSuchMethodError in this
// case, but that would prevent using a code shrinker like // case, but that would prevent using a code shrinker like
// ProGuard effectively. Instead, we just ignore it. // ProGuard effectively. Instead, we just ignore it.
// fprintf(stderr, "not found: %s.%s%s\n", &byteArrayBody(t, className(t, jclassVmClass(t, *c)), 0), methods[i].name, sig);
// abort(t);
} else { } else {
registerNative(t, method, methods[i].function); registerNative(t, method, methods[i].function);
} }

View File

@ -259,7 +259,10 @@ main(int ac, const char** av)
JNI_CreateJavaVM(&vm, &env, &vmArgs); JNI_CreateJavaVM(&vm, &env, &vmArgs);
JNIEnv* e = static_cast<JNIEnv*>(env); JNIEnv* e = static_cast<JNIEnv*>(env);
jclass c = e->FindClass(class_); jclass c = 0;
if (not e->ExceptionCheck()) {
c = e->FindClass(class_);
}
if (jar) { if (jar) {
free(const_cast<char*>(class_)); free(const_cast<char*>(class_));