mirror of https://github.com/icsharpcode/ILSpy.git
				
				
			
			You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							282 lines
						
					
					
						
							7.6 KiB
						
					
					
				
			
		
		
	
	
							282 lines
						
					
					
						
							7.6 KiB
						
					
					
				// Copyright (c) 2010-2013 AlphaSierraPapa for the SharpDevelop Team | 
						|
//  | 
						|
// 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.Generic; | 
						|
using System.Linq; | 
						|
using System.Threading; | 
						|
 | 
						|
namespace ICSharpCode.Decompiler.CSharp.Syntax | 
						|
{ | 
						|
	/// <summary> | 
						|
	/// Provides an interface to handle annotations in an object. | 
						|
	/// </summary> | 
						|
	public interface IAnnotatable | 
						|
	{ | 
						|
		/// <summary> | 
						|
		/// Gets all annotations stored on this IAnnotatable. | 
						|
		/// </summary> | 
						|
		IEnumerable<object> Annotations { | 
						|
			get; | 
						|
		} | 
						|
 | 
						|
		/// <summary> | 
						|
		/// Gets the first annotation of the specified type. | 
						|
		/// Returns null if no matching annotation exists. | 
						|
		/// </summary> | 
						|
		/// <typeparam name='T'> | 
						|
		/// The type of the annotation. | 
						|
		/// </typeparam> | 
						|
		T Annotation<T>() where T : class; | 
						|
 | 
						|
		/// <summary> | 
						|
		/// Gets the first annotation of the specified type. | 
						|
		/// Returns null if no matching annotation exists. | 
						|
		/// </summary> | 
						|
		/// <param name='type'> | 
						|
		/// The type of the annotation. | 
						|
		/// </param> | 
						|
		object Annotation(Type type); | 
						|
 | 
						|
		/// <summary> | 
						|
		/// Adds an annotation to this instance. | 
						|
		/// </summary> | 
						|
		/// <param name='annotation'> | 
						|
		/// The annotation to add. | 
						|
		/// </param> | 
						|
		void AddAnnotation(object annotation); | 
						|
 | 
						|
		/// <summary> | 
						|
		/// Removes all annotations of the specified type. | 
						|
		/// </summary> | 
						|
		/// <typeparam name='T'> | 
						|
		/// The type of the annotations to remove. | 
						|
		/// </typeparam> | 
						|
		void RemoveAnnotations<T>() where T : class; | 
						|
 | 
						|
		/// <summary> | 
						|
		/// Removes all annotations of the specified type. | 
						|
		/// </summary> | 
						|
		/// <param name='type'> | 
						|
		/// The type of the annotations to remove. | 
						|
		/// </param> | 
						|
		void RemoveAnnotations(Type type); | 
						|
	} | 
						|
 | 
						|
	/// <summary> | 
						|
	/// Base class used to implement the IAnnotatable interface. | 
						|
	/// This implementation is thread-safe. | 
						|
	/// </summary> | 
						|
	[Serializable] | 
						|
	public abstract class AbstractAnnotatable : IAnnotatable | 
						|
	{ | 
						|
		// Annotations: points either null (no annotations), to the single annotation, | 
						|
		// or to an AnnotationList. | 
						|
		// Once it is pointed at an AnnotationList, it will never change (this allows thread-safety support by locking the list) | 
						|
 | 
						|
		object annotations; | 
						|
 | 
						|
		/// <summary> | 
						|
		/// Clones all annotations. | 
						|
		/// This method is intended to be called by Clone() implementations in derived classes. | 
						|
		/// <code> | 
						|
		/// AstNode copy = (AstNode)MemberwiseClone(); | 
						|
		/// copy.CloneAnnotations(); | 
						|
		/// </code> | 
						|
		/// </summary> | 
						|
		protected void CloneAnnotations() | 
						|
		{ | 
						|
			ICloneable cloneable = annotations as ICloneable; | 
						|
			if (cloneable != null) | 
						|
				annotations = cloneable.Clone(); | 
						|
		} | 
						|
 | 
						|
		sealed class AnnotationList : List<object>, ICloneable | 
						|
		{ | 
						|
			// There are two uses for this custom list type: | 
						|
			// 1) it's private, and thus (unlike List<object>) cannot be confused with real annotations | 
						|
			// 2) It allows us to simplify the cloning logic by making the list behave the same as a clonable annotation. | 
						|
			public AnnotationList(int initialCapacity) : base(initialCapacity) | 
						|
			{ | 
						|
			} | 
						|
 | 
						|
			public object Clone() | 
						|
			{ | 
						|
				lock (this) | 
						|
				{ | 
						|
					AnnotationList copy = new AnnotationList(this.Count); | 
						|
					for (int i = 0; i < this.Count; i++) | 
						|
					{ | 
						|
						object obj = this[i]; | 
						|
						ICloneable c = obj as ICloneable; | 
						|
						copy.Add(c != null ? c.Clone() : obj); | 
						|
					} | 
						|
					return copy; | 
						|
				} | 
						|
			} | 
						|
		} | 
						|
 | 
						|
		public virtual void AddAnnotation(object annotation) | 
						|
		{ | 
						|
			if (annotation == null) | 
						|
				throw new ArgumentNullException(nameof(annotation)); | 
						|
			retry: // Retry until successful | 
						|
			object oldAnnotation = Interlocked.CompareExchange(ref this.annotations, annotation, null); | 
						|
			if (oldAnnotation == null) | 
						|
			{ | 
						|
				return; // we successfully added a single annotation | 
						|
			} | 
						|
			AnnotationList list = oldAnnotation as AnnotationList; | 
						|
			if (list == null) | 
						|
			{ | 
						|
				// we need to transform the old annotation into a list | 
						|
				list = new AnnotationList(4); | 
						|
				list.Add(oldAnnotation); | 
						|
				list.Add(annotation); | 
						|
				if (Interlocked.CompareExchange(ref this.annotations, list, oldAnnotation) != oldAnnotation) | 
						|
				{ | 
						|
					// the transformation failed (some other thread wrote to this.annotations first) | 
						|
					goto retry; | 
						|
				} | 
						|
			} | 
						|
			else | 
						|
			{ | 
						|
				// once there's a list, use simple locking | 
						|
				lock (list) | 
						|
				{ | 
						|
					list.Add(annotation); | 
						|
				} | 
						|
			} | 
						|
		} | 
						|
 | 
						|
		public virtual void RemoveAnnotations<T>() where T : class | 
						|
		{ | 
						|
			retry: // Retry until successful | 
						|
			object oldAnnotations = this.annotations; | 
						|
			AnnotationList list = oldAnnotations as AnnotationList; | 
						|
			if (list != null) | 
						|
			{ | 
						|
				lock (list) | 
						|
					list.RemoveAll(obj => obj is T); | 
						|
			} | 
						|
			else if (oldAnnotations is T) | 
						|
			{ | 
						|
				if (Interlocked.CompareExchange(ref this.annotations, null, oldAnnotations) != oldAnnotations) | 
						|
				{ | 
						|
					// Operation failed (some other thread wrote to this.annotations first) | 
						|
					goto retry; | 
						|
				} | 
						|
			} | 
						|
		} | 
						|
 | 
						|
		public virtual void RemoveAnnotations(Type type) | 
						|
		{ | 
						|
			if (type == null) | 
						|
				throw new ArgumentNullException(nameof(type)); | 
						|
			retry: // Retry until successful | 
						|
			object oldAnnotations = this.annotations; | 
						|
			AnnotationList list = oldAnnotations as AnnotationList; | 
						|
			if (list != null) | 
						|
			{ | 
						|
				lock (list) | 
						|
					list.RemoveAll(type.IsInstanceOfType); | 
						|
			} | 
						|
			else if (type.IsInstanceOfType(oldAnnotations)) | 
						|
			{ | 
						|
				if (Interlocked.CompareExchange(ref this.annotations, null, oldAnnotations) != oldAnnotations) | 
						|
				{ | 
						|
					// Operation failed (some other thread wrote to this.annotations first) | 
						|
					goto retry; | 
						|
				} | 
						|
			} | 
						|
		} | 
						|
 | 
						|
		public T Annotation<T>() where T : class | 
						|
		{ | 
						|
			object annotations = this.annotations; | 
						|
			AnnotationList list = annotations as AnnotationList; | 
						|
			if (list != null) | 
						|
			{ | 
						|
				lock (list) | 
						|
				{ | 
						|
					foreach (object obj in list) | 
						|
					{ | 
						|
						T t = obj as T; | 
						|
						if (t != null) | 
						|
							return t; | 
						|
					} | 
						|
					return null; | 
						|
				} | 
						|
			} | 
						|
			else | 
						|
			{ | 
						|
				return annotations as T; | 
						|
			} | 
						|
		} | 
						|
 | 
						|
		public object Annotation(Type type) | 
						|
		{ | 
						|
			if (type == null) | 
						|
				throw new ArgumentNullException(nameof(type)); | 
						|
			object annotations = this.annotations; | 
						|
			AnnotationList list = annotations as AnnotationList; | 
						|
			if (list != null) | 
						|
			{ | 
						|
				lock (list) | 
						|
				{ | 
						|
					foreach (object obj in list) | 
						|
					{ | 
						|
						if (type.IsInstanceOfType(obj)) | 
						|
							return obj; | 
						|
					} | 
						|
				} | 
						|
			} | 
						|
			else | 
						|
			{ | 
						|
				if (type.IsInstanceOfType(annotations)) | 
						|
					return annotations; | 
						|
			} | 
						|
			return null; | 
						|
		} | 
						|
 | 
						|
		/// <summary> | 
						|
		/// Gets all annotations stored on this AstNode. | 
						|
		/// </summary> | 
						|
		public IEnumerable<object> Annotations { | 
						|
			get { | 
						|
				object annotations = this.annotations; | 
						|
				AnnotationList list = annotations as AnnotationList; | 
						|
				if (list != null) | 
						|
				{ | 
						|
					lock (list) | 
						|
					{ | 
						|
						return list.ToArray(); | 
						|
					} | 
						|
				} | 
						|
				else | 
						|
				{ | 
						|
					if (annotations != null) | 
						|
						return new object[] { annotations }; | 
						|
					else | 
						|
						return Enumerable.Empty<object>(); | 
						|
				} | 
						|
			} | 
						|
		} | 
						|
	} | 
						|
} | 
						|
 | 
						|
 |