/* Copyright (C) 2008-2012 Jeroen Frijters This software is provided 'as-is', without any express or implied warranty. In no event will the authors be held liable for any damages arising from the use of this software. Permission is granted to anyone to use this software for any purpose, including commercial applications, and to alter it and redistribute it freely, subject to the following restrictions: 1. The origin of this software must not be misrepresented; you must not claim that you wrote the original software. If you use this software in a product, an acknowledgment in the product documentation would be appreciated but is not required. 2. Altered source versions must be plainly marked as such, and must not be misrepresented as being the original software. 3. This notice may not be removed or altered from any source distribution. Jeroen Frijters jeroen@frijters.net */ using System; using System.Collections.Generic; using System.IO; using System.Diagnostics; using System.Diagnostics.SymbolStore; using System.Security.Cryptography; using System.Resources; using System.Runtime.CompilerServices; using System.Runtime.InteropServices; using IKVM.Reflection.Impl; using IKVM.Reflection.Metadata; using IKVM.Reflection.Writer; namespace IKVM.Reflection.Emit { public sealed class ModuleBuilder : Module, ITypeOwner { private static readonly bool usePublicKeyAssemblyReference = false; private Guid mvid = Guid.NewGuid(); private long imageBaseAddress = 0x00400000; private long stackReserve = -1; private int fileAlignment = 0x200; private DllCharacteristics dllCharacteristics = DllCharacteristics.DynamicBase | DllCharacteristics.NoSEH | DllCharacteristics.NXCompat | DllCharacteristics.TerminalServerAware; private readonly AssemblyBuilder asm; internal readonly string moduleName; internal readonly string fileName; internal readonly ISymbolWriterImpl symbolWriter; private readonly TypeBuilder moduleType; private readonly List types = new List(); private readonly Dictionary typeTokens = new Dictionary(); private readonly Dictionary memberRefTypeTokens = new Dictionary(); internal readonly ByteBuffer methodBodies = new ByteBuffer(128 * 1024); internal readonly List tokenFixupOffsets = new List(); internal readonly ByteBuffer initializedData = new ByteBuffer(512); internal readonly ByteBuffer manifestResources = new ByteBuffer(512); internal ResourceSection unmanagedResources; private readonly Dictionary importedMemberRefs = new Dictionary(); private readonly Dictionary importedMethodSpecs = new Dictionary(); private readonly Dictionary referencedAssemblies = new Dictionary(); private List referencedAssemblyNames; private int nextPseudoToken = -1; private readonly List resolvedTokens = new List(); internal readonly TableHeap Tables = new TableHeap(); internal readonly StringHeap Strings = new StringHeap(); internal readonly UserStringHeap UserStrings = new UserStringHeap(); internal readonly GuidHeap Guids = new GuidHeap(); internal readonly BlobHeap Blobs = new BlobHeap(); internal readonly List vtablefixups = new List(); internal readonly List unmanagedExports = new List(); private List interfaceImplCustomAttributes; private List resourceWriters; private bool saved; private struct ResourceWriterRecord { private readonly string name; private readonly ResourceWriter rw; private readonly MemoryStream mem; private readonly ResourceAttributes attributes; internal ResourceWriterRecord(string name, ResourceWriter rw, MemoryStream mem, ResourceAttributes attributes) { this.name = name; this.rw = rw; this.mem = mem; this.attributes = attributes; } internal void Emit(ModuleBuilder mb) { rw.Generate(); mem.Position = 0; mb.DefineManifestResource(name, mem, attributes); rw.Close(); } } internal struct VTableFixups { internal uint initializedDataOffset; internal ushort count; internal ushort type; internal int SlotWidth { get { return (type & 0x02) == 0 ? 4 : 8; } } } struct InterfaceImplCustomAttribute { internal int type; internal int interfaceType; internal int pseudoToken; } struct MemberRefKey : IEquatable { private readonly Type type; private readonly string name; private readonly Signature signature; internal MemberRefKey(Type type, string name, Signature signature) { this.type = type; this.name = name; this.signature = signature; } public bool Equals(MemberRefKey other) { return other.type.Equals(type) && other.name == name && other.signature.Equals(signature); } public override bool Equals(object obj) { MemberRefKey? other = obj as MemberRefKey?; return other != null && Equals(other); } public override int GetHashCode() { return type.GetHashCode() + name.GetHashCode() + signature.GetHashCode(); } internal MethodBase LookupMethod() { return type.FindMethod(name, (MethodSignature)signature); } } struct MethodSpecKey : IEquatable { private readonly Type type; private readonly string name; private readonly MethodSignature signature; private readonly Type[] genericParameters; internal MethodSpecKey(Type type, string name, MethodSignature signature, Type[] genericParameters) { this.type = type; this.name = name; this.signature = signature; this.genericParameters = genericParameters; } public bool Equals(MethodSpecKey other) { return other.type.Equals(type) && other.name == name && other.signature.Equals(signature) && Util.ArrayEquals(other.genericParameters, genericParameters); } public override bool Equals(object obj) { MethodSpecKey? other = obj as MethodSpecKey?; return other != null && Equals(other); } public override int GetHashCode() { return type.GetHashCode() + name.GetHashCode() + signature.GetHashCode() + Util.GetHashCode(genericParameters); } } internal ModuleBuilder(AssemblyBuilder asm, string moduleName, string fileName, bool emitSymbolInfo) : base(asm.universe) { this.asm = asm; this.moduleName = moduleName; this.fileName = fileName; if (emitSymbolInfo) { symbolWriter = SymbolSupport.CreateSymbolWriterFor(this); } // must be the first record in the TypeDef table moduleType = new TypeBuilder(this, null, ""); types.Add(moduleType); } internal void PopulatePropertyAndEventTables() { // LAMESPEC the PropertyMap and EventMap tables are not required to be sorted by the CLI spec, // but .NET sorts them and Mono requires them to be sorted, so we have to populate the // tables in the right order foreach (TypeBuilder type in types) { type.PopulatePropertyAndEventTables(); } } internal void WriteTypeDefTable(MetadataWriter mw) { int fieldList = 1; int methodList = 1; foreach (TypeBuilder type in types) { type.WriteTypeDefRecord(mw, ref fieldList, ref methodList); } } internal void WriteMethodDefTable(int baseRVA, MetadataWriter mw) { int paramList = 1; foreach (TypeBuilder type in types) { type.WriteMethodDefRecords(baseRVA, mw, ref paramList); } } internal void WriteParamTable(MetadataWriter mw) { foreach (TypeBuilder type in types) { type.WriteParamRecords(mw); } } internal void WriteFieldTable(MetadataWriter mw) { foreach (TypeBuilder type in types) { type.WriteFieldRecords(mw); } } internal int AllocPseudoToken() { return nextPseudoToken--; } public TypeBuilder DefineType(string name) { return DefineType(name, TypeAttributes.Class); } public TypeBuilder DefineType(string name, TypeAttributes attr) { return DefineType(name, attr, null); } public TypeBuilder DefineType(string name, TypeAttributes attr, Type parent) { return DefineType(name, attr, parent, PackingSize.Unspecified, 0); } public TypeBuilder DefineType(string name, TypeAttributes attr, Type parent, int typesize) { return DefineType(name, attr, parent, PackingSize.Unspecified, typesize); } public TypeBuilder DefineType(string name, TypeAttributes attr, Type parent, PackingSize packsize) { return DefineType(name, attr, parent, packsize, 0); } public TypeBuilder DefineType(string name, TypeAttributes attr, Type parent, Type[] interfaces) { TypeBuilder tb = DefineType(name, attr, parent); foreach (Type iface in interfaces) { tb.AddInterfaceImplementation(iface); } return tb; } public TypeBuilder DefineType(string name, TypeAttributes attr, Type parent, PackingSize packingSize, int typesize) { string ns = null; int lastdot = name.LastIndexOf('.'); if (lastdot > 0) { ns = name.Substring(0, lastdot); name = name.Substring(lastdot + 1); } TypeBuilder typeBuilder = __DefineType(ns, name); typeBuilder.__SetAttributes(attr); typeBuilder.SetParent(parent); if (packingSize != PackingSize.Unspecified || typesize != 0) { typeBuilder.__SetLayout((int)packingSize, typesize); } return typeBuilder; } public TypeBuilder __DefineType(string ns, string name) { return DefineType(this, ns, name); } internal TypeBuilder DefineType(ITypeOwner owner, string ns, string name) { TypeBuilder typeBuilder = new TypeBuilder(owner, ns, name); types.Add(typeBuilder); return typeBuilder; } public EnumBuilder DefineEnum(string name, TypeAttributes visibility, Type underlyingType) { TypeBuilder tb = DefineType(name, (visibility & TypeAttributes.VisibilityMask) | TypeAttributes.Sealed, universe.System_Enum); FieldBuilder fb = tb.DefineField("value__", underlyingType, FieldAttributes.Public | FieldAttributes.SpecialName | FieldAttributes.RTSpecialName); return new EnumBuilder(tb, fb); } public FieldBuilder __DefineField(string name, Type type, CustomModifiers customModifiers, FieldAttributes attributes) { return moduleType.__DefineField(name, type, customModifiers, attributes); } [Obsolete("Please use __DefineField(string, Type, CustomModifiers, FieldAttributes) instead.")] public FieldBuilder __DefineField(string name, Type type, Type[] requiredCustomModifiers, Type[] optionalCustomModifiers, FieldAttributes attributes) { return moduleType.DefineField(name, type, requiredCustomModifiers, optionalCustomModifiers, attributes); } public ConstructorBuilder __DefineModuleInitializer(MethodAttributes visibility) { return moduleType.DefineConstructor(visibility | MethodAttributes.Static | MethodAttributes.SpecialName | MethodAttributes.RTSpecialName, CallingConventions.Standard, Type.EmptyTypes); } public FieldBuilder DefineUninitializedData(string name, int size, FieldAttributes attributes) { return moduleType.DefineUninitializedData(name, size, attributes); } public FieldBuilder DefineInitializedData(string name, byte[] data, FieldAttributes attributes) { return moduleType.DefineInitializedData(name, data, attributes); } public MethodBuilder DefineGlobalMethod(string name, MethodAttributes attributes, Type returnType, Type[] parameterTypes) { return moduleType.DefineMethod(name, attributes, returnType, parameterTypes); } public MethodBuilder DefineGlobalMethod(string name, MethodAttributes attributes, CallingConventions callingConvention, Type returnType, Type[] parameterTypes) { return moduleType.DefineMethod(name, attributes, callingConvention, returnType, parameterTypes); } public MethodBuilder DefineGlobalMethod(string name, MethodAttributes attributes, CallingConventions callingConvention, Type returnType, Type[] requiredReturnTypeCustomModifiers, Type[] optionalReturnTypeCustomModifiers, Type[] parameterTypes, Type[][] requiredParameterTypeCustomModifiers, Type[][] optionalParameterTypeCustomModifiers) { return moduleType.DefineMethod(name, attributes, callingConvention, returnType, requiredReturnTypeCustomModifiers, optionalReturnTypeCustomModifiers, parameterTypes, requiredParameterTypeCustomModifiers, optionalParameterTypeCustomModifiers); } public MethodBuilder DefinePInvokeMethod(string name, string dllName, MethodAttributes attributes, CallingConventions callingConvention, Type returnType, Type[] parameterTypes, CallingConvention nativeCallConv, CharSet nativeCharSet) { return moduleType.DefinePInvokeMethod(name, dllName, attributes, callingConvention, returnType, parameterTypes, nativeCallConv, nativeCharSet); } public MethodBuilder DefinePInvokeMethod(string name, string dllName, string entryName, MethodAttributes attributes, CallingConventions callingConvention, Type returnType, Type[] parameterTypes, CallingConvention nativeCallConv, CharSet nativeCharSet) { return moduleType.DefinePInvokeMethod(name, dllName, entryName, attributes, callingConvention, returnType, parameterTypes, nativeCallConv, nativeCharSet); } public void CreateGlobalFunctions() { moduleType.CreateType(); } internal void AddTypeForwarder(Type type) { ExportType(type); if (!type.__IsMissing) { foreach (Type nested in type.GetNestedTypes(BindingFlags.Public | BindingFlags.NonPublic)) { // we export all nested types (i.e. even the private ones) // (this behavior is the same as the C# compiler) AddTypeForwarder(nested); } } } private int ExportType(Type type) { ExportedTypeTable.Record rec = new ExportedTypeTable.Record(); rec.TypeDefId = type.MetadataToken; rec.TypeName = this.Strings.Add(type.__Name); string ns = type.__Namespace; rec.TypeNamespace = ns == null ? 0 : this.Strings.Add(ns); if (type.IsNested) { rec.Flags = 0; rec.Implementation = ExportType(type.DeclaringType); } else { rec.Flags = 0x00200000; // CorTypeAttr.tdForwarder rec.Implementation = ImportAssemblyRef(type.Assembly); } return 0x27000000 | this.ExportedType.FindOrAddRecord(rec); } public void SetCustomAttribute(ConstructorInfo con, byte[] binaryAttribute) { SetCustomAttribute(new CustomAttributeBuilder(con, binaryAttribute)); } public void SetCustomAttribute(CustomAttributeBuilder customBuilder) { SetCustomAttribute(0x00000001, customBuilder); } internal void SetCustomAttribute(int token, CustomAttributeBuilder customBuilder) { Debug.Assert(!customBuilder.IsPseudoCustomAttribute); CustomAttributeTable.Record rec = new CustomAttributeTable.Record(); rec.Parent = token; rec.Type = asm.IsWindowsRuntime ? customBuilder.Constructor.ImportTo(this) : GetConstructorToken(customBuilder.Constructor).Token; rec.Value = customBuilder.WriteBlob(this); this.CustomAttribute.AddRecord(rec); } internal void AddDeclarativeSecurity(int token, System.Security.Permissions.SecurityAction securityAction, System.Security.PermissionSet permissionSet) { DeclSecurityTable.Record rec = new DeclSecurityTable.Record(); rec.Action = (short)securityAction; rec.Parent = token; // like Ref.Emit, we're using the .NET 1.x xml format rec.PermissionSet = this.Blobs.Add(ByteBuffer.Wrap(System.Text.Encoding.Unicode.GetBytes(permissionSet.ToXml().ToString()))); this.DeclSecurity.AddRecord(rec); } internal void AddDeclarativeSecurity(int token, List declarativeSecurity) { Dictionary> ordered = new Dictionary>(); foreach (CustomAttributeBuilder cab in declarativeSecurity) { int action; // check for HostProtectionAttribute without SecurityAction if (cab.ConstructorArgumentCount == 0) { action = (int)System.Security.Permissions.SecurityAction.LinkDemand; } else { action = (int)cab.GetConstructorArgument(0); } List list; if (!ordered.TryGetValue(action, out list)) { list = new List(); ordered.Add(action, list); } list.Add(cab); } foreach (KeyValuePair> kv in ordered) { DeclSecurityTable.Record rec = new DeclSecurityTable.Record(); rec.Action = (short)kv.Key; rec.Parent = token; rec.PermissionSet = WriteDeclSecurityBlob(kv.Value); this.DeclSecurity.AddRecord(rec); } } private int WriteDeclSecurityBlob(List list) { string xml; if (list.Count == 1 && (xml = list[0].GetLegacyDeclSecurity()) != null) { // write .NET 1.1 format return this.Blobs.Add(ByteBuffer.Wrap(System.Text.Encoding.Unicode.GetBytes(xml))); } ByteBuffer namedArgs = new ByteBuffer(100); ByteBuffer bb = new ByteBuffer(list.Count * 100); bb.Write((byte)'.'); bb.WriteCompressedInt(list.Count); foreach (CustomAttributeBuilder cab in list) { bb.Write(cab.Constructor.DeclaringType.AssemblyQualifiedName); namedArgs.Clear(); cab.WriteNamedArgumentsForDeclSecurity(this, namedArgs); bb.WriteCompressedInt(namedArgs.Length); bb.Write(namedArgs); } return this.Blobs.Add(bb); } public void DefineManifestResource(string name, Stream stream, ResourceAttributes attribute) { manifestResources.Align(8); ManifestResourceTable.Record rec = new ManifestResourceTable.Record(); rec.Offset = manifestResources.Position; rec.Flags = (int)attribute; rec.Name = this.Strings.Add(name); rec.Implementation = 0; this.ManifestResource.AddRecord(rec); manifestResources.Write(0); // placeholder for the length manifestResources.Write(stream); int savePosition = manifestResources.Position; manifestResources.Position = rec.Offset; manifestResources.Write(savePosition - (manifestResources.Position + 4)); manifestResources.Position = savePosition; } public IResourceWriter DefineResource(string name, string description) { return DefineResource(name, description, ResourceAttributes.Public); } public IResourceWriter DefineResource(string name, string description, ResourceAttributes attribute) { // FXBUG we ignore the description, because there is no such thing if (resourceWriters == null) { resourceWriters = new List(); } MemoryStream mem = new MemoryStream(); ResourceWriter rw = new ResourceWriter(mem); resourceWriters.Add(new ResourceWriterRecord(name, rw, mem, attribute)); return rw; } internal void EmitResources() { if (resourceWriters != null) { foreach (ResourceWriterRecord rwr in resourceWriters) { rwr.Emit(this); } } } public override Assembly Assembly { get { return asm; } } internal override Type FindType(TypeName name) { foreach (Type type in types) { if (type.__Namespace == name.Namespace && type.__Name == name.Name) { return type; } } return null; } internal override Type FindTypeIgnoreCase(TypeName lowerCaseName) { foreach (Type type in types) { if (new TypeName(type.__Namespace, type.__Name).ToLowerInvariant() == lowerCaseName) { return type; } } return null; } internal override void GetTypesImpl(List list) { foreach (Type type in types) { if (type != moduleType) { list.Add(type); } } } public ISymbolDocumentWriter DefineDocument(string url, Guid language, Guid languageVendor, Guid documentType) { return symbolWriter.DefineDocument(url, language, languageVendor, documentType); } public int __GetAssemblyToken(Assembly assembly) { return ImportAssemblyRef(assembly); } public TypeToken GetTypeToken(string name) { return new TypeToken(GetType(name, true, false).MetadataToken); } public TypeToken GetTypeToken(Type type) { if (type.Module == this && !asm.IsWindowsRuntime) { return new TypeToken(type.GetModuleBuilderToken()); } else { return new TypeToken(ImportType(type)); } } internal int GetTypeTokenForMemberRef(Type type) { if (type.__IsMissing) { return ImportType(type); } else if (type.IsGenericTypeDefinition) { int token; if (!memberRefTypeTokens.TryGetValue(type, out token)) { ByteBuffer spec = new ByteBuffer(5); Signature.WriteTypeSpec(this, spec, type); token = 0x1B000000 | this.TypeSpec.AddRecord(this.Blobs.Add(spec)); memberRefTypeTokens.Add(type, token); } return token; } else if (type.IsModulePseudoType) { return 0x1A000000 | this.ModuleRef.FindOrAddRecord(this.Strings.Add(type.Module.ScopeName)); } else { return GetTypeToken(type).Token; } } private static bool IsFromGenericTypeDefinition(MemberInfo member) { Type decl = member.DeclaringType; return decl != null && !decl.__IsMissing && decl.IsGenericTypeDefinition; } public FieldToken GetFieldToken(FieldInfo field) { // NOTE for some reason, when TypeBuilder.GetFieldToken() is used on a field in a generic type definition, // a memberref token is returned (confirmed on .NET) unlike for Get(Method|Constructor)Token which always // simply returns the MethodDef token (if the method is from the same module). FieldBuilder fb = field as FieldBuilder; if (fb != null && fb.Module == this && !IsFromGenericTypeDefinition(fb)) { return new FieldToken(fb.MetadataToken); } else { return new FieldToken(field.ImportTo(this)); } } public MethodToken GetMethodToken(MethodInfo method) { MethodBuilder mb = method as MethodBuilder; if (mb != null && mb.ModuleBuilder == this) { return new MethodToken(mb.MetadataToken); } else { return new MethodToken(method.ImportTo(this)); } } // new in .NET 4.5 public MethodToken GetMethodToken(MethodInfo method, IEnumerable optionalParameterTypes) { return __GetMethodToken(method, Util.ToArray(optionalParameterTypes), null); } public MethodToken __GetMethodToken(MethodInfo method, Type[] optionalParameterTypes, CustomModifiers[] customModifiers) { ByteBuffer sig = new ByteBuffer(16); method.MethodSignature.WriteMethodRefSig(this, sig, optionalParameterTypes, customModifiers); MemberRefTable.Record record = new MemberRefTable.Record(); if (method.Module == this) { record.Class = method.MetadataToken; } else { record.Class = GetTypeTokenForMemberRef(method.DeclaringType ?? method.Module.GetModuleType()); } record.Name = Strings.Add(method.Name); record.Signature = Blobs.Add(sig); return new MethodToken(0x0A000000 | MemberRef.FindOrAddRecord(record)); } // when we refer to a method on a generic type definition in the IL stream, // we need to use a MemberRef (even if the method is in the same module) internal MethodToken GetMethodTokenForIL(MethodInfo method) { if (method.IsGenericMethodDefinition) { method = method.MakeGenericMethod(method.GetGenericArguments()); } if (IsFromGenericTypeDefinition(method)) { return new MethodToken(method.ImportTo(this)); } else { return GetMethodToken(method); } } internal int GetMethodTokenWinRT(MethodInfo method) { return asm.IsWindowsRuntime ? method.ImportTo(this) : GetMethodToken(method).Token; } public MethodToken GetConstructorToken(ConstructorInfo constructor) { return GetMethodToken(constructor.GetMethodInfo()); } // new in .NET 4.5 public MethodToken GetConstructorToken(ConstructorInfo constructor, IEnumerable optionalParameterTypes) { return GetMethodToken(constructor.GetMethodInfo(), optionalParameterTypes); } public MethodToken __GetConstructorToken(ConstructorInfo constructor, Type[] optionalParameterTypes, CustomModifiers[] customModifiers) { return __GetMethodToken(constructor.GetMethodInfo(), optionalParameterTypes, customModifiers); } internal int ImportMethodOrField(Type declaringType, string name, Signature sig) { int token; MemberRefKey key = new MemberRefKey(declaringType, name, sig); if (!importedMemberRefs.TryGetValue(key, out token)) { MemberRefTable.Record rec = new MemberRefTable.Record(); rec.Class = GetTypeTokenForMemberRef(declaringType); rec.Name = this.Strings.Add(name); ByteBuffer bb = new ByteBuffer(16); sig.WriteSig(this, bb); rec.Signature = this.Blobs.Add(bb); token = 0x0A000000 | this.MemberRef.AddRecord(rec); importedMemberRefs.Add(key, token); } return token; } internal int ImportMethodSpec(Type declaringType, MethodInfo method, Type[] genericParameters) { Debug.Assert(method.__IsMissing || method.GetMethodOnTypeDefinition() == method); int token; MethodSpecKey key = new MethodSpecKey(declaringType, method.Name, method.MethodSignature, genericParameters); if (!importedMethodSpecs.TryGetValue(key, out token)) { MethodSpecTable.Record rec = new MethodSpecTable.Record(); MethodBuilder mb = method as MethodBuilder; if (mb != null && mb.ModuleBuilder == this && !declaringType.IsGenericType) { rec.Method = mb.MetadataToken; } else { // we're calling ImportMethodOrField directly here, because 'method' may be a MethodDef on a generic TypeDef and 'declaringType' the type instance // (in order words the method and type have already been decoupled by the caller) rec.Method = ImportMethodOrField(declaringType, method.Name, method.MethodSignature); } Writer.ByteBuffer spec = new Writer.ByteBuffer(10); Signature.WriteMethodSpec(this, spec, genericParameters); rec.Instantiation = this.Blobs.Add(spec); token = 0x2B000000 | this.MethodSpec.FindOrAddRecord(rec); importedMethodSpecs.Add(key, token); } return token; } internal int ImportType(Type type) { int token; if (!typeTokens.TryGetValue(type, out token)) { if (type.HasElementType || type.IsConstructedGenericType || type.__IsFunctionPointer) { ByteBuffer spec = new ByteBuffer(5); Signature.WriteTypeSpec(this, spec, type); token = 0x1B000000 | this.TypeSpec.AddRecord(this.Blobs.Add(spec)); } else { TypeRefTable.Record rec = new TypeRefTable.Record(); if (type.IsNested) { rec.ResolutionScope = GetTypeToken(type.DeclaringType).Token; } else if (type.Module == this) { rec.ResolutionScope = 1; } else { rec.ResolutionScope = ImportAssemblyRef(type.Assembly); } rec.TypeName = this.Strings.Add(type.__Name); string ns = type.__Namespace; rec.TypeNameSpace = ns == null ? 0 : this.Strings.Add(ns); token = 0x01000000 | this.TypeRef.AddRecord(rec); } typeTokens.Add(type, token); } return token; } private int ImportAssemblyRef(Assembly asm) { int token; if (!referencedAssemblies.TryGetValue(asm, out token)) { // We can't write the AssemblyRef record here yet, because the identity of the assembly can still change // (if it's an AssemblyBuilder). token = AllocPseudoToken(); referencedAssemblies.Add(asm, token); } return token; } internal void FillAssemblyRefTable() { foreach (KeyValuePair kv in referencedAssemblies) { if (IsPseudoToken(kv.Value)) { RegisterTokenFixup(kv.Value, FindOrAddAssemblyRef(kv.Key.GetName(), false)); } } } private int FindOrAddAssemblyRef(AssemblyName name, bool alwaysAdd) { AssemblyRefTable.Record rec = new AssemblyRefTable.Record(); Version ver = name.Version ?? new Version(0, 0, 0, 0); rec.MajorVersion = (ushort)ver.Major; rec.MinorVersion = (ushort)ver.Minor; rec.BuildNumber = (ushort)ver.Build; rec.RevisionNumber = (ushort)ver.Revision; rec.Flags = (int)(name.Flags & ~AssemblyNameFlags.PublicKey); const AssemblyNameFlags afPA_Specified = (AssemblyNameFlags)0x0080; const AssemblyNameFlags afPA_Mask = (AssemblyNameFlags)0x0070; if ((name.RawFlags & afPA_Specified) != 0) { rec.Flags |= (int)(name.RawFlags & afPA_Mask); } if (name.ContentType == AssemblyContentType.WindowsRuntime) { rec.Flags |= 0x0200; } byte[] publicKeyOrToken = null; if (usePublicKeyAssemblyReference) { publicKeyOrToken = name.GetPublicKey(); } if (publicKeyOrToken == null || publicKeyOrToken.Length == 0) { publicKeyOrToken = name.GetPublicKeyToken() ?? Empty.Array; } else { const int PublicKey = 0x0001; rec.Flags |= PublicKey; } rec.PublicKeyOrToken = this.Blobs.Add(ByteBuffer.Wrap(publicKeyOrToken)); rec.Name = this.Strings.Add(name.Name); rec.Culture = name.Culture == null ? 0 : this.Strings.Add(name.Culture); if (name.hash != null) { rec.HashValue = this.Blobs.Add(ByteBuffer.Wrap(name.hash)); } else { rec.HashValue = 0; } return 0x23000000 | (alwaysAdd ? this.AssemblyRef.AddRecord(rec) : this.AssemblyRef.FindOrAddRecord(rec)); } internal void WriteSymbolTokenMap() { for (int i = 0; i < resolvedTokens.Count; i++) { int newToken = resolvedTokens[i]; // The symbol API doesn't support remapping arbitrary integers, the types have to be the same, // so we copy the type from the newToken, because our pseudo tokens don't have a type. // (see MethodToken.SymbolToken) int oldToken = (i + 1) | (newToken & ~0xFFFFFF); SymbolSupport.RemapToken(symbolWriter, oldToken, newToken); } } internal void RegisterTokenFixup(int pseudoToken, int realToken) { int index = -(pseudoToken + 1); while (resolvedTokens.Count <= index) { resolvedTokens.Add(0); } resolvedTokens[index] = realToken; } internal bool IsPseudoToken(int token) { return token < 0; } internal int ResolvePseudoToken(int pseudoToken) { int index = -(pseudoToken + 1); return resolvedTokens[index]; } internal void ApplyUnmanagedExports(ImageFileMachine imageFileMachine) { if (unmanagedExports.Count != 0) { int type; int size; if (imageFileMachine == ImageFileMachine.I386) { type = 0x05; size = 4; } else { type = 0x06; size = 8; } List methods = new List(); for (int i = 0; i < unmanagedExports.Count; i++) { if (unmanagedExports[i].mb != null) { methods.Add(unmanagedExports[i].mb); } } if (methods.Count != 0) { RelativeVirtualAddress rva = __AddVTableFixups(methods.ToArray(), type); for (int i = 0; i < unmanagedExports.Count; i++) { if (unmanagedExports[i].mb != null) { UnmanagedExport exp = unmanagedExports[i]; exp.rva = new RelativeVirtualAddress(rva.initializedDataOffset + (uint)(methods.IndexOf(unmanagedExports[i].mb) * size)); unmanagedExports[i] = exp; } } } } } internal void FixupMethodBodyTokens() { int methodToken = 0x06000001; int fieldToken = 0x04000001; int parameterToken = 0x08000001; foreach (TypeBuilder type in types) { type.ResolveMethodAndFieldTokens(ref methodToken, ref fieldToken, ref parameterToken); } foreach (int offset in tokenFixupOffsets) { methodBodies.Position = offset; int pseudoToken = methodBodies.GetInt32AtCurrentPosition(); methodBodies.Write(ResolvePseudoToken(pseudoToken)); } foreach (VTableFixups fixup in vtablefixups) { for (int i = 0; i < fixup.count; i++) { initializedData.Position = (int)fixup.initializedDataOffset + i * fixup.SlotWidth; initializedData.Write(ResolvePseudoToken(initializedData.GetInt32AtCurrentPosition())); } } } private int GetHeaderLength() { return 4 + // Signature 2 + // MajorVersion 2 + // MinorVersion 4 + // Reserved 4 + // ImageRuntimeVersion Length StringToPaddedUTF8Length(asm.ImageRuntimeVersion) + 2 + // Flags 2 + // Streams 4 + // #~ Offset 4 + // #~ Size 4 + // StringToPaddedUTF8Length("#~") 4 + // #Strings Offset 4 + // #Strings Size 12 + // StringToPaddedUTF8Length("#Strings") 4 + // #US Offset 4 + // #US Size 4 + // StringToPaddedUTF8Length("#US") 4 + // #GUID Offset 4 + // #GUID Size 8 + // StringToPaddedUTF8Length("#GUID") (Blobs.IsEmpty ? 0 : ( 4 + // #Blob Offset 4 + // #Blob Size 8 // StringToPaddedUTF8Length("#Blob") )); } internal int MetadataLength { get { return GetHeaderLength() + (Blobs.IsEmpty ? 0 : Blobs.Length) + Tables.Length + Strings.Length + UserStrings.Length + Guids.Length; } } internal void WriteMetadata(MetadataWriter mw) { mw.Write(0x424A5342); // Signature ("BSJB") mw.Write((ushort)1); // MajorVersion mw.Write((ushort)1); // MinorVersion mw.Write(0); // Reserved byte[] version = StringToPaddedUTF8(asm.ImageRuntimeVersion); mw.Write(version.Length); // Length mw.Write(version); mw.Write((ushort)0); // Flags // #Blob is the only optional heap if (Blobs.IsEmpty) { mw.Write((ushort)4); // Streams } else { mw.Write((ushort)5); // Streams } int offset = GetHeaderLength(); // Streams mw.Write(offset); // Offset mw.Write(Tables.Length); // Size mw.Write(StringToPaddedUTF8("#~")); offset += Tables.Length; mw.Write(offset); // Offset mw.Write(Strings.Length); // Size mw.Write(StringToPaddedUTF8("#Strings")); offset += Strings.Length; mw.Write(offset); // Offset mw.Write(UserStrings.Length); // Size mw.Write(StringToPaddedUTF8("#US")); offset += UserStrings.Length; mw.Write(offset); // Offset mw.Write(Guids.Length); // Size mw.Write(StringToPaddedUTF8("#GUID")); offset += Guids.Length; if (!Blobs.IsEmpty) { mw.Write(offset); // Offset mw.Write(Blobs.Length); // Size mw.Write(StringToPaddedUTF8("#Blob")); } Tables.Write(mw); Strings.Write(mw); UserStrings.Write(mw); Guids.Write(mw); if (!Blobs.IsEmpty) { Blobs.Write(mw); } } private static int StringToPaddedUTF8Length(string str) { return (System.Text.Encoding.UTF8.GetByteCount(str) + 4) & ~3; } private static byte[] StringToPaddedUTF8(string str) { byte[] buf = new byte[(System.Text.Encoding.UTF8.GetByteCount(str) + 4) & ~3]; System.Text.Encoding.UTF8.GetBytes(str, 0, str.Length, buf, 0); return buf; } internal override void ExportTypes(int fileToken, ModuleBuilder manifestModule) { manifestModule.ExportTypes(types.ToArray(), fileToken); } internal void ExportTypes(Type[] types, int fileToken) { Dictionary declaringTypes = new Dictionary(); foreach (Type type in types) { if (!type.IsModulePseudoType && IsVisible(type)) { ExportedTypeTable.Record rec = new ExportedTypeTable.Record(); rec.Flags = (int)type.Attributes; // LAMESPEC ECMA says that TypeDefId is a row index, but it should be a token rec.TypeDefId = type.MetadataToken; rec.TypeName = this.Strings.Add(type.__Name); string ns = type.__Namespace; rec.TypeNamespace = ns == null ? 0 : this.Strings.Add(ns); if (type.IsNested) { rec.Implementation = declaringTypes[type.DeclaringType]; } else { rec.Implementation = fileToken; } int exportTypeToken = 0x27000000 | this.ExportedType.AddRecord(rec); declaringTypes.Add(type, exportTypeToken); } } } private static bool IsVisible(Type type) { // NOTE this is not the same as Type.IsVisible, because that doesn't take into account family access return type.IsPublic || ((type.IsNestedFamily || type.IsNestedFamORAssem || type.IsNestedPublic) && IsVisible(type.DeclaringType)); } internal void AddConstant(int parentToken, object defaultValue) { ConstantTable.Record rec = new ConstantTable.Record(); rec.Parent = parentToken; ByteBuffer val = new ByteBuffer(16); if (defaultValue == null) { rec.Type = Signature.ELEMENT_TYPE_CLASS; val.Write((int)0); } else if (defaultValue is bool) { rec.Type = Signature.ELEMENT_TYPE_BOOLEAN; val.Write((bool)defaultValue ? (byte)1 : (byte)0); } else if (defaultValue is char) { rec.Type = Signature.ELEMENT_TYPE_CHAR; val.Write((char)defaultValue); } else if (defaultValue is sbyte) { rec.Type = Signature.ELEMENT_TYPE_I1; val.Write((sbyte)defaultValue); } else if (defaultValue is byte) { rec.Type = Signature.ELEMENT_TYPE_U1; val.Write((byte)defaultValue); } else if (defaultValue is short) { rec.Type = Signature.ELEMENT_TYPE_I2; val.Write((short)defaultValue); } else if (defaultValue is ushort) { rec.Type = Signature.ELEMENT_TYPE_U2; val.Write((ushort)defaultValue); } else if (defaultValue is int) { rec.Type = Signature.ELEMENT_TYPE_I4; val.Write((int)defaultValue); } else if (defaultValue is uint) { rec.Type = Signature.ELEMENT_TYPE_U4; val.Write((uint)defaultValue); } else if (defaultValue is long) { rec.Type = Signature.ELEMENT_TYPE_I8; val.Write((long)defaultValue); } else if (defaultValue is ulong) { rec.Type = Signature.ELEMENT_TYPE_U8; val.Write((ulong)defaultValue); } else if (defaultValue is float) { rec.Type = Signature.ELEMENT_TYPE_R4; val.Write((float)defaultValue); } else if (defaultValue is double) { rec.Type = Signature.ELEMENT_TYPE_R8; val.Write((double)defaultValue); } else if (defaultValue is string) { rec.Type = Signature.ELEMENT_TYPE_STRING; foreach (char c in (string)defaultValue) { val.Write(c); } } else if (defaultValue is DateTime) { rec.Type = Signature.ELEMENT_TYPE_I8; val.Write(((DateTime)defaultValue).Ticks); } else { throw new ArgumentException(); } rec.Value = this.Blobs.Add(val); this.Constant.AddRecord(rec); } ModuleBuilder ITypeOwner.ModuleBuilder { get { return this; } } internal override Type ResolveType(int metadataToken, IGenericContext context) { if (metadataToken >> 24 != TypeDefTable.Index) { throw new NotImplementedException(); } return types[(metadataToken & 0xFFFFFF) - 1]; } public override MethodBase ResolveMethod(int metadataToken, Type[] genericTypeArguments, Type[] genericMethodArguments) { if (genericTypeArguments != null || genericMethodArguments != null) { throw new NotImplementedException(); } // this method is inefficient, but since it isn't used we don't care if ((metadataToken >> 24) == MemberRefTable.Index) { foreach (KeyValuePair kv in importedMemberRefs) { if (kv.Value == metadataToken) { return kv.Key.LookupMethod(); } } } // HACK if we're given a SymbolToken, we need to convert back if ((metadataToken & 0xFF000000) == 0x06000000) { metadataToken = -(metadataToken & 0x00FFFFFF); } foreach (Type type in types) { MethodBase method = ((TypeBuilder)type).LookupMethod(metadataToken); if (method != null) { return method; } } return ((TypeBuilder)moduleType).LookupMethod(metadataToken); } public override FieldInfo ResolveField(int metadataToken, Type[] genericTypeArguments, Type[] genericMethodArguments) { throw new NotImplementedException(); } public override MemberInfo ResolveMember(int metadataToken, Type[] genericTypeArguments, Type[] genericMethodArguments) { throw new NotImplementedException(); } public override string ResolveString(int metadataToken) { throw new NotImplementedException(); } public override string FullyQualifiedName { get { return Path.GetFullPath(Path.Combine(asm.dir, fileName)); } } public override string Name { get { return fileName; } } public override Guid ModuleVersionId { get { return mvid; } } public void __SetModuleVersionId(Guid guid) { mvid = guid; } public override Type[] __ResolveOptionalParameterTypes(int metadataToken, Type[] genericTypeArguments, Type[] genericMethodArguments, out CustomModifiers[] customModifiers) { throw new NotImplementedException(); } public override string ScopeName { get { return moduleName; } } public ISymbolWriter GetSymWriter() { return symbolWriter; } public void DefineUnmanagedResource(string resourceFileName) { // This method reads the specified resource file (Win32 .res file) and converts it into the appropriate format and embeds it in the .rsrc section, // also setting the Resource Directory entry. unmanagedResources = new ResourceSection(); unmanagedResources.ExtractResources(System.IO.File.ReadAllBytes(resourceFileName)); } public bool IsTransient() { return false; } public void SetUserEntryPoint(MethodInfo entryPoint) { int token = entryPoint.MetadataToken; if (token < 0) { token = -token | 0x06000000; } if (symbolWriter != null) { symbolWriter.SetUserEntryPoint(new SymbolToken(token)); } } public StringToken GetStringConstant(string str) { return new StringToken(this.UserStrings.Add(str) | (0x70 << 24)); } public SignatureToken GetSignatureToken(SignatureHelper sigHelper) { return new SignatureToken(this.StandAloneSig.FindOrAddRecord(this.Blobs.Add(sigHelper.GetSignature(this))) | (StandAloneSigTable.Index << 24)); } public SignatureToken GetSignatureToken(byte[] sigBytes, int sigLength) { return new SignatureToken(this.StandAloneSig.FindOrAddRecord(this.Blobs.Add(ByteBuffer.Wrap(sigBytes, sigLength))) | (StandAloneSigTable.Index << 24)); } public MethodInfo GetArrayMethod(Type arrayClass, string methodName, CallingConventions callingConvention, Type returnType, Type[] parameterTypes) { return new ArrayMethod(this, arrayClass, methodName, callingConvention, returnType, parameterTypes); } public MethodToken GetArrayMethodToken(Type arrayClass, string methodName, CallingConventions callingConvention, Type returnType, Type[] parameterTypes) { return GetMethodToken(GetArrayMethod(arrayClass, methodName, callingConvention, returnType, parameterTypes)); } internal override Type GetModuleType() { return moduleType; } internal override IKVM.Reflection.Reader.ByteReader GetBlob(int blobIndex) { return Blobs.GetBlob(blobIndex); } internal int GetSignatureBlobIndex(Signature sig) { ByteBuffer bb = new ByteBuffer(16); sig.WriteSig(this, bb); return this.Blobs.Add(bb); } // non-standard API public new long __ImageBase { get { return imageBaseAddress; } set { imageBaseAddress = value; } } protected override long GetImageBaseImpl() { return imageBaseAddress; } public new long __StackReserve { get { return stackReserve; } set { stackReserve = value; } } protected override long GetStackReserveImpl() { return stackReserve; } [Obsolete("Use __StackReserve property.")] public void __SetStackReserve(long stackReserve) { __StackReserve = stackReserve; } internal ulong GetStackReserve(ulong defaultValue) { return stackReserve == -1 ? defaultValue : (ulong)stackReserve; } public new int __FileAlignment { get { return fileAlignment; } set { fileAlignment = value; } } protected override int GetFileAlignmentImpl() { return fileAlignment; } public new DllCharacteristics __DllCharacteristics { get { return dllCharacteristics; } set { dllCharacteristics = value; } } protected override DllCharacteristics GetDllCharacteristicsImpl() { return dllCharacteristics; } public override int MDStreamVersion { get { return asm.mdStreamVersion; } } private int AddTypeRefByName(int resolutionScope, string ns, string name) { TypeRefTable.Record rec = new TypeRefTable.Record(); rec.ResolutionScope = resolutionScope; rec.TypeName = this.Strings.Add(name); rec.TypeNameSpace = ns == null ? 0 : this.Strings.Add(ns); return 0x01000000 | this.TypeRef.AddRecord(rec); } public void __Save(PortableExecutableKinds portableExecutableKind, ImageFileMachine imageFileMachine) { SaveImpl(null, portableExecutableKind, imageFileMachine); } public void __Save(Stream stream, PortableExecutableKinds portableExecutableKind, ImageFileMachine imageFileMachine) { if (!stream.CanRead || !stream.CanWrite || !stream.CanSeek || stream.Position != 0) { throw new ArgumentException("Stream must support read/write/seek and current position must be zero.", "stream"); } SaveImpl(stream, portableExecutableKind, imageFileMachine); } private void SaveImpl(Stream streamOrNull, PortableExecutableKinds portableExecutableKind, ImageFileMachine imageFileMachine) { SetIsSaved(); PopulatePropertyAndEventTables(); IList attributes = asm.GetCustomAttributesData(null); if (attributes.Count > 0) { int mscorlib = ImportAssemblyRef(universe.Mscorlib); int[] placeholderTokens = new int[4]; string[] placeholderTypeNames = new string[] { "AssemblyAttributesGoHere", "AssemblyAttributesGoHereM", "AssemblyAttributesGoHereS", "AssemblyAttributesGoHereSM" }; foreach (CustomAttributeData cad in attributes) { int index; if (cad.Constructor.DeclaringType.BaseType == universe.System_Security_Permissions_CodeAccessSecurityAttribute) { if (cad.Constructor.DeclaringType.IsAllowMultipleCustomAttribute) { index = 3; } else { index = 2; } } else if (cad.Constructor.DeclaringType.IsAllowMultipleCustomAttribute) { index = 1; } else { index = 0; } if (placeholderTokens[index] == 0) { // we manually add a TypeRef without looking it up in mscorlib, because Mono and Silverlight's mscorlib don't have these types placeholderTokens[index] = AddTypeRefByName(mscorlib, "System.Runtime.CompilerServices", placeholderTypeNames[index]); } SetCustomAttribute(placeholderTokens[index], cad.__ToBuilder()); } } FillAssemblyRefTable(); EmitResources(); ModuleWriter.WriteModule(null, null, this, PEFileKinds.Dll, portableExecutableKind, imageFileMachine, unmanagedResources, 0, streamOrNull); } public void __AddAssemblyReference(AssemblyName assemblyName) { __AddAssemblyReference(assemblyName, null); } public void __AddAssemblyReference(AssemblyName assemblyName, Assembly assembly) { if (referencedAssemblyNames == null) { referencedAssemblyNames = new List(); } referencedAssemblyNames.Add((AssemblyName)assemblyName.Clone()); int token = FindOrAddAssemblyRef(assemblyName, true); if (assembly != null) { referencedAssemblies.Add(assembly, token); } } public override AssemblyName[] __GetReferencedAssemblies() { List list = new List(); if (referencedAssemblyNames != null) { foreach (AssemblyName name in referencedAssemblyNames) { if (!list.Contains(name)) { list.Add(name); } } } foreach (Assembly asm in referencedAssemblies.Keys) { AssemblyName name = asm.GetName(); if (!list.Contains(name)) { list.Add(name); } } return list.ToArray(); } public void __AddModuleReference(string module) { this.ModuleRef.FindOrAddRecord(module == null ? 0 : this.Strings.Add(module)); } public override string[] __GetReferencedModules() { string[] arr = new string[this.ModuleRef.RowCount]; for (int i = 0; i < arr.Length; i++) { arr[i] = this.Strings.Find(this.ModuleRef.records[i]); } return arr; } public override Type[] __GetReferencedTypes() { List list = new List(); foreach (KeyValuePair kv in typeTokens) { if (kv.Value >> 24 == TypeRefTable.Index) { list.Add(kv.Key); } } return list.ToArray(); } public override Type[] __GetExportedTypes() { throw new NotImplementedException(); } public int __AddModule(int flags, string name, byte[] hash) { FileTable.Record file = new FileTable.Record(); file.Flags = flags; file.Name = this.Strings.Add(name); file.HashValue = this.Blobs.Add(ByteBuffer.Wrap(hash)); return 0x26000000 + this.File.AddRecord(file); } public int __AddManifestResource(int offset, ResourceAttributes flags, string name, int implementation) { ManifestResourceTable.Record res = new ManifestResourceTable.Record(); res.Offset = offset; res.Flags = (int)flags; res.Name = this.Strings.Add(name); res.Implementation = implementation; return 0x28000000 + this.ManifestResource.AddRecord(res); } public void __SetCustomAttributeFor(int token, CustomAttributeBuilder customBuilder) { SetCustomAttribute(token, customBuilder); } public RelativeVirtualAddress __AddVTableFixups(MethodBuilder[] methods, int type) { initializedData.Align(8); VTableFixups fixups; fixups.initializedDataOffset = (uint)initializedData.Position; fixups.count = (ushort)methods.Length; fixups.type = (ushort)type; foreach (MethodBuilder mb in methods) { initializedData.Write(mb.MetadataToken); if (fixups.SlotWidth == 8) { initializedData.Write(0); } } vtablefixups.Add(fixups); return new RelativeVirtualAddress(fixups.initializedDataOffset); } public void __AddUnmanagedExportStub(string name, int ordinal, RelativeVirtualAddress rva) { AddUnmanagedExport(name, ordinal, null, rva); } internal void AddUnmanagedExport(string name, int ordinal, MethodBuilder methodBuilder, RelativeVirtualAddress rva) { UnmanagedExport export; export.name = name; export.ordinal = ordinal; export.mb = methodBuilder; export.rva = rva; unmanagedExports.Add(export); } internal void SetInterfaceImplementationCustomAttribute(TypeBuilder typeBuilder, Type interfaceType, CustomAttributeBuilder cab) { // NOTE since interfaceimpls are extremely common and custom attributes on them are extremely rare, // we store (and resolve) the custom attributes in such away as to avoid impacting the common case performance if (interfaceImplCustomAttributes == null) { interfaceImplCustomAttributes = new List(); } InterfaceImplCustomAttribute rec; rec.type = typeBuilder.MetadataToken; int token = GetTypeToken(interfaceType).Token; switch (token >> 24) { case TypeDefTable.Index: token = (token & 0xFFFFFF) << 2 | 0; break; case TypeRefTable.Index: token = (token & 0xFFFFFF) << 2 | 1; break; case TypeSpecTable.Index: token = (token & 0xFFFFFF) << 2 | 2; break; default: throw new InvalidOperationException(); } rec.interfaceType = token; rec.pseudoToken = AllocPseudoToken(); interfaceImplCustomAttributes.Add(rec); SetCustomAttribute(rec.pseudoToken, cab); } internal void ResolveInterfaceImplPseudoTokens() { if (interfaceImplCustomAttributes != null) { foreach (InterfaceImplCustomAttribute rec in interfaceImplCustomAttributes) { for (int i = 0; i < InterfaceImpl.records.Length; i++) { if (InterfaceImpl.records[i].Class == rec.type && InterfaceImpl.records[i].Interface == rec.interfaceType) { RegisterTokenFixup(rec.pseudoToken, (InterfaceImplTable.Index << 24) | (i + 1)); break; } } } } } internal void FixupPseudoToken(ref int token) { if (IsPseudoToken(token)) { token = ResolvePseudoToken(token); } } internal void SetIsSaved() { if (saved) { throw new InvalidOperationException(); } saved = true; } internal bool IsSaved { get { return saved; } } internal override string GetString(int index) { return this.Strings.Find(index); } } struct UnmanagedExport { internal string name; internal int ordinal; internal RelativeVirtualAddress rva; internal MethodBuilder mb; } public struct RelativeVirtualAddress { internal readonly uint initializedDataOffset; internal RelativeVirtualAddress(uint initializedDataOffset) { this.initializedDataOffset = initializedDataOffset; } public static RelativeVirtualAddress operator +(RelativeVirtualAddress rva, int offset) { return new RelativeVirtualAddress(rva.initializedDataOffset + (uint)offset); } } class ArrayMethod : MethodInfo { private readonly Module module; private readonly Type arrayClass; private readonly string methodName; private readonly CallingConventions callingConvention; private readonly Type returnType; protected readonly Type[] parameterTypes; private MethodSignature methodSignature; internal ArrayMethod(Module module, Type arrayClass, string methodName, CallingConventions callingConvention, Type returnType, Type[] parameterTypes) { this.module = module; this.arrayClass = arrayClass; this.methodName = methodName; this.callingConvention = callingConvention; this.returnType = returnType ?? module.universe.System_Void; this.parameterTypes = Util.Copy(parameterTypes); } public override MethodBody GetMethodBody() { throw new InvalidOperationException(); } public override int __MethodRVA { get { throw new InvalidOperationException(); } } public override MethodImplAttributes GetMethodImplementationFlags() { throw new NotSupportedException(); } public override ParameterInfo[] GetParameters() { throw new NotSupportedException(); } internal override int ImportTo(ModuleBuilder module) { return module.ImportMethodOrField(arrayClass, methodName, MethodSignature); } public override MethodAttributes Attributes { get { throw new NotSupportedException(); } } public override CallingConventions CallingConvention { get { return callingConvention; } } public override Type DeclaringType { get { return arrayClass; } } internal override MethodSignature MethodSignature { get { if (methodSignature == null) { methodSignature = MethodSignature.MakeFromBuilder(returnType, parameterTypes, new PackedCustomModifiers(), callingConvention, 0); } return methodSignature; } } public override Module Module { // FXBUG like .NET, we return the module that GetArrayMethod was called on, not the module associated with the array type get { return module; } } public override string Name { get { return methodName; } } internal override int ParameterCount { get { return parameterTypes.Length; } } public override ParameterInfo ReturnParameter { // FXBUG like .NET, we throw NotImplementedException get { throw new NotImplementedException(); } } public override Type ReturnType { get { return returnType; } } internal override bool HasThis { get { return (callingConvention & (CallingConventions.HasThis | CallingConventions.ExplicitThis)) == CallingConventions.HasThis; } } internal override int GetCurrentToken() { return this.MetadataToken; } internal override bool IsBaked { get { return arrayClass.IsBaked; } } } }