| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260 | /** * $Id$ * * Copyright (C) 2013 Konstantin Mosesov * * This file is part of Kamailio, a free SIP server. * * This file is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License as published by * the Free Software Foundation; either version 2 of the License, or * (at your option) any later version * * * This file is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the * GNU General Public License for more details. * * You should have received a copy of the GNU General Public License * along with this program; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA * */#include <libgen.h>#include "../../str.h"#include "../../sr_module.h"#include <jni.h>#include "global.h"#include "utils.h"#include "java_mod.h"#include "java_iface.h"#include "java_support.h"#include "java_native_methods.h"MODULE_VERSIONstatic str class_name = {.s = "Kamailio", .len = 10};static str child_init_mname = { .s = "child_init", .len = 0};static str java_options_str = { .s = "-Djava.compiler=NONE", .len = 21};static int mod_init(void);static int child_init(int rank);static void mod_destroy(void);/** module parameters */static param_export_t params[] = {    {"class_name",         STR_PARAM, &class_name },    {"child_init_method",  STR_PARAM, &child_init_mname },    {"java_options",	   STR_PARAM, &java_options_str },    {"force_cmd_exec", INT_PARAM, &force_cmd_exec },    {0,0,0}};/* * Exported functions */static cmd_export_t cmds[] = {    { "java_method_exec",		(cmd_function)j_nst_exec_0,	2,	NULL,	0,	ANY_ROUTE },    { "java_method_exec",		(cmd_function)j_nst_exec_1,	3,	NULL,	0,	ANY_ROUTE },    { "java_s_method_exec",		(cmd_function)j_s_nst_exec_0,	2,	NULL,	0,	ANY_ROUTE },    { "java_s_method_exec",		(cmd_function)j_s_nst_exec_1,	3,	NULL,	0,	ANY_ROUTE },    { "java_staticmethod_exec",		(cmd_function)j_st_exec_0,	2,	NULL,	0,	ANY_ROUTE },    { "java_staticmethod_exec",		(cmd_function)j_st_exec_1,	3,	NULL,	0,	ANY_ROUTE },    { "java_s_staticmethod_exec",	(cmd_function)j_s_st_exec_0,	2,	NULL,	0,	ANY_ROUTE },    { "java_s_staticmethod_exec",	(cmd_function)j_s_st_exec_1,	3,	NULL,	0,	ANY_ROUTE },    { 0, 0, 0, 0, 0, 0 }};/** module exports */struct module_exports exports = {    "app_java",                     /* module name *///    RTLD_NOW | RTLD_GLOBAL,         /* dlopen flags */    DEFAULT_DLFLAGS,		    /* dlopen flags */    cmds,                           /* exported functions */    params,                         /* exported parameters */    0,                              /* exported statistics */    0,                              /* exported MI functions */    0,                              /* exported pseudo-variables */    0,                              /* extra processes */    mod_init,                       /* module initialization function */    (response_function) NULL,       /* response handling function */    (destroy_function) mod_destroy, /* destroy function */    child_init                      /* per-child init function */};static int mod_init(void){    JavaVMInitArgs  vm_args;    jint res;    JavaVMOption *options;    char **opts;    int nOptions;    if (force_cmd_exec < 0 || force_cmd_exec > 1)    {	LM_ERR("Parameter force_cmd_exec should be either 0 or 1\n");	return -1;    }    if (force_cmd_exec)    {	LM_NOTICE("app_java: Parameter force_cmd_exec may cause a memory leaks if used from embedded languages\n");    }    options = (JavaVMOption *)pkg_malloc(sizeof(JavaVMOption));    if (!options)    {	LM_ERR("pkg_malloc() failed: Couldn't initialize Java VM: Not enough memory\n");	return -1;    }    memset(options, 0, sizeof(JavaVMOption));    LM_INFO("Initializing Java VM with options: %s\n", java_options_str.s);    opts = split(java_options_str.s, " ");    for (nOptions=0; opts[nOptions] != NULL; nOptions++)    {	options[nOptions].optionString = opts[nOptions];    }    /* IMPORTANT: specify vm_args version # if you use JDK1.1.2 and beyond */    vm_args.version = JNI_VERSION_1_2;    vm_args.nOptions = nOptions;    vm_args.ignoreUnrecognized = JNI_FALSE;    vm_args.options = options;    res = JNI_CreateJavaVM(&jvm, (void **)&env, &vm_args);    if (res < 0)    {	handle_VM_init_failure(res);	return -1;    }    LM_INFO("app_java: Java VM initialization OK\n");    // attach to current thread    (*jvm)->AttachCurrentThread(jvm, (void **)&env, NULL);    if ((*env)->ExceptionCheck(env))    {	handle_exception();	return -1;    }    KamailioClass = (*env)->FindClass(env, class_name.s);    if (!KamailioClass || (*env)->ExceptionCheck(env))    {	handle_exception();	(*jvm)->DetachCurrentThread(jvm);	return -1;    }    KamailioClassRef = (*env)->NewGlobalRef(env, KamailioClass);    if (!KamailioClassRef || (*env)->ExceptionCheck(env))    {	handle_exception();	(*jvm)->DetachCurrentThread(jvm);	return -1;    }    KamailioID = (*env)->GetMethodID(env, KamailioClass, "<init>", "()V");    if (!KamailioID || (*env)->ExceptionCheck(env))    {	handle_exception();	(*jvm)->DetachCurrentThread(jvm);	return -1;    }    // calling constructor    KamailioClassInstance = (*env)->NewObject(env, KamailioClass, KamailioID);    if (!KamailioClassInstance || (*env)->ExceptionCheck(env))    {	handle_exception();	(*jvm)->DetachCurrentThread(jvm);	return -1;    }    // keep a reference to kamailio class instance    KamailioClassInstanceRef = (*env)->NewGlobalRef(env, KamailioClassInstance);    if (!KamailioClassInstanceRef || (*env)->ExceptionCheck(env))    {	handle_exception();	(*jvm)->DetachCurrentThread(jvm);	return -1;    }    LM_INFO("app_java: module initialization OK\n");    if (jvm != NULL)        (*jvm)->DetachCurrentThread(jvm);    return 0;}static int child_init(int rank){    int retval;    jmethodID child_init_id;    // attach to current thread    (*jvm)->AttachCurrentThread(jvm, (void **)&env, NULL);    if ((*env)->ExceptionCheck(env))    {        handle_exception();        return -1;    }    child_init_id = (*env)->GetMethodID(env, KamailioClass, "child_init", "(I)I");    if ((*env)->ExceptionCheck(env))    {	handle_exception();    	(*jvm)->DetachCurrentThread(jvm);	return -1;    }    retval = (int)(*env)->CallIntMethod(env, KamailioClassInstanceRef, child_init_id, rank);    if ((*env)->ExceptionCheck(env))    {	handle_exception();    	(*jvm)->DetachCurrentThread(jvm);	return -1;    }    (*env)->DeleteLocalRef(env, child_init_id);    (*jvm)->DetachCurrentThread(jvm);    msg = NULL;    return retval;}static void mod_destroy(void){    if (env != NULL)    {	(*env)->DeleteGlobalRef(env, KamailioClassInstanceRef);	(*env)->DeleteGlobalRef(env, KamailioClassRef);    }    if (jvm != NULL)    {	(*jvm)->DetachCurrentThread(jvm);	(*jvm)->DestroyJavaVM(jvm);    }    if (msg)    {	pkg_free(msg);    }}
 |