| 
									
										
										
										
											2014-08-13 10:39:27 +01:00
										 |  |  | /* | 
					
						
							|  |  |  |   Copyright (C) 2009-2011 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.Runtime.CompilerServices; | 
					
						
							|  |  |  | using IKVM.Reflection.Metadata; | 
					
						
							|  |  |  | using IKVM.Reflection.Writer; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace IKVM.Reflection.Emit | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  | 	public sealed class EventBuilder : EventInfo | 
					
						
							|  |  |  | 	{ | 
					
						
							|  |  |  | 		private readonly TypeBuilder typeBuilder; | 
					
						
							|  |  |  | 		private readonly string name; | 
					
						
							|  |  |  | 		private EventAttributes attributes; | 
					
						
							|  |  |  | 		private readonly int eventtype; | 
					
						
							|  |  |  | 		private MethodBuilder addOnMethod; | 
					
						
							|  |  |  | 		private MethodBuilder removeOnMethod; | 
					
						
							|  |  |  | 		private MethodBuilder fireMethod; | 
					
						
							|  |  |  | 		private readonly List<Accessor> accessors = new List<Accessor>(); | 
					
						
							|  |  |  | 		private int lazyPseudoToken; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		private struct Accessor | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			internal short Semantics; | 
					
						
							|  |  |  | 			internal MethodBuilder Method; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		internal EventBuilder(TypeBuilder typeBuilder, string name, EventAttributes attributes, Type eventtype) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			this.typeBuilder = typeBuilder; | 
					
						
							|  |  |  | 			this.name = name; | 
					
						
							|  |  |  | 			this.attributes = attributes; | 
					
						
							|  |  |  | 			this.eventtype = typeBuilder.ModuleBuilder.GetTypeTokenForMemberRef(eventtype); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public void SetAddOnMethod(MethodBuilder mdBuilder) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			addOnMethod = mdBuilder; | 
					
						
							|  |  |  | 			Accessor acc; | 
					
						
							|  |  |  | 			acc.Semantics = MethodSemanticsTable.AddOn; | 
					
						
							|  |  |  | 			acc.Method = mdBuilder; | 
					
						
							|  |  |  | 			accessors.Add(acc); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public void SetRemoveOnMethod(MethodBuilder mdBuilder) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			removeOnMethod = mdBuilder; | 
					
						
							|  |  |  | 			Accessor acc; | 
					
						
							|  |  |  | 			acc.Semantics = MethodSemanticsTable.RemoveOn; | 
					
						
							|  |  |  | 			acc.Method = mdBuilder; | 
					
						
							|  |  |  | 			accessors.Add(acc); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public void SetRaiseMethod(MethodBuilder mdBuilder) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			fireMethod = mdBuilder; | 
					
						
							|  |  |  | 			Accessor acc; | 
					
						
							|  |  |  | 			acc.Semantics = MethodSemanticsTable.Fire; | 
					
						
							|  |  |  | 			acc.Method = mdBuilder; | 
					
						
							|  |  |  | 			accessors.Add(acc); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public void AddOtherMethod(MethodBuilder mdBuilder) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			Accessor acc; | 
					
						
							|  |  |  | 			acc.Semantics = MethodSemanticsTable.Other; | 
					
						
							|  |  |  | 			acc.Method = mdBuilder; | 
					
						
							|  |  |  | 			accessors.Add(acc); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public void SetCustomAttribute(ConstructorInfo con, byte[] binaryAttribute) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			SetCustomAttribute(new CustomAttributeBuilder(con, binaryAttribute)); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public void SetCustomAttribute(CustomAttributeBuilder customBuilder) | 
					
						
							|  |  |  | 		{ | 
					
						
							| 
									
										
										
										
											2015-08-26 07:17:56 -04:00
										 |  |  | 			if (customBuilder.KnownCA == KnownCA.SpecialNameAttribute) | 
					
						
							| 
									
										
										
										
											2014-08-13 10:39:27 +01:00
										 |  |  | 			{ | 
					
						
							|  |  |  | 				attributes |= EventAttributes.SpecialName; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			else | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				if (lazyPseudoToken == 0) | 
					
						
							|  |  |  | 				{ | 
					
						
							|  |  |  | 					lazyPseudoToken = typeBuilder.ModuleBuilder.AllocPseudoToken(); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				typeBuilder.ModuleBuilder.SetCustomAttribute(lazyPseudoToken, customBuilder); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public override EventAttributes Attributes | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			get { return attributes; } | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public override MethodInfo GetAddMethod(bool nonPublic) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			return nonPublic || (addOnMethod != null && addOnMethod.IsPublic) ? addOnMethod : null; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public override MethodInfo GetRemoveMethod(bool nonPublic) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			return nonPublic || (removeOnMethod != null && removeOnMethod.IsPublic) ? removeOnMethod : null; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public override MethodInfo GetRaiseMethod(bool nonPublic) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			return nonPublic || (fireMethod != null && fireMethod.IsPublic) ? fireMethod : null; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public override MethodInfo[] GetOtherMethods(bool nonPublic) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			List<MethodInfo> list = new List<MethodInfo>(); | 
					
						
							|  |  |  | 			foreach (Accessor acc in accessors) | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				if (acc.Semantics == MethodSemanticsTable.Other && (nonPublic || acc.Method.IsPublic)) | 
					
						
							|  |  |  | 				{ | 
					
						
							|  |  |  | 					list.Add(acc.Method); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			return list.ToArray(); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public override MethodInfo[] __GetMethods() | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			List<MethodInfo> list = new List<MethodInfo>(); | 
					
						
							|  |  |  | 			foreach (Accessor acc in accessors) | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				list.Add(acc.Method); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			return list.ToArray(); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public override Type DeclaringType | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			get { return typeBuilder; } | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public override string Name | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			get { return name; } | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public override Module Module | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			get { return typeBuilder.ModuleBuilder; } | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public EventToken GetEventToken() | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			if (lazyPseudoToken == 0) | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				lazyPseudoToken = typeBuilder.ModuleBuilder.AllocPseudoToken(); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			return new EventToken(lazyPseudoToken); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		public override Type EventHandlerType | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			get { return typeBuilder.ModuleBuilder.ResolveType(eventtype); } | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		internal void Bake() | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			EventTable.Record rec = new EventTable.Record(); | 
					
						
							|  |  |  | 			rec.EventFlags = (short)attributes; | 
					
						
							|  |  |  | 			rec.Name = typeBuilder.ModuleBuilder.Strings.Add(name); | 
					
						
							|  |  |  | 			rec.EventType = eventtype; | 
					
						
							|  |  |  | 			int token = 0x14000000 | typeBuilder.ModuleBuilder.Event.AddRecord(rec); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			if (lazyPseudoToken == 0) | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				lazyPseudoToken = token; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			else | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				typeBuilder.ModuleBuilder.RegisterTokenFixup(lazyPseudoToken, token); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 			foreach (Accessor acc in accessors) | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				AddMethodSemantics(acc.Semantics, acc.Method.MetadataToken, token); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		private void AddMethodSemantics(short semantics, int methodToken, int propertyToken) | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			MethodSemanticsTable.Record rec = new MethodSemanticsTable.Record(); | 
					
						
							|  |  |  | 			rec.Semantics = semantics; | 
					
						
							|  |  |  | 			rec.Method = methodToken; | 
					
						
							|  |  |  | 			rec.Association = propertyToken; | 
					
						
							|  |  |  | 			typeBuilder.ModuleBuilder.MethodSemantics.AddRecord(rec); | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		internal override bool IsPublic | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			get | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				foreach (Accessor acc in accessors) | 
					
						
							|  |  |  | 				{ | 
					
						
							|  |  |  | 					if (acc.Method.IsPublic) | 
					
						
							|  |  |  | 					{ | 
					
						
							|  |  |  | 						return true; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				return false; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		internal override bool IsNonPrivate | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			get | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				foreach (Accessor acc in accessors) | 
					
						
							|  |  |  | 				{ | 
					
						
							|  |  |  | 					if ((acc.Method.Attributes & MethodAttributes.MemberAccessMask) > MethodAttributes.Private) | 
					
						
							|  |  |  | 					{ | 
					
						
							|  |  |  | 						return true; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				return false; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		internal override bool IsStatic | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			get | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				foreach (Accessor acc in accessors) | 
					
						
							|  |  |  | 				{ | 
					
						
							|  |  |  | 					if (acc.Method.IsStatic) | 
					
						
							|  |  |  | 					{ | 
					
						
							|  |  |  | 						return true; | 
					
						
							|  |  |  | 					} | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 				return false; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		internal override bool IsBaked | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			get { return typeBuilder.IsBaked; } | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		internal override int GetCurrentToken() | 
					
						
							|  |  |  | 		{ | 
					
						
							|  |  |  | 			if (typeBuilder.ModuleBuilder.IsSaved && ModuleBuilder.IsPseudoToken(lazyPseudoToken)) | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				return typeBuilder.ModuleBuilder.ResolvePseudoToken(lazyPseudoToken); | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			else | 
					
						
							|  |  |  | 			{ | 
					
						
							|  |  |  | 				return lazyPseudoToken; | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | } |