//
// System.Reflection/RuntimeModule.cs
//
// Author:
//   Rodrigo Kumpera (rkumpera@novell.com)
//
// Copyright (C) 2010 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.Collections;
using System.Collections.Generic;
using System.Globalization;
using System.Runtime.InteropServices;
using System.Runtime.CompilerServices;
#if !NETCORE
using System.Security.Cryptography.X509Certificates;
using System.Security;
using System.Security.Permissions;
#endif
using System.Runtime.Serialization;

namespace System.Reflection {

	[ComVisible (true)]
#if !NETCORE
	[ComDefaultInterfaceAttribute (typeof (_Module))]
#endif
	[Serializable]
	[ClassInterface(ClassInterfaceType.None)]
	[StructLayout (LayoutKind.Sequential)]
	class RuntimeModule : Module
	{
#pragma warning disable 649
		#region Sync with object-internals.h
		#region Sync with ModuleBuilder
		internal 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;
		#endregion
		#endregion
#pragma warning restore 649

		public
		override
		Assembly Assembly {
			get { return assembly; }
		}

		public
		override
		// Note: we do not ask for PathDiscovery because no path is returned here.
		// However MS Fx requires it (see FDBK23572 for details).
		string Name {
			get { return name; }
		}
	
		public
		override
		string ScopeName {
			get { return scopename; }
		}

		public
		override
		int MDStreamVersion {
			get {
				if (_impl == IntPtr.Zero)
					throw new NotSupportedException ();
				return GetMDStreamVersion (_impl);
			}
		}

		public
		override
		Guid ModuleVersionId {
			get {
				return GetModuleVersionId ();
			}
		}

		public override
		string FullyQualifiedName {
			get {
#if !MOBILE && !NETCORE
				if (SecurityManager.SecurityEnabled) {
					new FileIOPermission (FileIOPermissionAccess.PathDiscovery, fqname).Demand ();
				}
#endif
				return fqname;
			}
		}

		public
		override
		bool IsResource()
		{
			return is_resource;
		}

		public override
		Type[] FindTypes(TypeFilter filter, object filterCriteria) 
		{
			var filtered = new List<Type> ();
			Type[] types = GetTypes ();
			foreach (Type t in types)
				if (filter (t, filterCriteria))
					filtered.Add (t);
			return filtered.ToArray ();
		}

		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);
		}

		public override
		FieldInfo GetField (string name, BindingFlags bindingAttr) 
		{
			if (name == null)
				throw new ArgumentNullException("name");

			if (IsResource ())
				return null;

			Type globalType = GetGlobalType (_impl);
			return (globalType != null) ? globalType.GetField (name, bindingAttr) : null;
		}

		public override
		FieldInfo[] GetFields (BindingFlags bindingFlags)
		{
			if (IsResource ())
				return new FieldInfo [0];

			Type globalType = GetGlobalType (_impl);
			return (globalType != null) ? globalType.GetFields (bindingFlags) : new FieldInfo [0];
		}

		public override
		int MetadataToken {
			get {
				return get_MetadataToken (this);
			}
		}

		protected
		override
		MethodInfo GetMethodImpl (string name, BindingFlags bindingAttr, Binder binder, CallingConventions callConvention, Type[] types, ParameterModifier[] modifiers) 
		{
			if (IsResource ())
				return null;

			Type globalType = GetGlobalType (_impl);
			if (globalType == null)
				return null;
			if (types == null)
				return globalType.GetMethod (name);
			return globalType.GetMethod (name, bindingAttr, binder, callConvention, types, modifiers);
		}

		public
		override
		MethodInfo[] GetMethods (BindingFlags bindingFlags) {
			if (IsResource ())
				return new MethodInfo [0];

			Type globalType = GetGlobalType (_impl);
			return (globalType != null) ? globalType.GetMethods (bindingFlags) : new MethodInfo [0];
		}

#if !NETCORE
		internal override ModuleHandle GetModuleHandleImpl() => new ModuleHandle (_impl);
