mirror of
				https://github.com/godotengine/godot.git
				synced 2025-11-04 07:31:16 +00:00 
			
		
		
		
	
		
			
	
	
		
			301 lines
		
	
	
	
		
			8.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
		
		
			
		
	
	
			301 lines
		
	
	
	
		
			8.9 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| 
								 | 
							
								/*************************************************************************/
							 | 
						||
| 
								 | 
							
								/*  godot_plugin_jni.cpp                                                 */
							 | 
						||
| 
								 | 
							
								/*************************************************************************/
							 | 
						||
| 
								 | 
							
								/*                       This file is part of:                           */
							 | 
						||
| 
								 | 
							
								/*                           GODOT ENGINE                                */
							 | 
						||
| 
								 | 
							
								/*                      https://godotengine.org                          */
							 | 
						||
| 
								 | 
							
								/*************************************************************************/
							 | 
						||
| 
								 | 
							
								/* Copyright (c) 2007-2020 Juan Linietsky, Ariel Manzur.                 */
							 | 
						||
| 
								 | 
							
								/* Copyright (c) 2014-2020 Godot Engine contributors (cf. AUTHORS.md).   */
							 | 
						||
| 
								 | 
							
								/*                                                                       */
							 | 
						||
| 
								 | 
							
								/* Permission is hereby granted, free of charge, to any person obtaining */
							 | 
						||
| 
								 | 
							
								/* a copy of this software and associated documentation files (the       */
							 | 
						||
| 
								 | 
							
								/* "Software"), to deal in the Software without restriction, including   */
							 | 
						||
| 
								 | 
							
								/* without limitation the rights to use, copy, modify, merge, publish,   */
							 | 
						||
| 
								 | 
							
								/* distribute, sublicense, and/or sell copies of the Software, and to    */
							 | 
						||
| 
								 | 
							
								/* permit persons to whom the Software is furnished to do so, subject to */
							 | 
						||
| 
								 | 
							
								/* the following conditions:                                             */
							 | 
						||
| 
								 | 
							
								/*                                                                       */
							 | 
						||
| 
								 | 
							
								/* The above copyright notice and this permission notice shall be        */
							 | 
						||
| 
								 | 
							
								/* included in all copies or substantial portions of the Software.       */
							 | 
						||
| 
								 | 
							
								/*                                                                       */
							 | 
						||
| 
								 | 
							
								/* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,       */
							 | 
						||
| 
								 | 
							
								/* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF    */
							 | 
						||
| 
								 | 
							
								/* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.*/
							 | 
						||
| 
								 | 
							
								/* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY  */
							 | 
						||
| 
								 | 
							
								/* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT,  */
							 | 
						||
| 
								 | 
							
								/* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE     */
							 | 
						||
| 
								 | 
							
								/* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.                */
							 | 
						||
| 
								 | 
							
								/*************************************************************************/
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#include "godot_plugin_jni.h"
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								#include <core/engine.h>
							 | 
						||
| 
								 | 
							
								#include <core/error_macros.h>
							 | 
						||
| 
								 | 
							
								#include <core/project_settings.h>
							 | 
						||
| 
								 | 
							
								#include <platform/android/jni_utils.h>
							 | 
						||
| 
								 | 
							
								#include <platform/android/string_android.h>
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								class JNISingleton : public Object {
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									GDCLASS(JNISingleton, Object);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									struct MethodData {
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
										jmethodID method;
							 | 
						||
| 
								 | 
							
										Variant::Type ret_type;
							 | 
						||
| 
								 | 
							
										Vector<Variant::Type> argtypes;
							 | 
						||
| 
								 | 
							
									};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									jobject instance;
							 | 
						||
| 
								 | 
							
									Map<StringName, MethodData> method_map;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								public:
							 | 
						||
| 
								 | 
							
									virtual Variant call(const StringName &p_method, const Variant **p_args, int p_argcount, Variant::CallError &r_error) {
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
										ERR_FAIL_COND_V(!instance, Variant());
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
										r_error.error = Variant::CallError::CALL_OK;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
										Map<StringName, MethodData>::Element *E = method_map.find(p_method);
							 | 
						||
| 
								 | 
							
										if (!E) {
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
											r_error.error = Variant::CallError::CALL_ERROR_INVALID_METHOD;
							 | 
						||
| 
								 | 
							
											return Variant();
							 | 
						||
| 
								 | 
							
										}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
										int ac = E->get().argtypes.size();
							 | 
						||
| 
								 | 
							
										if (ac < p_argcount) {
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
											r_error.error = Variant::CallError::CALL_ERROR_TOO_FEW_ARGUMENTS;
							 | 
						||
| 
								 | 
							
											r_error.argument = ac;
							 | 
						||
| 
								 | 
							
											return Variant();
							 | 
						||
| 
								 | 
							
										}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
										if (ac > p_argcount) {
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
											r_error.error = Variant::CallError::CALL_ERROR_TOO_MANY_ARGUMENTS;
							 | 
						||
| 
								 | 
							
											r_error.argument = ac;
							 | 
						||
| 
								 | 
							
											return Variant();
							 | 
						||
| 
								 | 
							
										}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
										for (int i = 0; i < p_argcount; i++) {
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
											if (!Variant::can_convert(p_args[i]->get_type(), E->get().argtypes[i])) {
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
												r_error.error = Variant::CallError::CALL_ERROR_INVALID_ARGUMENT;
							 | 
						||
| 
								 | 
							
												r_error.argument = i;
							 | 
						||
| 
								 | 
							
												r_error.expected = E->get().argtypes[i];
							 | 
						||
| 
								 | 
							
											}
							 | 
						||
| 
								 | 
							
										}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
										jvalue *v = NULL;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
										if (p_argcount) {
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
											v = (jvalue *)alloca(sizeof(jvalue) * p_argcount);
							 | 
						||
| 
								 | 
							
										}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
										JNIEnv *env = ThreadAndroid::get_env();
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
										int res = env->PushLocalFrame(16);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
										ERR_FAIL_COND_V(res != 0, Variant());
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
										List<jobject> to_erase;
							 | 
						||
| 
								 | 
							
										for (int i = 0; i < p_argcount; i++) {
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
											jvalret vr = _variant_to_jvalue(env, E->get().argtypes[i], p_args[i]);
							 | 
						||
| 
								 | 
							
											v[i] = vr.val;
							 | 
						||
| 
								 | 
							
											if (vr.obj)
							 | 
						||
| 
								 | 
							
												to_erase.push_back(vr.obj);
							 | 
						||
| 
								 | 
							
										}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
										Variant ret;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
										switch (E->get().ret_type) {
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
											case Variant::NIL: {
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
												env->CallVoidMethodA(instance, E->get().method, v);
							 | 
						||
| 
								 | 
							
											} break;
							 | 
						||
| 
								 | 
							
											case Variant::BOOL: {
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
												ret = env->CallBooleanMethodA(instance, E->get().method, v) == JNI_TRUE;
							 | 
						||
| 
								 | 
							
											} break;
							 | 
						||
| 
								 | 
							
											case Variant::INT: {
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
												ret = env->CallIntMethodA(instance, E->get().method, v);
							 | 
						||
| 
								 | 
							
											} break;
							 | 
						||
| 
								 | 
							
											case Variant::REAL: {
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
												ret = env->CallFloatMethodA(instance, E->get().method, v);
							 | 
						||
| 
								 | 
							
											} break;
							 | 
						||
| 
								 | 
							
											case Variant::STRING: {
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
												jobject o = env->CallObjectMethodA(instance, E->get().method, v);
							 | 
						||
| 
								 | 
							
												ret = jstring_to_string((jstring)o, env);
							 | 
						||
| 
								 | 
							
												env->DeleteLocalRef(o);
							 | 
						||
| 
								 | 
							
											} break;
							 | 
						||
| 
								 | 
							
											case Variant::POOL_STRING_ARRAY: {
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
												jobjectArray arr = (jobjectArray)env->CallObjectMethodA(instance, E->get().method, v);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
												ret = _jobject_to_variant(env, arr);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
												env->DeleteLocalRef(arr);
							 | 
						||
| 
								 | 
							
											} break;
							 | 
						||
| 
								 | 
							
											case Variant::POOL_INT_ARRAY: {
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
												jintArray arr = (jintArray)env->CallObjectMethodA(instance, E->get().method, v);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
												int fCount = env->GetArrayLength(arr);
							 | 
						||
| 
								 | 
							
												PoolVector<int> sarr;
							 | 
						||
| 
								 | 
							
												sarr.resize(fCount);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
												PoolVector<int>::Write w = sarr.write();
							 | 
						||
| 
								 | 
							
												env->GetIntArrayRegion(arr, 0, fCount, w.ptr());
							 | 
						||
| 
								 | 
							
												w.release();
							 | 
						||
| 
								 | 
							
												ret = sarr;
							 | 
						||
| 
								 | 
							
												env->DeleteLocalRef(arr);
							 | 
						||
| 
								 | 
							
											} break;
							 | 
						||
| 
								 | 
							
											case Variant::POOL_REAL_ARRAY: {
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
												jfloatArray arr = (jfloatArray)env->CallObjectMethodA(instance, E->get().method, v);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
												int fCount = env->GetArrayLength(arr);
							 | 
						||
| 
								 | 
							
												PoolVector<float> sarr;
							 | 
						||
| 
								 | 
							
												sarr.resize(fCount);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
												PoolVector<float>::Write w = sarr.write();
							 | 
						||
| 
								 | 
							
												env->GetFloatArrayRegion(arr, 0, fCount, w.ptr());
							 | 
						||
| 
								 | 
							
												w.release();
							 | 
						||
| 
								 | 
							
												ret = sarr;
							 | 
						||
| 
								 | 
							
												env->DeleteLocalRef(arr);
							 | 
						||
| 
								 | 
							
											} break;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
											case Variant::DICTIONARY: {
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
												jobject obj = env->CallObjectMethodA(instance, E->get().method, v);
							 | 
						||
| 
								 | 
							
												ret = _jobject_to_variant(env, obj);
							 | 
						||
| 
								 | 
							
												env->DeleteLocalRef(obj);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
											} break;
							 | 
						||
| 
								 | 
							
											default: {
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
												env->PopLocalFrame(NULL);
							 | 
						||
| 
								 | 
							
												ERR_FAIL_V(Variant());
							 | 
						||
| 
								 | 
							
											} break;
							 | 
						||
| 
								 | 
							
										}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
										while (to_erase.size()) {
							 | 
						||
| 
								 | 
							
											env->DeleteLocalRef(to_erase.front()->get());
							 | 
						||
| 
								 | 
							
											to_erase.pop_front();
							 | 
						||
| 
								 | 
							
										}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
										env->PopLocalFrame(NULL);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
										return ret;
							 | 
						||
| 
								 | 
							
									}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									jobject get_instance() const {
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
										return instance;
							 | 
						||
| 
								 | 
							
									}
							 | 
						||
| 
								 | 
							
									void set_instance(jobject p_instance) {
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
										instance = p_instance;
							 | 
						||
| 
								 | 
							
									}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									void add_method(const StringName &p_name, jmethodID p_method, const Vector<Variant::Type> &p_args, Variant::Type p_ret_type) {
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
										MethodData md;
							 | 
						||
| 
								 | 
							
										md.method = p_method;
							 | 
						||
| 
								 | 
							
										md.argtypes = p_args;
							 | 
						||
| 
								 | 
							
										md.ret_type = p_ret_type;
							 | 
						||
| 
								 | 
							
										method_map[p_name] = md;
							 | 
						||
| 
								 | 
							
									}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									JNISingleton() {
							 | 
						||
| 
								 | 
							
										instance = NULL;
							 | 
						||
| 
								 | 
							
									}
							 | 
						||
| 
								 | 
							
								};
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								static HashMap<String, JNISingleton *> jni_singletons;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								extern "C" {
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								JNIEXPORT void JNICALL Java_org_godotengine_godot_plugin_GodotPlugin_nativeRegisterSingleton(JNIEnv *env, jclass clazz, jstring name, jobject obj) {
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									String singname = jstring_to_string(name, env);
							 | 
						||
| 
								 | 
							
									JNISingleton *s = memnew(JNISingleton);
							 | 
						||
| 
								 | 
							
									s->set_instance(env->NewGlobalRef(obj));
							 | 
						||
| 
								 | 
							
									jni_singletons[singname] = s;
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									Engine::get_singleton()->add_singleton(Engine::Singleton(singname, s));
							 | 
						||
| 
								 | 
							
									ProjectSettings::get_singleton()->set(singname, s);
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								JNIEXPORT void JNICALL Java_org_godotengine_godot_plugin_GodotPlugin_nativeRegisterMethod(JNIEnv *env, jclass clazz, jstring sname, jstring name, jstring ret, jobjectArray args) {
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									String singname = jstring_to_string(sname, env);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									ERR_FAIL_COND(!jni_singletons.has(singname));
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									JNISingleton *s = jni_singletons.get(singname);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									String mname = jstring_to_string(name, env);
							 | 
						||
| 
								 | 
							
									String retval = jstring_to_string(ret, env);
							 | 
						||
| 
								 | 
							
									Vector<Variant::Type> types;
							 | 
						||
| 
								 | 
							
									String cs = "(";
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									int stringCount = env->GetArrayLength(args);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									for (int i = 0; i < stringCount; i++) {
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
										jstring string = (jstring)env->GetObjectArrayElement(args, i);
							 | 
						||
| 
								 | 
							
										const String rawString = jstring_to_string(string, env);
							 | 
						||
| 
								 | 
							
										types.push_back(get_jni_type(rawString));
							 | 
						||
| 
								 | 
							
										cs += get_jni_sig(rawString);
							 | 
						||
| 
								 | 
							
									}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									cs += ")";
							 | 
						||
| 
								 | 
							
									cs += get_jni_sig(retval);
							 | 
						||
| 
								 | 
							
									jclass cls = env->GetObjectClass(s->get_instance());
							 | 
						||
| 
								 | 
							
									jmethodID mid = env->GetMethodID(cls, mname.ascii().get_data(), cs.ascii().get_data());
							 | 
						||
| 
								 | 
							
									if (!mid) {
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
										print_line("Failed getting method ID " + mname);
							 | 
						||
| 
								 | 
							
									}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									s->add_method(mname, mid, types, get_jni_type(retval));
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
								JNIEXPORT void JNICALL Java_org_godotengine_godot_plugin_GodotPlugin_nativeRegisterGDNativeLibraries(JNIEnv *env, jobject obj, jobjectArray gdnlib_paths) {
							 | 
						||
| 
								 | 
							
									int gdnlib_count = env->GetArrayLength(gdnlib_paths);
							 | 
						||
| 
								 | 
							
									if (gdnlib_count == 0) {
							 | 
						||
| 
								 | 
							
										return;
							 | 
						||
| 
								 | 
							
									}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									// Retrieve the current list of gdnative libraries.
							 | 
						||
| 
								 | 
							
									Array singletons = Array();
							 | 
						||
| 
								 | 
							
									if (ProjectSettings::get_singleton()->has_setting("gdnative/singletons")) {
							 | 
						||
| 
								 | 
							
										singletons = ProjectSettings::get_singleton()->get("gdnative/singletons");
							 | 
						||
| 
								 | 
							
									}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									// Insert the libraries provided by the plugin
							 | 
						||
| 
								 | 
							
									for (int i = 0; i < gdnlib_count; i++) {
							 | 
						||
| 
								 | 
							
										jstring relative_path = (jstring)env->GetObjectArrayElement(gdnlib_paths, i);
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
										String path = "res://" + jstring_to_string(relative_path, env);
							 | 
						||
| 
								 | 
							
										if (!singletons.has(path)) {
							 | 
						||
| 
								 | 
							
											singletons.push_back(path);
							 | 
						||
| 
								 | 
							
										}
							 | 
						||
| 
								 | 
							
										env->DeleteLocalRef(relative_path);
							 | 
						||
| 
								 | 
							
									}
							 | 
						||
| 
								 | 
							
								
							 | 
						||
| 
								 | 
							
									// Insert the updated list back into project settings.
							 | 
						||
| 
								 | 
							
									ProjectSettings::get_singleton()->set("gdnative/singletons", singletons);
							 | 
						||
| 
								 | 
							
								}
							 | 
						||
| 
								 | 
							
								}
							 |