// ====================================================================================== // Android archive compilation using xgo // ====================================================================================== #include #include #include bool StatusServiceSignalEvent(const char *jsonEvent); static JavaVM *gJavaVM = NULL; static jclass JavaClassPtr_StatusService = NULL; static jmethodID JavaMethodPtr_signalEvent = NULL; static bool JniLibraryInit(JNIEnv *env); /*! * @brief Get interface to JNI. * * @return true if thread should be detached from JNI. */ static bool JniAttach(JNIEnv **env) { jint status; if (gJavaVM == NULL) { env = NULL; } status = (*gJavaVM)->GetEnv(gJavaVM, (void **)env, JNI_VERSION_1_6); if (status == JNI_EDETACHED) { // attach thread to JNI //(*gJavaVM)->AttachCurrentThread( gJavaVM, (void **)env, NULL ); // Oracle JNI API (*gJavaVM)->AttachCurrentThread(gJavaVM, env, NULL); // Android JNI API return true; } else if (status != JNI_OK) { return false; } return false; } /*! * @brief The VM calls JNI_OnLoad when the native library is loaded. */ JNIEXPORT jint JNI_OnLoad(JavaVM* vm, void* reserved) { bool detach; JNIEnv *env; int result = JNI_VERSION_1_6; gJavaVM = vm; // attach thread to JNI detach = JniAttach(&env); if (env == NULL) { // failed gJavaVM = NULL; return 0; } if (!JniLibraryInit(env)) { // fail loading of JNI library result = 0; } if (detach) { // detach thread from JNI (*gJavaVM)->DetachCurrentThread(gJavaVM); } if (result != JNI_VERSION_1_6) { gJavaVM = NULL; } return result; } /*! * @brief Initialize library. */ bool JniLibraryInit(JNIEnv *env) { int i; JavaClassPtr_StatusService = (*env)->FindClass(env, "im/status/ethereum/module/StatusService"); if (JavaClassPtr_StatusService == NULL) return false; JavaClassPtr_StatusService = (jclass)(*env)->NewGlobalRef(env, JavaClassPtr_StatusService); if (JavaClassPtr_StatusService == NULL) return false; struct { bool bStatic; jclass classPtr; jmethodID *methodPtr; const char *methodId; const char *params; } javaMethodDescriptors[] = { { true, JavaClassPtr_StatusService, &JavaMethodPtr_signalEvent, // &JavaMethodPtr_someNonStaticMethod "signalEvent", // someNonStaticMethod "(Ljava/lang/String;)V" }, }; for (i = 0; i < sizeof(javaMethodDescriptors) / sizeof(javaMethodDescriptors[0]); i++) { if (javaMethodDescriptors[i].bStatic) { *(javaMethodDescriptors[i].methodPtr) = (*env)->GetStaticMethodID( env, javaMethodDescriptors[i].classPtr, javaMethodDescriptors[i].methodId, javaMethodDescriptors[i].params); } else { *(javaMethodDescriptors[i].methodPtr) = (*env)->GetMethodID( env, javaMethodDescriptors[i].classPtr, javaMethodDescriptors[i].methodId, javaMethodDescriptors[i].params); } if (*(javaMethodDescriptors[i].methodPtr) == NULL) return false; } return true; } /*! * @brief Calls static method signalEvent of class im.status.ethereum.module.StatusService. * * @param jsonEvent - UTF8 string */ bool StatusServiceSignalEvent(const char *jsonEvent) { bool detach; JNIEnv *env; // attach thread to JNI detach = JniAttach( &env ); if (env == NULL) { // failed return false; } jstring javaJsonEvent = NULL; if (jsonEvent != NULL) { javaJsonEvent = (*env)->NewStringUTF(env, jsonEvent); } (*env)->CallStaticVoidMethod(env, JavaClassPtr_StatusService, JavaMethodPtr_signalEvent, javaJsonEvent); if (javaJsonEvent != NULL) (*env)->DeleteLocalRef(env, javaJsonEvent); if (detach) { // detach thread from JNI (*gJavaVM)->DetachCurrentThread(gJavaVM); } return true; } void SetEventCallback(void *cb) { }