#endif

		public override
		void GetPEKind (out PortableExecutableKinds peKind, out ImageFileMachine machine) {
			RuntimeModule.GetPEKind (_impl, out peKind, out machine);
		}

		public override
		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);
		}
	
		public override
		bool IsDefined (Type attributeType, bool inherit) 
		{
			return MonoCustomAttrs.IsDefined (this, attributeType, inherit);
		}

		public
		override
		FieldInfo ResolveField (int metadataToken, Type [] genericTypeArguments, Type [] genericMethodArguments) {
			return ResolveField (this, _impl, metadataToken, genericTypeArguments, genericMethodArguments);
		}

		internal static FieldInfo ResolveField (Module module, IntPtr monoModule, int metadataToken, Type [] genericTypeArguments, Type [] genericMethodArguments) {
			ResolveTokenError error;

			IntPtr handle = ResolveFieldToken (monoModule, metadataToken, ptrs_from_types (genericTypeArguments), ptrs_from_types (genericMethodArguments), out error);
			if (handle == IntPtr.Zero)
				throw resolve_token_exception (module.Name, metadataToken, error, "Field");
			else
				return FieldInfo.GetFieldFromHandle (new RuntimeFieldHandle (handle));
		}

		public
		override
		MemberInfo ResolveMember (int metadataToken, Type [] genericTypeArguments, Type [] genericMethodArguments) {
			return ResolveMember (this, _impl, metadataToken, genericTypeArguments, genericMethodArguments);
		}

		internal static MemberInfo ResolveMember (Module module, IntPtr monoModule, int metadataToken, Type [] genericTypeArguments, Type [] genericMethodArguments) {
			ResolveTokenError error;

			MemberInfo m = ResolveMemberToken (monoModule, metadataToken, ptrs_from_types (genericTypeArguments), ptrs_from_types (genericMethodArguments), out error);
			if (m == null)
				throw resolve_token_exception (module.Name, metadataToken, error, "MemberInfo");
			else
				return m;
		}

		public
		override
		MethodBase ResolveMethod (int metadataToken, Type [] genericTypeArguments, Type [] genericMethodArguments) {
			return ResolveMethod (this, _impl, metadataToken, genericTypeArguments, genericMethodArguments);
		}

		internal static MethodBase ResolveMethod (Module module, IntPtr monoModule, int metadataToken, Type [] genericTypeArguments, Type [] genericMethodArguments) {
			ResolveTokenError error;

			IntPtr handle = ResolveMethodToken (monoModule, metadataToken, ptrs_from_types (genericTypeArguments), ptrs_from_types (genericMethodArguments), out error);
			if (handle == IntPtr.Zero)
				throw resolve_token_exception (module.Name, metadataToken, error, "MethodBase");
			else
				return RuntimeMethodInfo.GetMethodFromHandleNoGenericCheck (new RuntimeMethodHandle (handle));
		}

		public
		override
		string ResolveString (int metadataToken) {
			return ResolveString (this, _impl, metadataToken);
		}

		internal static string ResolveString (Module module, IntPtr monoModule, int metadataToken) {
			ResolveTokenError error;

			string s = ResolveStringToken (monoModule, metadataToken, out error);
			if (s == null)
				throw resolve_token_exception (module.Name, metadataToken, error, "string");
			else
				return s;
		}

		public
		override
		Type ResolveType (int metadataToken, Type [] genericTypeArguments, Type [] genericMethodArguments) {
			return ResolveType (this, _impl,  metadataToken, genericTypeArguments, genericMethodArguments);
		}

		internal static Type ResolveType (Module module, IntPtr monoModule, int metadataToken, Type [] genericTypeArguments, Type [] genericMethodArguments) {
			ResolveTokenError error;

			IntPtr handle = ResolveTypeToken (monoModule, metadataToken, ptrs_from_types (genericTypeArguments), ptrs_from_types (genericMethodArguments), out error);
			if (handle == IntPtr.Zero)
				throw resolve_token_exception (module.Name, metadataToken, error, "Type");
			else
				return Type.GetTypeFromHandle (new RuntimeTypeHandle (handle));
		}

		public
		override
		byte[] ResolveSignature (int metadataToken) {
			return ResolveSignature (this, _impl, metadataToken);
		}

		internal static byte[] ResolveSignature (Module module, IntPtr monoModule, int metadataToken) {
			ResolveTokenError error;

		    byte[] res = ResolveSignature (monoModule, metadataToken, out error);
			if (res == null)
				throw resolve_token_exception (module.Name, metadataToken, error, "signature");
			else
				return res;
		}

