| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291 | /*************************************************************************//*  gd_mono_method.cpp                                                   *//*************************************************************************//*                       This file is part of:                           *//*                           GODOT ENGINE                                *//*                      https://godotengine.org                          *//*************************************************************************//* Copyright (c) 2007-2019 Juan Linietsky, Ariel Manzur.                 *//* Copyright (c) 2014-2019 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 "gd_mono_method.h"#include "gd_mono_class.h"#include "gd_mono_marshal.h"#include <mono/metadata/attrdefs.h>void GDMonoMethod::_update_signature() {	// Apparently MonoMethodSignature needs not to be freed.	// mono_method_signature caches the result, we don't need to cache it ourselves.	MonoMethodSignature *method_sig = mono_method_signature(mono_method);	_update_signature(method_sig);}void GDMonoMethod::_update_signature(MonoMethodSignature *p_method_sig) {	params_count = mono_signature_get_param_count(p_method_sig);	MonoType *ret_type = mono_signature_get_return_type(p_method_sig);	if (ret_type) {		return_type.type_encoding = mono_type_get_type(ret_type);		if (return_type.type_encoding != MONO_TYPE_VOID) {			MonoClass *ret_type_class = mono_class_from_mono_type(ret_type);			return_type.type_class = GDMono::get_singleton()->get_class(ret_type_class);		}	}	void *iter = NULL;	MonoType *param_raw_type;	while ((param_raw_type = mono_signature_get_params(p_method_sig, &iter)) != NULL) {		ManagedType param_type;		param_type.type_encoding = mono_type_get_type(param_raw_type);		MonoClass *param_type_class = mono_class_from_mono_type(param_raw_type);		param_type.type_class = GDMono::get_singleton()->get_class(param_type_class);		param_types.push_back(param_type);	}	// clear the cache	method_info_fetched = false;	method_info = MethodInfo();}bool GDMonoMethod::is_static() {	return mono_method_get_flags(mono_method, NULL) & MONO_METHOD_ATTR_STATIC;}IMonoClassMember::Visibility GDMonoMethod::get_visibility() {	switch (mono_method_get_flags(mono_method, NULL) & MONO_METHOD_ATTR_ACCESS_MASK) {		case MONO_METHOD_ATTR_PRIVATE:			return IMonoClassMember::PRIVATE;		case MONO_METHOD_ATTR_FAM_AND_ASSEM:			return IMonoClassMember::PROTECTED_AND_INTERNAL;		case MONO_METHOD_ATTR_ASSEM:			return IMonoClassMember::INTERNAL;		case MONO_METHOD_ATTR_FAMILY:			return IMonoClassMember::PROTECTED;		case MONO_METHOD_ATTR_PUBLIC:			return IMonoClassMember::PUBLIC;		default:			ERR_FAIL_V(IMonoClassMember::PRIVATE);	}}void *GDMonoMethod::get_thunk() {	return mono_method_get_unmanaged_thunk(mono_method);}MonoObject *GDMonoMethod::invoke(MonoObject *p_object, const Variant **p_params, MonoException **r_exc) {	if (get_return_type().type_encoding != MONO_TYPE_VOID || get_parameters_count() > 0) {		MonoArray *params = mono_array_new(mono_domain_get(), CACHED_CLASS_RAW(MonoObject), get_parameters_count());		for (int i = 0; i < params_count; i++) {			MonoObject *boxed_param = GDMonoMarshal::variant_to_mono_object(p_params[i], param_types[i]);			mono_array_set(params, MonoObject *, i, boxed_param);		}		MonoException *exc = NULL;		MonoObject *ret = GDMonoUtils::runtime_invoke_array(mono_method, p_object, params, &exc);		if (exc) {			ret = NULL;			if (r_exc) {				*r_exc = exc;			} else {				GDMonoUtils::set_pending_exception(exc);			}		}		return ret;	} else {		MonoException *exc = NULL;		GDMonoUtils::runtime_invoke(mono_method, p_object, NULL, &exc);		if (exc) {			if (r_exc) {				*r_exc = exc;			} else {				GDMonoUtils::set_pending_exception(exc);			}		}		return NULL;	}}MonoObject *GDMonoMethod::invoke(MonoObject *p_object, MonoException **r_exc) {	ERR_FAIL_COND_V(get_parameters_count() > 0, NULL);	return invoke_raw(p_object, NULL, r_exc);}MonoObject *GDMonoMethod::invoke_raw(MonoObject *p_object, void **p_params, MonoException **r_exc) {	MonoException *exc = NULL;	MonoObject *ret = GDMonoUtils::runtime_invoke(mono_method, p_object, p_params, &exc);	if (exc) {		ret = NULL;		if (r_exc) {			*r_exc = exc;		} else {			GDMonoUtils::set_pending_exception(exc);		}	}	return ret;}bool GDMonoMethod::has_attribute(GDMonoClass *p_attr_class) {	ERR_FAIL_NULL_V(p_attr_class, false);	if (!attrs_fetched)		fetch_attributes();	if (!attributes)		return false;	return mono_custom_attrs_has_attr(attributes, p_attr_class->get_mono_ptr());}MonoObject *GDMonoMethod::get_attribute(GDMonoClass *p_attr_class) {	ERR_FAIL_NULL_V(p_attr_class, NULL);	if (!attrs_fetched)		fetch_attributes();	if (!attributes)		return NULL;	return mono_custom_attrs_get_attr(attributes, p_attr_class->get_mono_ptr());}void GDMonoMethod::fetch_attributes() {	ERR_FAIL_COND(attributes != NULL);	attributes = mono_custom_attrs_from_method(mono_method);	attrs_fetched = true;}String GDMonoMethod::get_full_name(bool p_signature) const {	char *res = mono_method_full_name(mono_method, p_signature);	String full_name(res);	mono_free(res);	return full_name;}String GDMonoMethod::get_full_name_no_class() const {	String res;	MonoMethodSignature *method_sig = mono_method_signature(mono_method);	char *ret_str = mono_type_full_name(mono_signature_get_return_type(method_sig));	res += ret_str;	mono_free(ret_str);	res += " ";	res += name;	res += "(";	char *sig_desc = mono_signature_get_desc(method_sig, true);	res += sig_desc;	mono_free(sig_desc);	res += ")";	return res;}String GDMonoMethod::get_ret_type_full_name() const {	MonoMethodSignature *method_sig = mono_method_signature(mono_method);	char *ret_str = mono_type_full_name(mono_signature_get_return_type(method_sig));	String res = ret_str;	mono_free(ret_str);	return res;}String GDMonoMethod::get_signature_desc(bool p_namespaces) const {	MonoMethodSignature *method_sig = mono_method_signature(mono_method);	char *sig_desc = mono_signature_get_desc(method_sig, p_namespaces);	String res = sig_desc;	mono_free(sig_desc);	return res;}void GDMonoMethod::get_parameter_names(Vector<StringName> &names) const {	if (params_count > 0) {		const char **_names = memnew_arr(const char *, params_count);		mono_method_get_param_names(mono_method, _names);		for (int i = 0; i < params_count; ++i) {			names.push_back(StringName(_names[i]));		}		memdelete_arr(_names);	}}void GDMonoMethod::get_parameter_types(Vector<ManagedType> &types) const {	for (int i = 0; i < param_types.size(); ++i) {		types.push_back(param_types[i]);	}}const MethodInfo &GDMonoMethod::get_method_info() {	if (!method_info_fetched) {		method_info.name = name;		method_info.return_val = PropertyInfo(GDMonoMarshal::managed_to_variant_type(return_type), "");		Vector<StringName> names;		get_parameter_names(names);		for (int i = 0; i < params_count; ++i) {			method_info.arguments.push_back(PropertyInfo(GDMonoMarshal::managed_to_variant_type(param_types[i]), names[i]));		}		// TODO: default arguments		method_info_fetched = true;	}	return method_info;}GDMonoMethod::GDMonoMethod(StringName p_name, MonoMethod *p_method) {	name = p_name;	mono_method = p_method;	method_info_fetched = false;	attrs_fetched = false;	attributes = NULL;	_update_signature();}GDMonoMethod::~GDMonoMethod() {	if (attributes) {		mono_custom_attrs_free(attributes);	}}
 |