1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
|
#include <stdlib.h>
#include <stdio.h>
#include <unistd.h>
#include <dlfcn.h>
#include <pthread.h>
#include <assert.h>
#include <android/log.h>
#include "jvm/jvm.h"
#include "wrapper/wrapper.h"
static int pfd[2];
static void*
log_thread(void *arg)
{
char buf[255];
ssize_t r;
while ((r = read(pfd[0], buf, sizeof(buf) - 1)) > 0) {
r = (r > 0 && buf[r - 1] == '\n' ? r - 1 : r);
buf[r] = 0;
__android_log_write(ANDROID_LOG_INFO, "native", buf);
}
return NULL;
}
static void
stdlog_workaround(void)
{
setvbuf(stdout, 0, _IOLBF, 0);
setvbuf(stderr, 0, _IONBF, 0);
pipe(pfd);
dup2(pfd[1], 1);
dup2(pfd[1], 2);
pthread_t thread;
pthread_create(&thread, 0, log_thread, NULL);
pthread_detach(thread);
}
JNIEXPORT jint JNICALL
JNI_OnLoad(JavaVM *vm, void *reserved)
{
stdlog_workaround();
__android_log_print(ANDROID_LOG_INFO, "native", "Loading liborig.so");
void *handle = dlopen("liborig.so", RTLD_NOW);
assert(handle);
jint (*JNI_OnLoad)(JavaVM*, void*) = dlsym(handle, "JNI_OnLoad");
assert(JNI_OnLoad);
struct jvm jvm;
jvm_init(&jvm);
__android_log_print(ANDROID_LOG_INFO, "native", "Call liborig's JNI_OnLoad with fakevm");
JNI_OnLoad(&jvm.vm, NULL);
JNI_OnLoad(vm, NULL);
JNIEnv *env;
if ((*vm)->GetEnv(vm, (void**)&env, JNI_VERSION_1_6) != JNI_OK)
return -1;
__android_log_print(ANDROID_LOG_INFO, "native", "Wrapping natives methods");
jclass old_klass = jvm.methods[0].method.klass;
JNINativeMethod methods[255];
for (size_t i = 0, c = 0; c < 255 && i < 255; ++i, ++c) {
if (jvm.methods[i].method.klass != old_klass) {
assert(old_klass > 0);
jclass klass = (*env)->FindClass(env, jvm.objects[(intptr_t)old_klass - 1].klass.name.data);
(*env)->RegisterNatives(env, klass, methods, c);
old_klass = jvm.methods[i].method.klass;
c = 0;
}
if (!jvm.methods[i].function)
break;
methods[c].name = jvm.methods[i].method.name.data;
methods[c].signature = jvm.methods[i].method.signature.data;
methods[c].fnPtr = wrapper_create(methods[c].name, jvm.methods[i].function);
}
return JNI_VERSION_1_6;
}
|