#if !NETCORE
		public override void GetObjectData (SerializationInfo info, StreamingContext context)
		{
			if (info == null)
				throw new ArgumentNullException ("info");

			UnitySerializationHolder.GetUnitySerializationInfo (info, UnitySerializationHolder.ModuleUnity, this.ScopeName, this.GetRuntimeAssembly ());
		}
#endif

#if !MOBILE && !NETCORE
		public
		override
		X509Certificate GetSignerCertificate ()
		{
			try {
				return X509Certificate.CreateFromSignedFile (assembly.Location);
			}
			catch {
				return null;
			}
		}
#endif

		public override
		Type[] GetTypes() 
		{
			return InternalGetTypes (_impl);
		}

		public override IList<CustomAttributeData> GetCustomAttributesData () {
			return CustomAttributeData.GetCustomAttributes (this);
		}

		internal RuntimeAssembly GetRuntimeAssembly ()
		{
			return (RuntimeAssembly)assembly;
		}

		internal IntPtr MonoModule {
			get {
				return _impl;
			}
		}

#if NETCORE
		internal Guid GetModuleVersionId ()
#else
		internal override Guid GetModuleVersionId ()
#endif
		{
			var guid = new byte [16];
			GetGuidInternal (_impl, guid);
			return new Guid (guid);
		}

		internal static Exception resolve_token_exception (string name, 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");
		}

		internal static IntPtr[] ptrs_from_types (Type[] types) {
			if (types == null)
				return null;
			else {
				IntPtr[] res = new IntPtr [types.Length];
				for (int i = 0; i < types.Length; ++i) {
					if (types [i] == null)
						throw new ArgumentException ();
					res [i] = types [i].TypeHandle.Value;
				}
				return res;
			}
		}

		// This calls ves_icall_reflection_get_token, so needs a Module argument
		[MethodImplAttribute (MethodImplOptions.InternalCall)]
		internal static extern int get_MetadataToken (Module module);

		[MethodImplAttribute (MethodImplOptions.InternalCall)]
		internal static extern int GetMDStreamVersion (IntPtr module);

		[MethodImplAttribute (MethodImplOptions.InternalCall)]
		internal static extern Type[] InternalGetTypes (IntPtr module);

		[MethodImplAttribute (MethodImplOptions.InternalCall)]
		internal static extern IntPtr GetHINSTANCE (IntPtr module);

		[MethodImplAttribute (MethodImplOptions.InternalCall)]
		private static extern void GetGuidInternal (IntPtr module, byte[] guid);

		[MethodImplAttribute (MethodImplOptions.InternalCall)]
		internal static extern Type GetGlobalType (IntPtr module);

		[MethodImplAttribute (MethodImplOptions.InternalCall)]
		internal static extern IntPtr ResolveTypeToken (IntPtr module, int token, IntPtr[] type_args, IntPtr[] method_args, out ResolveTokenError error);

		[MethodImplAttribute (MethodImplOptions.InternalCall)]
		internal static extern IntPtr ResolveMethodToken (IntPtr module, int token, IntPtr[] type_args, IntPtr[] method_args, out ResolveTokenError error);

		[MethodImplAttribute (MethodImplOptions.InternalCall)]
		internal static extern IntPtr ResolveFieldToken (IntPtr module, int token, IntPtr[] type_args, IntPtr[] method_args, 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, IntPtr[] type_args, IntPtr[] method_args, out ResolveTokenError error);

		[MethodImplAttribute (MethodImplOptions.InternalCall)]
		internal static extern byte[] ResolveSignature (IntPtr module, int metadataToken, out ResolveTokenError error);

		[MethodImplAttribute (MethodImplOptions.InternalCall)]
		internal static extern void GetPEKind (IntPtr module, out PortableExecutableKinds peKind, out ImageFileMachine machine);
	}

	internal enum ResolveTokenError {
		OutOfRange,
		BadTable,
		Other
	}	
}