- Split bemain to first prepare all irgs, then instrument and then do the rest
[libfirm] / ir / be / bejavacoal.c
1
2 #ifdef HAVE_CONFIG_H
3 #include "config.h"
4 #endif
5
6 #ifdef WITH_LIBCORE
7 #include <libcore/lc_opts.h>
8 #include <libcore/lc_opts_enum.h>
9 #endif /* WITH_LIBCORE */
10
11 #ifdef _WIN32
12 #include <windows.h>
13 #else
14 #include <dlfcn.h>
15 #endif
16
17 #include <signal.h>
18 #include <stdlib.h>
19 #include <assert.h>
20 #include <stdio.h>
21
22 #include "bejavacoal.h"
23
24 /* Path to the jar file. A little OS dependent convenience. */
25 #ifdef _WIN32
26 static char jar_file[512] = "y:\\user\\hack\\public\\coal.jar";
27 #else
28 static char jar_file[512] = "/ben/hack/public/coal.jar";
29 #endif
30
31 static char cls_name[256] = "coalescing/mst/safe/Algo";
32
33 /* Name of teh JVM dll/so */
34 static char jvm_lib[512] = { 0 };
35
36 #ifdef WITH_LIBCORE
37 static const lc_opt_table_entry_t options[] = {
38         LC_OPT_ENT_STR      ("jvm",  "absolute path to jvm dll",                    jvm_lib, sizeof(jvm_lib)),
39         LC_OPT_ENT_STR      ("jar",  "jar file of the coalescer",                   jar_file, sizeof(jar_file)),
40         LC_OPT_ENT_STR      ("cls",  "name of the class providing the factory",     cls_name, sizeof(cls_name)),
41         { NULL }
42 };
43
44 void be_java_coal_register_options(lc_opt_entry_t *grp)
45 {
46         lc_opt_entry_t *jc_grp = lc_opt_get_grp(grp, "jc");
47         lc_opt_add_table(jc_grp, options);
48 }
49 #endif
50
51 #ifdef WITH_JVM
52 #include <jni.h>
53
54 typedef struct _jni_env_t {
55         JavaVM *jvm;
56         JNIEnv *jni;
57 } jni_env_t;
58
59 /*
60
61         Ugly code to retrieve the JVM dll/so file.
62
63 */
64
65 #ifdef _WIN32
66 /* Win32 version */
67 static void *find_jvm_symbol(const char *vmlibpath, const char *sym)
68 {
69         HINSTANCE hVM = LoadLibrary(vmlibpath);
70         return hVM ? GetProcAddress(hVM, sym) : NULL;
71 }
72
73 #define JRE_KEY "SOFTWARE\\JavaSoft\\Java Development Kit"
74
75 static char *locate_jvm_lib(char *path, size_t path_len)
76 {
77         char version[32];
78         char buf[256];
79         DWORD version_len = sizeof(version);
80         DWORD dwPathLen = path_len;
81         HKEY hKey;
82
83         RegOpenKeyEx(HKEY_LOCAL_MACHINE, JRE_KEY, 0, KEY_QUERY_VALUE, &hKey);
84         RegQueryValueEx(hKey, "CurrentVersion", NULL, NULL, (LPBYTE) version, &version_len);
85         RegCloseKey(hKey);
86
87         _snprintf(buf, sizeof(buf), JRE_KEY "\\%s", version);
88         RegOpenKeyEx(HKEY_LOCAL_MACHINE, buf, 0, KEY_QUERY_VALUE, &hKey);
89         RegQueryValueEx(hKey, "JavaHome", NULL, NULL, (LPBYTE) path, &dwPathLen);
90         RegCloseKey(hKey);
91
92         strncat(path, "\\jre\\bin\\server\\jvm.dll", path_len);
93         return path;
94 }
95
96 #else
97 /* Unix version */
98 static void *find_jvm_symbol(const char *vmlibpath, const char *sym)
99 {
100         void *libVM = dlopen(vmlibpath, RTLD_LAZY);
101         return libVM ? dlsym(libVM, sym) : NULL;
102 }
103
104 static char *locate_jvm_lib(char *path, size_t n)
105 {
106         return NULL;
107 }
108 #endif
109
110 static int start_vm(jni_env_t *env, int argc, char *argv[])
111 {
112         int i;
113         long ret;
114         JavaVMInitArgs args;
115         JavaVMOption *opts;
116
117         long (JNICALL * create_func)(JavaVM **, void **, void *) = find_jvm_symbol(jvm_lib, "JNI_CreateJavaVM");
118
119         if(!create_func) {
120                 fprintf(stderr, "could not find JVM creation function\n");
121                 exit(1);
122         }
123
124         memset(&args, 0, sizeof(args));
125         opts = malloc(argc * sizeof(opts[0]));
126         for(i = 0; i < argc; ++i) {
127                 opts[i].optionString = argv[i];
128                 opts[i].extraInfo    = NULL;
129         }
130
131         args.version  = JNI_VERSION_1_4;
132         args.nOptions = argc;
133         args.options  = opts;
134         args.ignoreUnrecognized = JNI_FALSE;
135
136         ret = create_func(&env->jvm, (void **) &env->jni, &args);
137         free(opts);
138         if(ret != JNI_OK) {
139                 fprintf(stderr, "JNI_CreateJavaVM returned errrocode %ld\n" , ret);
140                 return 0;
141         }
142
143         return 1;
144 }
145
146 static void stop_vm(jni_env_t *env)
147 {
148         JavaVM *jvm = env->jvm;
149         (*jvm)->DetachCurrentThread(jvm);
150         (*jvm)->DestroyJavaVM(jvm);
151 }
152
153 static int jvm_inited = 0;
154 static jni_env_t env;
155 void (*old_int_handler)(int);
156 void (*old_abrt_handler)(int);
157
158 static void sig_jvm_destroy_at_exit(int signal)
159 {
160         if(jvm_inited)
161                 stop_vm(&env);
162
163         switch(signal) {
164         case SIGABRT:
165                 old_abrt_handler(signal);
166                 break;
167         case SIGINT:
168                 old_int_handler(signal);
169                 break;
170         default:;
171         }
172 }
173
174 static void jvm_destroy_at_exit(void)
175 {
176         sig_jvm_destroy_at_exit(0);
177 }
178
179 static jni_env_t *get_jvm(void)
180 {
181         char cp_param[512];
182         char *args[1];
183
184         if(!jvm_inited) {
185                 /* Find the dll */
186                 if(strlen(jvm_lib) == 0) {
187                         if(!locate_jvm_lib(jvm_lib, sizeof(jvm_lib))) {
188                                 fprintf(stderr, "could not find jvm library\n");
189                                 exit(1);
190                         }
191                 }
192
193                 snprintf(cp_param, sizeof(cp_param), "-Djava.class.path=%s", jar_file);
194                 args[0] = cp_param;
195                 if(!start_vm(&env, sizeof(args) / sizeof(args[0]), args)) {
196                         fprintf(stderr, "Couldn't initialize java VM\n");
197                         abort();
198                 }
199                 jvm_inited = 1;
200                 old_int_handler  = signal(SIGINT,  sig_jvm_destroy_at_exit);
201                 old_abrt_handler = signal(SIGABRT, sig_jvm_destroy_at_exit);
202                 atexit(jvm_destroy_at_exit);
203         }
204
205         return &env;
206 }
207
208 static void check(jni_env_t *env, const char *file, int line)
209 {
210         JNIEnv *jni = env->jni;
211         jboolean exc = (*jni)->ExceptionCheck(jni);
212         if(exc) {
213                 fprintf(stderr, "%s:%d: ", file, line);
214                 (*jni)->ExceptionDescribe(jni);
215                 (*jni)->ExceptionClear(jni);
216                 stop_vm(env);
217                 abort();
218         }
219 }
220
221 #define CHECK(env) check(env, __FILE__, __LINE__)
222
223 enum {
224         mth_add_int_edge,
225         mth_add_aff_edge,
226         mth_set_color,
227         //mth_set_debug,
228         mth_get_color,
229         mth_forbid_color,
230         mth_coalesce,
231         mth_dump,
232         mth_finish,
233         mth_last
234 };
235
236 struct _mth_info_t {
237         const char *name;
238         const char *sig;
239 };
240
241 static const struct _mth_info_t mthis[mth_last] = {
242         { "addIntEdge",  "(II)V"                   }, /* public void addIntEdge(int, int); */
243         { "addAffEdge",  "(III)V"                  }, /* public void addAffEdge(int, int, int); */
244         { "setColor",    "(II)V"                   }, /* public void setColor(int, int); */
245         //{ "setDebug",    "(ILjava/lang/String;)V"  }, /* public void setDebug(int, String); */
246         { "getColor",    "(I)I"                    }, /* public int getColor(int); */
247         { "forbidColor", "(II)V"                   }, /* public void forbidColor(int, int); */
248         { "coalesce",    "()V"                     }, /* public void coalesce(); */
249         { "dump",        "(Ljava/lang/String;)V"   }, /* public void dump(String); */
250         { "finish",      "()V"                     }  /* public void finish(); */
251 };
252
253 /* public static coalescing.Extern createExtern(java.lang.String, int, int, int); */
254 static const struct _mth_info_t mthi_factory = {
255         "createExtern", "(Ljava/lang/String;III)Lcoalescing/Extern;"
256 };
257
258 struct _be_java_coal_t {
259         jni_env_t *env;
260         jclass    cls;
261         jobject   obj;
262
263         jmethodID mth_ids[mth_last];
264 };
265
266 static void jc_call_void(be_java_coal_t *c, int mth_index, ...)
267 {
268         JNIEnv *jni   = c->env->jni;
269         jmethodID mid = c->mth_ids[mth_index];
270
271         va_list args;
272
273         va_start(args, mth_index);
274         (*jni)->CallVoidMethodV(jni, c->obj, mid, args);
275         CHECK(c->env);
276         va_end(args);
277 }
278
279 static int jc_call_int(be_java_coal_t *c, int mth_index, ...)
280 {
281         JNIEnv *jni   = c->env->jni;
282         jmethodID mid = c->mth_ids[mth_index];
283
284         int res;
285         va_list args;
286
287         va_start(args, mth_index);
288         res = (*jni)->CallIntMethodV(jni, c->obj, mid, args);
289         CHECK(c->env);
290         va_end(args);
291
292         return res;
293 }
294
295 be_java_coal_t *be_java_coal_init(const char *graph_name, int n_nodes, int n_regs, int dbg_level)
296 {
297         be_java_coal_t *c;
298         jni_env_t *env = get_jvm();
299         JNIEnv *jni = env->jni;
300         jmethodID fact;
301         jclass cls;
302         jstring str;
303         int i;
304
305         c = malloc(sizeof(c[0]));
306         memset(c, 0, sizeof(c[0]));
307         c->env = env;
308
309         /* Find the class we are are looking for. */
310         cls = (*jni)->FindClass(jni, cls_name);
311         CHECK(env);
312
313         /* Get the static factory method. */
314         fact = (*jni)->GetStaticMethodID(jni, cls, mthi_factory.name, mthi_factory.sig);
315         CHECK(env);
316
317         /* Call the factory. */
318         str = (*jni)->NewStringUTF(jni, graph_name);
319         CHECK(env);
320         c->obj = (*jni)->CallStaticObjectMethod(jni, cls, fact, str, n_nodes, n_regs, dbg_level);
321         CHECK(env);
322         c->cls = (*jni)->GetObjectClass(jni, c->obj);
323
324         /* Reference the created object. */
325         c->obj = (*jni)->NewGlobalRef(jni, c->obj);
326         CHECK(env);
327
328         /* Lookup the member methods of the object. */
329         for(i = 0; i < mth_last; ++i) {
330                 c->mth_ids[i] = (*jni)->GetMethodID(jni, c->cls, mthis[i].name, mthis[i].sig);
331                 CHECK(env);
332         }
333
334         return c;
335 }
336
337 void be_java_coal_destroy(be_java_coal_t *c) {
338         JNIEnv *jni = c->env->jni;
339         jc_call_void(c, mth_finish);
340         (*jni)->DeleteGlobalRef(jni, c->obj);
341         free(c);
342 }
343
344 void be_java_coal_add_int_edge(be_java_coal_t *c, int n, int m)
345 {
346         jc_call_void(c, mth_add_int_edge, (jint) n, (jint) m);
347 }
348
349 void be_java_coal_add_aff_edge(be_java_coal_t *c, int n, int m, int weight)
350 {
351         jc_call_void(c, mth_add_aff_edge, (jint) n, (jint) m, (jint) weight);
352 }
353
354 void be_java_coal_set_color(be_java_coal_t *c, int n, int col)
355 {
356         jc_call_void(c, mth_set_color, (jint) n, (jint) col);
357 }
358
359 void be_java_coal_set_debug(be_java_coal_t *c, int n, const char *dbg)
360 {
361 #if 0
362         JNIEnv *jni   = c->env->jni;
363         jmethodID mid = c->mth_ids[mth_set_debug];
364         jstring str;
365
366         str = (*jni)->NewStringUTF(jni, dbg);
367         CHECK(c->env);
368         (*jni)->CallVoidMethod(jni, c->obj, mid, (jint) n, str);
369         CHECK(c->env);
370 #endif
371 }
372
373 void be_java_coal_forbid_color(be_java_coal_t *c, int n, int col)
374 {
375         jc_call_void(c, mth_forbid_color, (jint) n, (jint) col);
376 }
377
378 void be_java_coal_coalesce(be_java_coal_t *c)
379 {
380         jc_call_void(c, mth_coalesce);
381 }
382
383 void be_java_coal_dump(be_java_coal_t *c, const char *fn)
384 {
385         JNIEnv *jni   = c->env->jni;
386         jmethodID mid = c->mth_ids[mth_dump];
387         jstring str;
388
389         str = (*jni)->NewStringUTF(jni, fn);
390         CHECK(c->env);
391         (*jni)->CallVoidMethod(jni, c->obj, mid, str);
392         CHECK(c->env);
393 }
394
395 int be_java_coal_get_color(be_java_coal_t *c, int n)
396 {
397         return jc_call_int(c, mth_get_color, (jint) n);
398 }
399
400 #else
401
402 be_java_coal_t *be_java_coal_init(const char *graph_name, int n_nodes, int n_regs, int dbg_level)
403 {
404         assert(0 && "use --enable-jvm");
405         return NULL;
406 }
407
408 void be_java_coal_destroy(be_java_coal_t *c)
409 {
410         assert(0 && "use --enable-jvm");
411 }
412
413
414 void be_java_coal_add_int_edge(be_java_coal_t *c, int n, int m)
415 {
416         assert(0 && "use --enable-jvm");
417 }
418
419 void be_java_coal_add_aff_edge(be_java_coal_t *c, int n, int m, int weight)
420 {
421         assert(0 && "use --enable-jvm");
422 }
423
424 void be_java_coal_set_color(be_java_coal_t *c, int n, int col)
425 {
426         assert(0 && "use --enable-jvm");
427 }
428
429 void be_java_coal_set_debug(be_java_coal_t *c, int n, const char *dbg)
430 {
431         assert(0 && "use --enable-jvm");
432 }
433
434 void be_java_coal_forbid_color(be_java_coal_t *c, int n, int col)
435 {
436         assert(0 && "use --enable-jvm");
437 }
438
439 void be_java_coal_coalesce(be_java_coal_t *c)
440 {
441         assert(0 && "use --enable-jvm");
442 }
443
444 void be_java_coal_dump(be_java_coal_t *c, const char *fn)
445 {
446         assert(0 && "use --enable-jvm");
447 }
448
449 int be_java_coal_get_color(be_java_coal_t *c, int n)
450 {
451         assert(0 && "use --enable-jvm");
452         return -1;
453 }
454
455
456 #endif /* WITH_JVM */