// // System.Reflection/MonoMethod.cs // The class used to represent methods from the mono runtime. // // Author: // Paolo Molaro (lupus@ximian.com) // // (C) 2001 Ximian, Inc. http://www.ximian.com // Copyright (C) 2004-2005 Novell, Inc (http://www.novell.com) // // 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. // using System.Globalization; using System.Runtime.CompilerServices; using System.Runtime.InteropServices; using System.Runtime.Serialization; using System.Reflection.Emit; using System.Security; using System.Threading; using System.Text; namespace System.Reflection { internal struct MonoMethodInfo { internal Type parent; internal Type ret; internal MethodAttributes attrs; internal MethodImplAttributes iattrs; internal CallingConventions callconv; [MethodImplAttribute(MethodImplOptions.InternalCall)] internal static extern void get_method_info (IntPtr handle, out MonoMethodInfo info); [MethodImplAttribute(MethodImplOptions.InternalCall)] internal static extern ParameterInfo[] get_parameter_info (IntPtr handle); [MethodImplAttribute(MethodImplOptions.InternalCall)] internal static extern UnmanagedMarshal get_retval_marshal (IntPtr handle); }; /* * Note: most of this class needs to be duplicated for the contructor, since * the .NET reflection class hierarchy is so broken. */ [Serializable()] internal class MonoMethod : MethodInfo, ISerializable { #pragma warning disable 649 internal IntPtr mhandle; string name; Type reftype; #pragma warning restore 649 internal MonoMethod () { } internal MonoMethod (RuntimeMethodHandle mhandle) { this.mhandle = mhandle.Value; } [MethodImplAttribute(MethodImplOptions.InternalCall)] internal static extern string get_name (MethodBase method); [MethodImplAttribute(MethodImplOptions.InternalCall)] internal static extern MonoMethod get_base_definition (MonoMethod method); public override MethodInfo GetBaseDefinition () { return get_base_definition (this); } #if NET_2_0 || BOOTSTRAP_NET_2_0 public override ParameterInfo ReturnParameter { get { return new ParameterInfo (ReturnType, this, MonoMethodInfo.get_retval_marshal (mhandle)); } } #endif public override Type ReturnType { get { MonoMethodInfo info; MonoMethodInfo.get_method_info (mhandle, out info); return info.ret; } } public override ICustomAttributeProvider ReturnTypeCustomAttributes { get { return new ParameterInfo (ReturnType, this, MonoMethodInfo.get_retval_marshal (mhandle)); } } public override MethodImplAttributes GetMethodImplementationFlags() { MonoMethodInfo info; MonoMethodInfo.get_method_info (mhandle, out info); return info.iattrs; } public override ParameterInfo[] GetParameters() { return MonoMethodInfo.get_parameter_info (mhandle); } /* * InternalInvoke() receives the parameters correctly converted by the * binder to match the types of the method signature. */ [MethodImplAttribute(MethodImplOptions.InternalCall)] internal extern Object InternalInvoke (Object obj, Object[] parameters, out Exception exc); public override Object Invoke (Object obj, BindingFlags invokeAttr, Binder binder, Object[] parameters, CultureInfo culture) { if (binder == null) binder = Binder.DefaultBinder; ParameterInfo[] pinfo = GetParameters (); if (!Binder.ConvertArgs (binder, parameters, pinfo, culture)) throw new ArgumentException ("parameters"); #if !NET_2_1 if (SecurityManager.SecurityEnabled) { // sadly Attributes doesn't tell us which kind of security action this is so // we must do it the hard way - and it also means that we can skip calling // Attribute (which is another an icall) SecurityManager.ReflectedLinkDemandInvoke (this); } #endif #if NET_2_0 if (ContainsGenericParameters) throw new InvalidOperationException ("Late bound operations cannot be performed on types or methods for which ContainsGenericParameters is true."); #endif Exception exc; object o = null; try { // The ex argument is used to distinguish exceptions thrown by the icall // from the exceptions thrown by the called method (which need to be // wrapped in TargetInvocationException). o = InternalInvoke (obj, parameters, out exc); #if NET_2_0 } catch (ThreadAbortException) { throw; #endif #if NET_2_1 } catch (MethodAccessException) { throw; #endif } catch (Exception e) { throw new TargetInvocationException (e); } if (exc != null) throw exc; return o; } public override RuntimeMethodHandle MethodHandle { get {return new RuntimeMethodHandle (mhandle);} } public override MethodAttributes Attributes { get { MonoMethodInfo info; MonoMethodInfo.get_method_info (mhandle, out info); return info.attrs; } } public override CallingConventions CallingConvention { get { MonoMethodInfo info; MonoMethodInfo.get_method_info (mhandle, out info); return info.callconv; } } public override Type ReflectedType { get { return reftype; } } public override Type DeclaringType { get { MonoMethodInfo info; MonoMethodInfo.get_method_info (mhandle, out info); return info.parent; } } public override string Name { get { if (name != null) return name; return get_name (this); } } public override bool IsDefined (Type attributeType, bool inherit) { return MonoCustomAttrs.IsDefined (this, attributeType, inherit); } public override object[] GetCustomAttributes( bool inherit) { return MonoCustomAttrs.GetCustomAttributes (this, inherit); } public override object[] GetCustomAttributes( Type attributeType, bool inherit) { return MonoCustomAttrs.GetCustomAttributes (this, attributeType, inherit); } [MethodImplAttribute(MethodImplOptions.InternalCall)] internal static extern DllImportAttribute GetDllImportAttribute (IntPtr mhandle); internal object[] GetPseudoCustomAttributes () { int count = 0; /* MS.NET doesn't report MethodImplAttribute */ MonoMethodInfo info; MonoMethodInfo.get_method_info (mhandle, out info); if ((info.iattrs & MethodImplAttributes.PreserveSig) != 0) count ++; if ((info.attrs & MethodAttributes.PinvokeImpl) != 0) count ++; if (count == 0) return null; object[] attrs = new object [count]; count = 0; if ((info.iattrs & MethodImplAttributes.PreserveSig) != 0) attrs [count ++] = new PreserveSigAttribute (); if ((info.attrs & MethodAttributes.PinvokeImpl) != 0) { DllImportAttribute attr = GetDllImportAttribute (mhandle); if ((info.iattrs & MethodImplAttributes.PreserveSig) != 0) attr.PreserveSig = true; attrs [count ++] = attr; } return attrs; } public override string ToString () { StringBuilder sb = new StringBuilder (); if (ReturnType.IsClass) sb.Append (ReturnType.ToString ()); else sb.Append (ReturnType.Name); sb.Append (" "); sb.Append (Name); #if NET_2_0 || BOOTSTRAP_NET_2_0 if (IsGenericMethod) { Type[] gen_params = GetGenericArguments (); sb.Append ("["); for (int j = 0; j < gen_params.Length; j++) { if (j > 0) sb.Append (","); sb.Append (gen_params [j].Name); } sb.Append ("]"); } #endif sb.Append ("("); ParameterInfo[] p = GetParameters (); for (int i = 0; i < p.Length; ++i) { if (i > 0) sb.Append (", "); Type pt = p[i].ParameterType; bool byref = pt.IsByRef; if (byref) pt = pt.GetElementType (); if (pt.IsClass) sb.Append (pt.ToString ()); else sb.Append (pt.Name); if (byref) sb.Append (" ByRef"); } if ((CallingConvention & CallingConventions.VarArgs) != 0) { if (p.Length > 0) sb.Append (", "); sb.Append ("..."); } sb.Append (")"); return sb.ToString (); } // ISerializable public void GetObjectData(SerializationInfo info, StreamingContext context) { #if NET_2_0 Type[] genericArguments = IsGenericMethod && !IsGenericMethodDefinition ? GetGenericArguments () : null; MemberInfoSerializationHolder.Serialize ( info, Name, ReflectedType, ToString(), MemberTypes.Method, genericArguments); #else MemberInfoSerializationHolder.Serialize ( info, Name, ReflectedType, ToString(), MemberTypes.Method); #endif } #if NET_2_0 || BOOTSTRAP_NET_2_0 public override MethodInfo MakeGenericMethod (Type [] methodInstantiation) { if (methodInstantiation == null) throw new ArgumentNullException ("methodInstantiation"); foreach (Type type in methodInstantiation) if (type == null) throw new ArgumentNullException (); MethodInfo ret = MakeGenericMethod_impl (methodInstantiation); if (ret == null) throw new ArgumentException (String.Format ("The method has {0} generic parameter(s) but {1} generic argument(s) were provided.", GetGenericArguments ().Length, methodInstantiation.Length)); return ret; } [MethodImplAttribute(MethodImplOptions.InternalCall)] extern MethodInfo MakeGenericMethod_impl (Type [] types); [MethodImplAttribute(MethodImplOptions.InternalCall)] public override extern Type [] GetGenericArguments (); [MethodImplAttribute(MethodImplOptions.InternalCall)] extern MethodInfo GetGenericMethodDefinition_impl (); public override MethodInfo GetGenericMethodDefinition () { MethodInfo res = GetGenericMethodDefinition_impl (); if (res == null) throw new InvalidOperationException (); return res; } public override extern bool IsGenericMethodDefinition { [MethodImplAttribute(MethodImplOptions.InternalCall)] get; } public override extern bool IsGenericMethod { [MethodImplAttribute(MethodImplOptions.InternalCall)] get; } public override bool ContainsGenericParameters { get { if (IsGenericMethod) { foreach (Type arg in GetGenericArguments ()) if (arg.ContainsGenericParameters) return true; } return DeclaringType.ContainsGenericParameters; } } #endif #if NET_2_0 public override MethodBody GetMethodBody () { return GetMethodBody (mhandle); } #endif } internal class MonoCMethod : ConstructorInfo, ISerializable { #pragma warning disable 649 internal IntPtr mhandle; string name; Type reftype; #pragma warning restore 649 public override MethodImplAttributes GetMethodImplementationFlags() { MonoMethodInfo info; MonoMethodInfo.get_method_info (mhandle, out info); return info.iattrs; } public override ParameterInfo[] GetParameters() { return MonoMethodInfo.get_parameter_info (mhandle); } /* * InternalInvoke() receives the parameters corretcly converted by the binder * to match the types of the method signature. */ [MethodImplAttribute(MethodImplOptions.InternalCall)] internal extern Object InternalInvoke (Object obj, Object[] parameters, out Exception exc); public override Object Invoke (Object obj, BindingFlags invokeAttr, Binder binder, Object[] parameters, CultureInfo culture) { if (binder == null) binder = Binder.DefaultBinder; ParameterInfo[] pinfo = GetParameters (); if (!Binder.ConvertArgs (binder, parameters, pinfo, culture)) throw new ArgumentException ("parameters"); if (SecurityManager.SecurityEnabled) { // sadly Attributes doesn't tell us which kind of security action this is so // we must do it the hard way - and it also means that we can skip calling // Attribute (which is another an icall) SecurityManager.ReflectedLinkDemandInvoke (this); } #if NET_2_0 if (obj == null && DeclaringType.ContainsGenericParameters) throw new MemberAccessException ("Cannot create an instance of " + DeclaringType + " because Type.ContainsGenericParameters is true."); #endif Exception exc = null; object o = null; try { o = InternalInvoke (obj, parameters, out exc); #if NET_2_1 } catch (MethodAccessException) { throw; #endif } catch (Exception e) { throw new TargetInvocationException (e); } if (exc != null) throw exc; return o; } public override Object Invoke (BindingFlags invokeAttr, Binder binder, Object[] parameters, CultureInfo culture) { return Invoke (null, invokeAttr, binder, parameters, culture); } public override RuntimeMethodHandle MethodHandle { get {return new RuntimeMethodHandle (mhandle);} } public override MethodAttributes Attributes { get { MonoMethodInfo info; MonoMethodInfo.get_method_info (mhandle, out info); return info.attrs; } } public override CallingConventions CallingConvention { get { MonoMethodInfo info; MonoMethodInfo.get_method_info (mhandle, out info); return info.callconv; } } public override Type ReflectedType { get { return reftype; } } public override Type DeclaringType { get { MonoMethodInfo info; MonoMethodInfo.get_method_info (mhandle, out info); return info.parent; } } public override string Name { get { if (name != null) return name; return MonoMethod.get_name (this); } } public override bool IsDefined (Type attributeType, bool inherit) { return MonoCustomAttrs.IsDefined (this, attributeType, inherit); } public override object[] GetCustomAttributes( bool inherit) { return MonoCustomAttrs.GetCustomAttributes (this, inherit); } public override object[] GetCustomAttributes( Type attributeType, bool inherit) { return MonoCustomAttrs.GetCustomAttributes (this, attributeType, inherit); } #if NET_2_0 public override MethodBody GetMethodBody () { return GetMethodBody (mhandle); } #endif public override string ToString () { StringBuilder sb = new StringBuilder (); sb.Append ("Void "); sb.Append (Name); sb.Append ("("); ParameterInfo[] p = GetParameters (); for (int i = 0; i < p.Length; ++i) { if (i > 0) sb.Append (", "); sb.Append (p[i].ParameterType.Name); } sb.Append (")"); return sb.ToString (); } // ISerializable public void GetObjectData(SerializationInfo info, StreamingContext context) { MemberInfoSerializationHolder.Serialize ( info, Name, ReflectedType, ToString(), MemberTypes.Constructor); } } }