// // System.Reflection.Emit.DynamicMethod.cs // // Author: // Paolo Molaro (lupus@ximian.com) // Zoltan Varga (vargaz@freemail.hu) // // (C) 2003 Ximian, Inc. http://www.ximian.com // // // Copyright (C) 2004 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. // #if !FULL_AOT_RUNTIME using System; using System.Reflection; using System.Reflection.Emit; using System.Globalization; using System.Runtime.CompilerServices; using System.Runtime.InteropServices; namespace System.Reflection.Emit { [ComVisible (true)] [StructLayout (LayoutKind.Sequential)] public sealed class DynamicMethod : MethodInfo { #pragma warning disable 169, 414, 649 #region Sync with reflection.h private RuntimeMethodHandle mhandle; private string name; private Type returnType; private Type[] parameters; private MethodAttributes attributes; private CallingConventions callingConvention; private Module module; private bool skipVisibility; private bool init_locals = true; private ILGenerator ilgen; private int nrefs; private object[] refs; private IntPtr referenced_by; private Type owner; #endregion #pragma warning restore 169, 414, 649 private Delegate deleg; private MonoMethod method; private ParameterBuilder[] pinfo; internal bool creating; private DynamicILInfo il_info; public DynamicMethod (string name, Type returnType, Type[] parameterTypes, Module m) : this (name, returnType, parameterTypes, m, false) { } public DynamicMethod (string name, Type returnType, Type[] parameterTypes, Type owner) : this (name, returnType, parameterTypes, owner, false) { } public DynamicMethod (string name, Type returnType, Type[] parameterTypes, Module m, bool skipVisibility) : this (name, MethodAttributes.Public | MethodAttributes.Static, CallingConventions.Standard, returnType, parameterTypes, m, skipVisibility) { } public DynamicMethod (string name, Type returnType, Type[] parameterTypes, Type owner, bool skipVisibility) : this (name, MethodAttributes.Public | MethodAttributes.Static, CallingConventions.Standard, returnType, parameterTypes, owner, skipVisibility) { } public DynamicMethod (string name, MethodAttributes attributes, CallingConventions callingConvention, Type returnType, Type[] parameterTypes, Type owner, bool skipVisibility) : this (name, attributes, callingConvention, returnType, parameterTypes, owner, owner.Module, skipVisibility, false) { } public DynamicMethod (string name, MethodAttributes attributes, CallingConventions callingConvention, Type returnType, Type[] parameterTypes, Module m, bool skipVisibility) : this (name, attributes, callingConvention, returnType, parameterTypes, null, m, skipVisibility, false) { } public DynamicMethod (string name, Type returnType, Type[] parameterTypes) : this (name, returnType, parameterTypes, false) { } [MonoTODO ("Visibility is not restricted")] public DynamicMethod (string name, Type returnType, Type[] parameterTypes, bool restrictedSkipVisibility) : this (name, MethodAttributes.Public | MethodAttributes.Static, CallingConventions.Standard, returnType, parameterTypes, null, null, restrictedSkipVisibility, true) { } DynamicMethod (string name, MethodAttributes attributes, CallingConventions callingConvention, Type returnType, Type [] parameterTypes, Type owner, Module m, bool skipVisibility, bool anonHosted) { if (name == null) throw new ArgumentNullException ("name"); if (returnType == null) returnType = typeof (void); if ((m == null) && !anonHosted) throw new ArgumentNullException ("m"); if (returnType.IsByRef) throw new ArgumentException ("Return type can't be a byref type", "returnType"); if (parameterTypes != null) { for (int i = 0; i < parameterTypes.Length; ++i) if (parameterTypes [i] == null) throw new ArgumentException ("Parameter " + i + " is null", "parameterTypes"); } if (owner != null && (owner.IsArray || owner.IsInterface)) { throw new ArgumentException ("Owner can't be an array or an interface."); } if (m == null) m = AnonHostModuleHolder.AnonHostModule; this.name = name; this.attributes = attributes | MethodAttributes.Static; this.callingConvention = callingConvention; this.returnType = returnType; this.parameters = parameterTypes; this.owner = owner; this.module = m; this.skipVisibility = skipVisibility; } [MethodImplAttribute(MethodImplOptions.InternalCall)] private static extern void create_dynamic_method (DynamicMethod m); private void CreateDynMethod () { if (mhandle.Value == IntPtr.Zero) { if (ilgen == null || ilgen.ILOffset == 0) throw new InvalidOperationException ("Method '" + name + "' does not have a method body."); ilgen.label_fixup (this); // Have to create all DynamicMethods referenced by this one try { // Used to avoid cycles creating = true; if (refs != null) { for (int i = 0; i < refs.Length; ++i) { if (refs [i] is DynamicMethod) { DynamicMethod m = (DynamicMethod)refs [i]; if (!m.creating) m.CreateDynMethod (); } } } } finally { creating = false; } create_dynamic_method (this); } } [ComVisible (true)] sealed override public Delegate CreateDelegate (Type delegateType) { if (delegateType == null) throw new ArgumentNullException ("delegateType"); if (deleg != null) return deleg; CreateDynMethod (); deleg = Delegate.CreateDelegate (delegateType, null, this); return deleg; } [ComVisible (true)] sealed override public Delegate CreateDelegate (Type delegateType, object target) { if (delegateType == null) throw new ArgumentNullException ("delegateType"); CreateDynMethod (); /* Can't cache the delegate since it is different for each target */ return Delegate.CreateDelegate (delegateType, target, this); } public ParameterBuilder DefineParameter (int position, ParameterAttributes attributes, string parameterName) { // // Extension: Mono allows position == 0 for the return attribute // if ((position < 0) || (position > parameters.Length)) throw new ArgumentOutOfRangeException ("position"); RejectIfCreated (); ParameterBuilder pb = new ParameterBuilder (this, position, attributes, parameterName); if (pinfo == null) pinfo = new ParameterBuilder [parameters.Length + 1]; pinfo [position] = pb; return pb; } public override MethodInfo GetBaseDefinition () { return this; } public override object[] GetCustomAttributes (bool inherit) { // support for MethodImplAttribute PCA return new Object[] { new MethodImplAttribute(GetMethodImplementationFlags()) }; } public override object[] GetCustomAttributes (Type attributeType, bool inherit) { if (attributeType == null) throw new ArgumentNullException ("attributeType"); if (attributeType.IsAssignableFrom (typeof (MethodImplAttribute))) return new Object[] { new MethodImplAttribute (GetMethodImplementationFlags()) }; else return EmptyArray.Value; } public DynamicILInfo GetDynamicILInfo () { if (il_info == null) il_info = new DynamicILInfo (this); return il_info; } public ILGenerator GetILGenerator () { return GetILGenerator (64); } public ILGenerator GetILGenerator (int streamSize) { if (((GetMethodImplementationFlags () & MethodImplAttributes.CodeTypeMask) != MethodImplAttributes.IL) || ((GetMethodImplementationFlags () & MethodImplAttributes.ManagedMask) != MethodImplAttributes.Managed)) throw new InvalidOperationException ("Method body should not exist."); if (ilgen != null) return ilgen; ilgen = new ILGenerator (Module, new DynamicMethodTokenGenerator (this), streamSize); return ilgen; } public override MethodImplAttributes GetMethodImplementationFlags () { return MethodImplAttributes.IL | MethodImplAttributes.Managed | MethodImplAttributes.NoInlining; } public override ParameterInfo[] GetParameters () { return GetParametersInternal (); } internal override ParameterInfo[] GetParametersInternal () { if (parameters == null) return EmptyArray.Value; ParameterInfo[] retval = new ParameterInfo [parameters.Length]; for (int i = 0; i < parameters.Length; i++) { retval [i] = ParameterInfo.New (pinfo == null ? null : pinfo [i + 1], parameters [i], this, i + 1); } return retval; } internal override int GetParametersCount () { return parameters == null ? 0 : parameters.Length; } internal override Type GetParameterType (int pos) { return parameters [pos]; } /* public override object Invoke (object obj, object[] parameters) { CreateDynMethod (); if (method == null) method = new MonoMethod (mhandle); return method.Invoke (obj, parameters); } */ public override object Invoke (object obj, BindingFlags invokeAttr, Binder binder, object[] parameters, CultureInfo culture) { try { CreateDynMethod (); if (method == null) method = new MonoMethod (mhandle); return method.Invoke (obj, invokeAttr, binder, parameters, culture); } catch (MethodAccessException mae) { throw new TargetInvocationException ("Method cannot be invoked.", mae); } } public override bool IsDefined (Type attributeType, bool inherit) { if (attributeType == null) throw new ArgumentNullException ("attributeType"); if (attributeType.IsAssignableFrom (typeof (MethodImplAttribute))) return true; else return false; } public override string ToString () { string parms = String.Empty; ParameterInfo[] p = GetParametersInternal (); for (int i = 0; i < p.Length; ++i) { if (i > 0) parms = parms + ", "; parms = parms + p [i].ParameterType.Name; } return ReturnType.Name+" "+Name+"("+parms+")"; } public override MethodAttributes Attributes { get { return attributes; } } public override CallingConventions CallingConvention { get { return callingConvention; } } public override Type DeclaringType { get { return null; } } public bool InitLocals { get { return init_locals; } set { init_locals = value; } } public override RuntimeMethodHandle MethodHandle { get { return mhandle; } } public override Module Module { get { return module; } } public override string Name { get { return name; } } public override Type ReflectedType { get { return null; } } [MonoTODO("Not implemented")] public override ParameterInfo ReturnParameter { get { throw new NotImplementedException (); } } public override Type ReturnType { get { return returnType; } } [MonoTODO("Not implemented")] public override ICustomAttributeProvider ReturnTypeCustomAttributes { get { throw new NotImplementedException (); } } /* public override int MetadataToken { get { return 0; } } */ private void RejectIfCreated () { if (mhandle.Value != IntPtr.Zero) throw new InvalidOperationException ("Type definition of the method is complete."); } internal int AddRef (object reference) { if (refs == null) refs = new object [4]; if (nrefs >= refs.Length - 1) { object [] new_refs = new object [refs.Length * 2]; System.Array.Copy (refs, new_refs, refs.Length); refs = new_refs; } refs [nrefs] = reference; /* Reserved by the runtime */ refs [nrefs + 1] = null; nrefs += 2; return nrefs - 1; } // This class takes care of constructing the module in a thread safe manner class AnonHostModuleHolder { public static Module anon_host_module; static AnonHostModuleHolder () { AssemblyName aname = new AssemblyName (); aname.Name = "Anonymously Hosted DynamicMethods Assembly"; AssemblyBuilder ab = AppDomain.CurrentDomain.DefineDynamicAssembly (aname, AssemblyBuilderAccess.Run); anon_host_module = ab.GetManifestModule (); } public static Module AnonHostModule { get { return anon_host_module; } } } } internal class DynamicMethodTokenGenerator : TokenGenerator { private DynamicMethod m; public DynamicMethodTokenGenerator (DynamicMethod m) { this.m = m; } public int GetToken (string str) { return m.AddRef (str); } public int GetToken (MethodBase method, Type[] opt_param_types) { throw new InvalidOperationException (); } public int GetToken (MemberInfo member, bool create_open_instance) { return m.AddRef (member); } public int GetToken (SignatureHelper helper) { return m.AddRef (helper); } } } #endif