// // System.Reflection/Module.cs // // Author: // Paolo Molaro (lupus@ximian.com) // // (C) 2001 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. // using System; using System.Reflection; using System.Runtime.Serialization; using System.Security.Cryptography.X509Certificates; using System.Runtime.InteropServices; using System.Runtime.CompilerServices; namespace System.Reflection { internal enum ResolveTokenError { OutOfRange, BadTable, Other }; [Serializable] public class Module : ISerializable, ICustomAttributeProvider { public static readonly TypeFilter FilterTypeName; public static readonly TypeFilter FilterTypeNameIgnoreCase; private IntPtr _impl; /* a pointer to a MonoImage */ internal Assembly assembly; internal string fqname; internal string name; internal string scopename; internal bool is_resource; internal int token; const BindingFlags defaultBindingFlags = BindingFlags.Public | BindingFlags.Static | BindingFlags.Instance; static Module () { FilterTypeName = new TypeFilter (filter_by_type_name); FilterTypeNameIgnoreCase = new TypeFilter (filter_by_type_name_ignore_case); } internal Module () { } ~Module () { Close (); } public Assembly Assembly { get { return assembly; } } public virtual string FullyQualifiedName { get { return fqname; } } public string Name { get { return name; } } public string ScopeName { get { return scopename; } } #if NET_2_0 public ModuleHandle ModuleHandle { get { return new ModuleHandle (_impl); } } public extern int MetadataToken { [MethodImplAttribute (MethodImplOptions.InternalCall)] get; } #endif public virtual Type[] FindTypes(TypeFilter filter, object filterCriteria) { System.Collections.ArrayList filtered = new System.Collections.ArrayList (); Type[] types = GetTypes (); foreach (Type t in types) if (filter (t, filterCriteria)) filtered.Add (t); return (Type[])filtered.ToArray (typeof(Type)); } public virtual object[] GetCustomAttributes(bool inherit) { return MonoCustomAttrs.GetCustomAttributes (this, inherit); } public virtual object[] GetCustomAttributes(Type attributeType, bool inherit) { return MonoCustomAttrs.GetCustomAttributes (this, attributeType, inherit); } public FieldInfo GetField (string name) { if (IsResource ()) return null; return GetGlobalType ().GetField (name, BindingFlags.Public | BindingFlags.Static); } public FieldInfo GetField (string name, BindingFlags flags) { if (IsResource ()) return null; return GetGlobalType ().GetField (name, flags); } public FieldInfo[] GetFields () { if (IsResource ()) return new FieldInfo [0]; return GetGlobalType ().GetFields (BindingFlags.Public | BindingFlags.Static); } public MethodInfo GetMethod (string name) { return GetMethodImpl (name, defaultBindingFlags, null, CallingConventions.Any, Type.EmptyTypes, null); } public MethodInfo GetMethod (string name, Type[] types) { return GetMethodImpl (name, defaultBindingFlags, null, CallingConventions.Any, types, null); } public MethodInfo GetMethod (string name, BindingFlags bindingAttr, Binder binder, CallingConventions callConvention, Type[] types, ParameterModifier[] modifiers) { return GetMethodImpl (name, bindingAttr, binder, callConvention, types, modifiers); } protected virtual MethodInfo GetMethodImpl (string name, BindingFlags bindingAttr, Binder binder, CallingConventions callConvention, Type[] types, ParameterModifier[] modifiers) { if (IsResource ()) return null; return GetGlobalType ().GetMethod (name, bindingAttr, binder, callConvention, types, modifiers); } public MethodInfo[] GetMethods () { if (IsResource ()) return new MethodInfo [0]; return GetGlobalType ().GetMethods (); } public virtual void GetObjectData (SerializationInfo info, StreamingContext context) { UnitySerializationHolder.GetModuleData (this, info, context); } public X509Certificate GetSignerCertificate () { try { return X509Certificate.CreateFromSignedFile (assembly.Location); } catch { return null; } } public virtual Type GetType(string className) { return GetType (className, false, false); } public virtual Type GetType(string className, bool ignoreCase) { return GetType (className, false, ignoreCase); } public virtual Type GetType(string className, bool throwOnError, bool ignoreCase) { if (className == null) throw new ArgumentNullException ("className"); if (className == String.Empty) throw new ArgumentException ("Type name can't be empty"); return assembly.InternalGetType (this, className, throwOnError, ignoreCase); } [MethodImplAttribute (MethodImplOptions.InternalCall)] private extern Type[] InternalGetTypes (); public virtual Type[] GetTypes() { return InternalGetTypes (); } public virtual bool IsDefined (Type attributeType, bool inherit) { return MonoCustomAttrs.IsDefined (this, attributeType, inherit); } public bool IsResource() { return is_resource; } public override string ToString () { return name; } #if NET_2_0 || BOOTSTRAP_NET_2_0 public #else internal #endif Guid Mvid { get { return Mono_GetGuid (this); } } #if NET_2_0 private Exception resolve_token_exception (int metadataToken, ResolveTokenError error, string tokenType) { if (error == ResolveTokenError.OutOfRange) return new ArgumentOutOfRangeException ("metadataToken", String.Format ("Token 0x{0:x} is not valid in the scope of module {1}", metadataToken, name)); else return new ArgumentException (String.Format ("Token 0x{0:x} is not a valid {1} token in the scope of module {2}", metadataToken, tokenType, name), "metadataToken"); } public FieldInfo ResolveField (int metadataToken) { ResolveTokenError error; IntPtr handle = ResolveFieldToken (_impl, metadataToken, out error); if (handle == IntPtr.Zero) throw resolve_token_exception (metadataToken, error, "Field"); else return FieldInfo.GetFieldFromHandle (new RuntimeFieldHandle (handle)); } [MonoTODO] public MemberInfo ResolveMember (int metadataToken) { ResolveTokenError error; MemberInfo m = ResolveMemberToken (_impl, metadataToken, out error); if (m == null) throw resolve_token_exception (metadataToken, error, "MemberInfo"); else return m; } public MethodBase ResolveMethod (int metadataToken) { ResolveTokenError error; IntPtr handle = ResolveMethodToken (_impl, metadataToken, out error); if (handle == IntPtr.Zero) throw resolve_token_exception (metadataToken, error, "MethodBase"); else return MethodBase.GetMethodFromHandle (new RuntimeMethodHandle (handle)); } public string ResolveString (int metadataToken) { ResolveTokenError error; string s = ResolveStringToken (_impl, metadataToken, out error); if (s == null) throw resolve_token_exception (metadataToken, error, "string"); else return s; } public Type ResolveType (int metadataToken) { ResolveTokenError error; IntPtr handle = ResolveTypeToken (_impl, metadataToken, out error); if (handle == IntPtr.Zero) throw resolve_token_exception (metadataToken, error, "Type"); else return Type.GetTypeFromHandle (new RuntimeTypeHandle (handle)); } #endif // Mono Extension: returns the GUID of this module internal static Guid Mono_GetGuid (Module module) { return new Guid (module.GetGuidInternal ()); } private static bool filter_by_type_name (Type m, object filterCriteria) { string s = (string)filterCriteria; if (s.EndsWith ("*")) return m.Name.StartsWith (s.Substring (0, s.Length - 1)); else return m.Name == s; } private static bool filter_by_type_name_ignore_case (Type m, object filterCriteria) { string s = (string)filterCriteria; if (s.EndsWith ("*")) return m.Name.ToLower ().StartsWith (s.Substring (0, s.Length - 1).ToLower ()); else return String.Compare (m.Name, s, true) == 0; } [MethodImplAttribute (MethodImplOptions.InternalCall)] private extern string GetGuidInternal (); [MethodImplAttribute (MethodImplOptions.InternalCall)] private extern Type GetGlobalType (); [MethodImplAttribute (MethodImplOptions.InternalCall)] private extern void Close (); [MethodImplAttribute (MethodImplOptions.InternalCall)] internal static extern IntPtr ResolveTypeToken (IntPtr module, int token, out ResolveTokenError error); [MethodImplAttribute (MethodImplOptions.InternalCall)] internal static extern IntPtr ResolveMethodToken (IntPtr module, int token, out ResolveTokenError error); [MethodImplAttribute (MethodImplOptions.InternalCall)] internal static extern IntPtr ResolveFieldToken (IntPtr module, int token, out ResolveTokenError error); [MethodImplAttribute (MethodImplOptions.InternalCall)] internal static extern string ResolveStringToken (IntPtr module, int token, out ResolveTokenError error); [MethodImplAttribute (MethodImplOptions.InternalCall)] internal static extern MemberInfo ResolveMemberToken (IntPtr module, int token, out ResolveTokenError error); [MethodImplAttribute (MethodImplOptions.InternalCall)] internal static extern void GetPEKind (IntPtr module, out PortableExecutableKind peKind, out ImageFileMachine machine); } }