| 
									
										
										
										
											2017-10-02 23:24:00 +02:00
										 |  |  | /*************************************************************************/ | 
					
						
							| 
									
										
										
										
											2018-01-05 00:50:27 +01:00
										 |  |  | /*  gd_mono_internals.cpp                                                */ | 
					
						
							| 
									
										
										
										
											2017-10-02 23:24:00 +02:00
										 |  |  | /*************************************************************************/ | 
					
						
							|  |  |  | /*                       This file is part of:                           */ | 
					
						
							|  |  |  | /*                           GODOT ENGINE                                */ | 
					
						
							|  |  |  | /*                      https://godotengine.org                          */ | 
					
						
							|  |  |  | /*************************************************************************/ | 
					
						
							| 
									
										
										
										
											2022-01-03 21:27:34 +01:00
										 |  |  | /* Copyright (c) 2007-2022 Juan Linietsky, Ariel Manzur.                 */ | 
					
						
							|  |  |  | /* Copyright (c) 2014-2022 Godot Engine contributors (cf. AUTHORS.md).   */ | 
					
						
							| 
									
										
										
										
											2017-10-02 23:24:00 +02:00
										 |  |  | /*                                                                       */ | 
					
						
							|  |  |  | /* 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.                */ | 
					
						
							|  |  |  | /*************************************************************************/ | 
					
						
							| 
									
										
										
										
											2018-01-05 00:50:27 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-02 23:24:00 +02:00
										 |  |  | #include "gd_mono_internals.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "../csharp_script.h"
 | 
					
						
							|  |  |  | #include "../mono_gc_handle.h"
 | 
					
						
							| 
									
										
										
										
											2018-06-26 21:03:42 +02:00
										 |  |  | #include "../utils/macros.h"
 | 
					
						
							| 
									
										
										
										
											2019-02-04 20:39:02 +01:00
										 |  |  | #include "gd_mono_class.h"
 | 
					
						
							|  |  |  | #include "gd_mono_marshal.h"
 | 
					
						
							| 
									
										
										
										
											2017-10-02 23:24:00 +02:00
										 |  |  | #include "gd_mono_utils.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-02-27 03:30:20 +01:00
										 |  |  | #include "core/debugger/engine_debugger.h"
 | 
					
						
							|  |  |  | #include "core/debugger/script_debugger.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-06-26 21:03:42 +02:00
										 |  |  | #include <mono/metadata/exception.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-02 23:24:00 +02:00
										 |  |  | namespace GDMonoInternals { | 
					
						
							|  |  |  | void tie_managed_to_unmanaged(MonoObject *managed, Object *unmanaged) { | 
					
						
							|  |  |  | 	// This method should not fail
 | 
					
						
							| 
									
										
										
										
											2021-08-16 17:16:36 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-10-02 23:24:00 +02:00
										 |  |  | 	CRASH_COND(!unmanaged); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-08-09 03:39:45 +02:00
										 |  |  | 	// All mono objects created from the managed world (e.g.: 'new Player()')
 | 
					
						
							| 
									
										
										
										
											2017-10-02 23:24:00 +02:00
										 |  |  | 	// need to have a CSharpScript in order for their methods to be callable from the unmanaged side
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-04 18:03:15 +02:00
										 |  |  | 	RefCounted *rc = Object::cast_to<RefCounted>(unmanaged); | 
					
						
							| 
									
										
										
										
											2017-10-02 23:24:00 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	GDMonoClass *klass = GDMonoUtils::get_object_class(managed); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	CRASH_COND(!klass); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-04 20:39:02 +01:00
										 |  |  | 	GDMonoClass *native = GDMonoUtils::get_class_native_base(klass); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-04-02 01:20:12 +02:00
										 |  |  | 	CRASH_COND(native == nullptr); | 
					
						
							| 
									
										
										
										
											2019-02-04 20:39:02 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (native == klass) { | 
					
						
							|  |  |  | 		// If it's just a wrapper Godot class and not a custom inheriting class, then attach a
 | 
					
						
							|  |  |  | 		// script binding instead. One of the advantages of this is that if a script is attached
 | 
					
						
							|  |  |  | 		// later and it's not a C# script, then the managed object won't have to be disposed.
 | 
					
						
							|  |  |  | 		// Another reason for doing this is that this instance could outlive CSharpLanguage, which would
 | 
					
						
							|  |  |  | 		// be problematic when using a script. See: https://github.com/godotengine/godot/issues/25621
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		CSharpScriptBinding script_binding; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		script_binding.inited = true; | 
					
						
							|  |  |  | 		script_binding.type_name = NATIVE_GDMONOCLASS_NAME(klass); | 
					
						
							|  |  |  | 		script_binding.wrapper_class = klass; | 
					
						
							| 
									
										
										
										
											2021-06-04 18:03:15 +02:00
										 |  |  | 		script_binding.gchandle = rc ? MonoGCHandleData::new_weak_handle(managed) : MonoGCHandleData::new_strong_handle(managed); | 
					
						
							| 
									
										
										
										
											2019-07-03 09:44:53 +02:00
										 |  |  | 		script_binding.owner = unmanaged; | 
					
						
							| 
									
										
										
										
											2019-02-04 20:39:02 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-04 18:03:15 +02:00
										 |  |  | 		if (rc) { | 
					
						
							| 
									
										
										
										
											2019-02-04 20:39:02 +01:00
										 |  |  | 			// Unsafe refcount increment. The managed instance also counts as a reference.
 | 
					
						
							|  |  |  | 			// This way if the unmanaged world has no references to our owner
 | 
					
						
							|  |  |  | 			// but the managed instance is alive, the refcount will be 1 instead of 0.
 | 
					
						
							| 
									
										
										
										
											2021-06-13 11:15:44 +02:00
										 |  |  | 			// See: godot_icall_RefCounted_Dtor(MonoObject *p_obj, Object *p_ptr)
 | 
					
						
							| 
									
										
										
										
											2019-12-25 19:33:59 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			// May not me referenced yet, so we must use init_ref() instead of reference()
 | 
					
						
							| 
									
										
										
										
											2021-06-04 18:03:15 +02:00
										 |  |  | 			if (rc->init_ref()) { | 
					
						
							|  |  |  | 				CSharpLanguage::get_singleton()->post_unsafe_reference(rc); | 
					
						
							| 
									
										
										
										
											2020-01-13 21:00:07 +01:00
										 |  |  | 			} | 
					
						
							| 
									
										
										
										
											2019-02-04 20:39:02 +01:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		// The object was just created, no script instance binding should have been attached
 | 
					
						
							| 
									
										
										
										
											2021-08-16 17:16:36 +02:00
										 |  |  | 		CRASH_COND(CSharpLanguage::has_instance_binding(unmanaged)); | 
					
						
							| 
									
										
										
										
											2019-02-04 20:39:02 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-23 09:37:27 -04:00
										 |  |  | 		void *data; | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			MutexLock lock(CSharpLanguage::get_singleton()->get_language_bind_mutex()); | 
					
						
							|  |  |  | 			data = (void *)CSharpLanguage::get_singleton()->insert_script_binding(unmanaged, script_binding); | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2019-02-04 20:39:02 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		// Should be thread safe because the object was just created and nothing else should be referencing it
 | 
					
						
							| 
									
										
										
										
											2021-08-16 17:16:36 +02:00
										 |  |  | 		CSharpLanguage::set_instance_binding(unmanaged, data); | 
					
						
							| 
									
										
										
										
											2019-02-04 20:39:02 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		return; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-06-04 18:03:15 +02:00
										 |  |  | 	MonoGCHandleData gchandle = rc ? MonoGCHandleData::new_weak_handle(managed) : MonoGCHandleData::new_strong_handle(managed); | 
					
						
							| 
									
										
										
										
											2017-10-02 23:24:00 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-04 20:39:02 +01:00
										 |  |  | 	Ref<CSharpScript> script = CSharpScript::create_for_managed_type(klass, native); | 
					
						
							| 
									
										
										
										
											2017-10-02 23:24:00 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	CRASH_COND(script.is_null()); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-14 19:20:17 +01:00
										 |  |  | 	CSharpInstance *csharp_instance = CSharpInstance::create_for_managed_type(unmanaged, script.ptr(), gchandle); | 
					
						
							| 
									
										
										
										
											2017-10-02 23:24:00 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-03-14 19:20:17 +01:00
										 |  |  | 	unmanaged->set_script_and_instance(script, csharp_instance); | 
					
						
							| 
									
										
										
										
											2017-10-02 23:24:00 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2018-06-26 21:03:42 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | void unhandled_exception(MonoException *p_exc) { | 
					
						
							| 
									
										
										
										
											2020-06-25 23:28:51 +02:00
										 |  |  | 	mono_print_unhandled_exception((MonoObject *)p_exc); | 
					
						
							| 
									
										
										
										
											2021-02-18 16:18:33 +01:00
										 |  |  | 	gd_unhandled_exception_event(p_exc); | 
					
						
							| 
									
										
										
										
											2019-07-24 23:13:24 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	if (GDMono::get_singleton()->get_unhandled_exception_policy() == GDMono::POLICY_TERMINATE_APP) { | 
					
						
							|  |  |  | 		// Too bad 'mono_invoke_unhandled_exception_hook' is not exposed to embedders
 | 
					
						
							| 
									
										
										
										
											2021-02-18 16:18:33 +01:00
										 |  |  | 		mono_unhandled_exception((MonoObject *)p_exc); | 
					
						
							| 
									
										
										
										
											2020-04-02 01:20:12 +02:00
										 |  |  | 		GDMono::unhandled_exception_hook((MonoObject *)p_exc, nullptr); | 
					
						
							| 
									
										
										
										
											2019-07-24 23:13:24 +02:00
										 |  |  | 		GD_UNREACHABLE(); | 
					
						
							|  |  |  | 	} else { | 
					
						
							|  |  |  | #ifdef DEBUG_ENABLED
 | 
					
						
							| 
									
										
										
										
											2020-07-05 19:19:36 +02:00
										 |  |  | 		GDMonoUtils::debug_send_unhandled_exception_error(p_exc); | 
					
						
							|  |  |  | 		if (EngineDebugger::is_active()) { | 
					
						
							| 
									
										
										
										
											2020-02-27 03:30:20 +01:00
										 |  |  | 			EngineDebugger::get_singleton()->poll_events(false); | 
					
						
							| 
									
										
										
										
											2020-07-05 19:19:36 +02:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2019-07-24 23:13:24 +02:00
										 |  |  | #endif
 | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2018-06-26 21:03:42 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2021-02-18 16:18:33 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | void gd_unhandled_exception_event(MonoException *p_exc) { | 
					
						
							|  |  |  | 	MonoImage *mono_image = GDMono::get_singleton()->get_core_api_assembly()->get_image(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	MonoClass *gd_klass = mono_class_from_name(mono_image, "Godot", "GD"); | 
					
						
							|  |  |  | 	MonoMethod *unhandled_exception_method = mono_class_get_method_from_name(gd_klass, "OnUnhandledException", -1); | 
					
						
							|  |  |  | 	void *args[1]; | 
					
						
							|  |  |  | 	args[0] = p_exc; | 
					
						
							|  |  |  | 	mono_runtime_invoke(unhandled_exception_method, nullptr, (void **)args, nullptr); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2017-12-06 21:36:34 +01:00
										 |  |  | } // namespace GDMonoInternals
 |