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