mirror of https://github.com/AMT-Cheif/Stylet.git
Sort out the thread-safety
This commit is contained in:
parent
b89c61a91e
commit
af06ee5019
|
@ -4,14 +4,16 @@ using System.Linq;
|
||||||
using System.Linq.Expressions;
|
using System.Linq.Expressions;
|
||||||
using System.Reflection;
|
using System.Reflection;
|
||||||
using System.Text;
|
using System.Text;
|
||||||
|
using System.Threading;
|
||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
|
|
||||||
namespace StyletIoC
|
namespace StyletIoC
|
||||||
{
|
{
|
||||||
internal class BuilderUpper
|
internal class BuilderUpper
|
||||||
{
|
{
|
||||||
private Type type;
|
private readonly Type type;
|
||||||
private StyletIoCContainer container;
|
private readonly StyletIoCContainer container;
|
||||||
|
private readonly object implementorLock = new object();
|
||||||
private Action<object> implementor;
|
private Action<object> implementor;
|
||||||
|
|
||||||
public BuilderUpper(Type type, StyletIoCContainer container)
|
public BuilderUpper(Type type, StyletIoCContainer container)
|
||||||
|
@ -50,15 +52,18 @@ namespace StyletIoC
|
||||||
|
|
||||||
public Action<object> GetImplementor()
|
public Action<object> GetImplementor()
|
||||||
{
|
{
|
||||||
if (this.implementor != null)
|
lock (this.implementorLock)
|
||||||
|
{
|
||||||
|
if (this.implementor != null)
|
||||||
|
return this.implementor;
|
||||||
|
|
||||||
|
var parameterExpression = Expression.Parameter(typeof(object), "inputParameter");
|
||||||
|
var typedParameterExpression = Expression.Convert(parameterExpression, this.type);
|
||||||
|
var expression = this.GetExpression(typedParameterExpression);
|
||||||
|
this.implementor = Expression.Lambda<Action<object>>(this.GetExpression(typedParameterExpression), parameterExpression).Compile();
|
||||||
|
|
||||||
return this.implementor;
|
return this.implementor;
|
||||||
|
}
|
||||||
var parameterExpression = Expression.Parameter(typeof(object), "inputParameter");
|
|
||||||
var typedParameterExpression = Expression.Convert(parameterExpression, this.type);
|
|
||||||
var expression = this.GetExpression(typedParameterExpression);
|
|
||||||
this.implementor = Expression.Lambda<Action<object>>(this.GetExpression(typedParameterExpression), parameterExpression).Compile();
|
|
||||||
|
|
||||||
return this.implementor;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -4,6 +4,7 @@ using System.Linq;
|
||||||
using System.Linq.Expressions;
|
using System.Linq.Expressions;
|
||||||
using System.Reflection;
|
using System.Reflection;
|
||||||
using System.Text;
|
using System.Text;
|
||||||
|
using System.Threading;
|
||||||
using System.Threading.Tasks;
|
using System.Threading.Tasks;
|
||||||
|
|
||||||
namespace StyletIoC
|
namespace StyletIoC
|
||||||
|
@ -52,7 +53,11 @@ namespace StyletIoC
|
||||||
// Sealed so Code Analysis doesn't moan about us setting the virtual Type property
|
// Sealed so Code Analysis doesn't moan about us setting the virtual Type property
|
||||||
internal sealed class TypeCreator : CreatorBase
|
internal sealed class TypeCreator : CreatorBase
|
||||||
{
|
{
|
||||||
public string AttributeKey { get; private set; }
|
private readonly string _attributeKey;
|
||||||
|
public string AttributeKey
|
||||||
|
{
|
||||||
|
get { return this._attributeKey; }
|
||||||
|
}
|
||||||
private Expression creationExpression;
|
private Expression creationExpression;
|
||||||
|
|
||||||
public TypeCreator(Type type, StyletIoCContainer container) : base(container)
|
public TypeCreator(Type type, StyletIoCContainer container) : base(container)
|
||||||
|
@ -62,7 +67,7 @@ namespace StyletIoC
|
||||||
// Use the key from InjectAttribute (if present), and let someone else override it if they want
|
// Use the key from InjectAttribute (if present), and let someone else override it if they want
|
||||||
var attribute = (InjectAttribute)type.GetCustomAttributes(typeof(InjectAttribute), false).FirstOrDefault();
|
var attribute = (InjectAttribute)type.GetCustomAttributes(typeof(InjectAttribute), false).FirstOrDefault();
|
||||||
if (attribute != null)
|
if (attribute != null)
|
||||||
this.AttributeKey = attribute.Key;
|
this._attributeKey = attribute.Key;
|
||||||
}
|
}
|
||||||
|
|
||||||
private string KeyForParameter(ParameterInfo parameter)
|
private string KeyForParameter(ParameterInfo parameter)
|
||||||
|
@ -132,15 +137,15 @@ namespace StyletIoC
|
||||||
|
|
||||||
var completeExpression = this.CompleteExpressionFromCreator(creator);
|
var completeExpression = this.CompleteExpressionFromCreator(creator);
|
||||||
|
|
||||||
this.creationExpression = completeExpression;
|
Interlocked.CompareExchange(ref this.creationExpression, completeExpression, null);
|
||||||
return completeExpression;
|
return this.creationExpression;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Sealed for consistency with TypeCreator
|
// Sealed for consistency with TypeCreator
|
||||||
internal sealed class FactoryCreator<T> : CreatorBase
|
internal sealed class FactoryCreator<T> : CreatorBase
|
||||||
{
|
{
|
||||||
private Func<StyletIoCContainer, T> factory;
|
private readonly Func<StyletIoCContainer, T> factory;
|
||||||
private Expression instanceExpression;
|
private Expression instanceExpression;
|
||||||
|
|
||||||
public override Type Type { get { return typeof(T); } }
|
public override Type Type { get { return typeof(T); } }
|
||||||
|
@ -160,8 +165,8 @@ namespace StyletIoC
|
||||||
|
|
||||||
var completeExpression = this.CompleteExpressionFromCreator(invoked);
|
var completeExpression = this.CompleteExpressionFromCreator(invoked);
|
||||||
|
|
||||||
this.instanceExpression = completeExpression;
|
Interlocked.CompareExchange(ref this.instanceExpression, completeExpression, null);
|
||||||
return completeExpression;
|
return this.instanceExpression;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -17,10 +17,10 @@ namespace StyletIoC
|
||||||
|
|
||||||
internal abstract class RegistrationBase : IRegistration
|
internal abstract class RegistrationBase : IRegistration
|
||||||
{
|
{
|
||||||
protected ICreator creator;
|
protected readonly ICreator creator;
|
||||||
|
|
||||||
public Type Type { get { return this.creator.Type; } }
|
public Type Type { get { return this.creator.Type; } }
|
||||||
|
protected readonly object generatorLock = new object();
|
||||||
|
// Value type, so needs locked access
|
||||||
protected Func<object> generator { get; set; }
|
protected Func<object> generator { get; set; }
|
||||||
|
|
||||||
public RegistrationBase(ICreator creator)
|
public RegistrationBase(ICreator creator)
|
||||||
|
@ -44,9 +44,18 @@ namespace StyletIoC
|
||||||
|
|
||||||
public override Func<object> GetGenerator()
|
public override Func<object> GetGenerator()
|
||||||
{
|
{
|
||||||
if (this.generator == null)
|
// Compiling the generator might be expensive, but there's nothing to be gained from
|
||||||
this.generator = Expression.Lambda<Func<object>>(this.GetInstanceExpression()).Compile();
|
// doing it outside of the lock - the altnerative is having two threads compiling it in parallel,
|
||||||
return this.generator;
|
// while would take just as long and use more resources
|
||||||
|
lock (this.generatorLock)
|
||||||
|
{
|
||||||
|
if (this.generator != null)
|
||||||
|
return this.generator;
|
||||||
|
var generator = Expression.Lambda<Func<object>>(this.GetInstanceExpression()).Compile();
|
||||||
|
if (this.generator == null)
|
||||||
|
this.generator = generator;
|
||||||
|
return this.generator;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -63,17 +72,21 @@ namespace StyletIoC
|
||||||
return;
|
return;
|
||||||
|
|
||||||
// Ensure we don't end up creating two singletons, one used by each thread
|
// Ensure we don't end up creating two singletons, one used by each thread
|
||||||
Interlocked.CompareExchange(ref this.instance, Expression.Lambda<Func<object>>(this.creator.GetInstanceExpression()).Compile()(), null);
|
var instance = Expression.Lambda<Func<object>>(this.creator.GetInstanceExpression()).Compile()();
|
||||||
|
Interlocked.CompareExchange(ref this.instance, instance, null);
|
||||||
}
|
}
|
||||||
|
|
||||||
public override Func<object> GetGenerator()
|
public override Func<object> GetGenerator()
|
||||||
{
|
{
|
||||||
this.EnsureInstantiated();
|
this.EnsureInstantiated();
|
||||||
|
|
||||||
if (this.generator == null)
|
// Cheap delegate creation, so doesn't need to be outside the lock
|
||||||
this.generator = () => this.instance;
|
lock (this.generatorLock)
|
||||||
|
{
|
||||||
return this.generator;
|
if (this.generator == null)
|
||||||
|
this.generator = () => this.instance;
|
||||||
|
return this.generator;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public override Expression GetInstanceExpression()
|
public override Expression GetInstanceExpression()
|
||||||
|
@ -84,32 +97,41 @@ namespace StyletIoC
|
||||||
this.EnsureInstantiated();
|
this.EnsureInstantiated();
|
||||||
|
|
||||||
// This expression yields the actual type of instance, not 'object'
|
// This expression yields the actual type of instance, not 'object'
|
||||||
this.instanceExpression = Expression.Constant(this.instance);
|
var instanceExpression = Expression.Constant(this.instance);
|
||||||
|
Interlocked.CompareExchange(ref this.instanceExpression, instanceExpression, null);
|
||||||
return this.instanceExpression;
|
return this.instanceExpression;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
internal class GetAllRegistration : IRegistration
|
internal class GetAllRegistration : IRegistration
|
||||||
{
|
{
|
||||||
private StyletIoCContainer container;
|
private readonly StyletIoCContainer container;
|
||||||
|
|
||||||
public string Key { get; set; }
|
public string Key { get; set; }
|
||||||
public Type Type { get; private set; }
|
private readonly Type _type;
|
||||||
|
public Type Type
|
||||||
|
{
|
||||||
|
get { return this._type; }
|
||||||
|
}
|
||||||
|
|
||||||
private Expression expression;
|
private Expression expression;
|
||||||
|
private readonly object generatorLock = new object();
|
||||||
private Func<object> generator;
|
private Func<object> generator;
|
||||||
|
|
||||||
public GetAllRegistration(Type type, StyletIoCContainer container)
|
public GetAllRegistration(Type type, StyletIoCContainer container)
|
||||||
{
|
{
|
||||||
this.Type = type;
|
this._type = type;
|
||||||
this.container = container;
|
this.container = container;
|
||||||
}
|
}
|
||||||
|
|
||||||
public Func<object> GetGenerator()
|
public Func<object> GetGenerator()
|
||||||
{
|
{
|
||||||
if (this.generator == null)
|
lock (this.generatorLock)
|
||||||
this.generator = Expression.Lambda<Func<object>>(this.GetInstanceExpression()).Compile();
|
{
|
||||||
return this.generator;
|
if (this.generator == null)
|
||||||
|
this.generator = Expression.Lambda<Func<object>>(this.GetInstanceExpression()).Compile();
|
||||||
|
return this.generator;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
public Expression GetInstanceExpression()
|
public Expression GetInstanceExpression()
|
||||||
|
@ -120,7 +142,7 @@ namespace StyletIoC
|
||||||
var list = Expression.New(this.Type);
|
var list = Expression.New(this.Type);
|
||||||
var init = Expression.ListInit(list, this.container.GetRegistrations(new TypeKey(this.Type.GenericTypeArguments[0], this.Key), false).GetAll().Select(x => x.GetInstanceExpression()));
|
var init = Expression.ListInit(list, this.container.GetRegistrations(new TypeKey(this.Type.GenericTypeArguments[0], this.Key), false).GetAll().Select(x => x.GetInstanceExpression()));
|
||||||
|
|
||||||
this.expression = init;
|
Interlocked.CompareExchange(ref this.expression, init, null);
|
||||||
return this.expression;
|
return this.expression;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -16,7 +16,7 @@ namespace StyletIoC
|
||||||
|
|
||||||
internal class SingleRegistration : IRegistrationCollection
|
internal class SingleRegistration : IRegistrationCollection
|
||||||
{
|
{
|
||||||
private IRegistration registration;
|
private readonly IRegistration registration;
|
||||||
|
|
||||||
public SingleRegistration(IRegistration registration)
|
public SingleRegistration(IRegistration registration)
|
||||||
{
|
{
|
||||||
|
@ -43,7 +43,8 @@ namespace StyletIoC
|
||||||
|
|
||||||
internal class RegistrationCollection : IRegistrationCollection
|
internal class RegistrationCollection : IRegistrationCollection
|
||||||
{
|
{
|
||||||
private List<IRegistration> registrations;
|
private readonly object registrationsLock = new object();
|
||||||
|
private readonly List<IRegistration> registrations;
|
||||||
|
|
||||||
public RegistrationCollection(List<IRegistration> registrations)
|
public RegistrationCollection(List<IRegistration> registrations)
|
||||||
{
|
{
|
||||||
|
@ -58,14 +59,14 @@ namespace StyletIoC
|
||||||
public List<IRegistration> GetAll()
|
public List<IRegistration> GetAll()
|
||||||
{
|
{
|
||||||
List<IRegistration> registrationsCopy;
|
List<IRegistration> registrationsCopy;
|
||||||
lock (this.registrations) { registrationsCopy = registrations.ToList(); }
|
lock (this.registrationsLock) { registrationsCopy = registrations.ToList(); }
|
||||||
return registrationsCopy;
|
return registrationsCopy;
|
||||||
}
|
}
|
||||||
|
|
||||||
public IRegistrationCollection AddRegistration(IRegistration registration)
|
public IRegistrationCollection AddRegistration(IRegistration registration)
|
||||||
{
|
{
|
||||||
// Need to lock the list, as someone might be fetching from it while we do this
|
// Need to lock the list, as someone might be fetching from it while we do this
|
||||||
lock (this.registrations)
|
lock (this.registrationsLock)
|
||||||
{
|
{
|
||||||
// Should have been caught by SingleRegistration.AddRegistration
|
// Should have been caught by SingleRegistration.AddRegistration
|
||||||
Debug.Assert(!this.registrations.Any(x => x.Type == registration.Type));
|
Debug.Assert(!this.registrations.Any(x => x.Type == registration.Type));
|
||||||
|
|
|
@ -100,9 +100,9 @@ namespace StyletIoC
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Maps a [type, key] pair, where 'type' is an unbound generic (something like IValidator{}) to something which, given a type, can create an IRegistration for that type.
|
/// Maps a [type, key] pair, where 'type' is an unbound generic (something like IValidator{}) to something which, given a type, can create an IRegistration for that type.
|
||||||
/// So if they've bound an IValidator{} to a an IntValidator, StringValidator, etc, and request an IValidator{string}, one of the UnboundGenerics here can generatr a StringValidator.
|
/// So if they've bound an IValidator{} to a an IntValidator, StringValidator, etc, and request an IValidator{string}, one of the UnboundGenerics here can generatr a StringValidator.
|
||||||
/// Type-safety with the List is ensured by locking using the list as the lock object before modifying / iterating.
|
|
||||||
/// </summary>
|
/// </summary>
|
||||||
private readonly ConcurrentDictionary<TypeKey, List<UnboundGeneric>> unboundGenerics = new ConcurrentDictionary<TypeKey, List<UnboundGeneric>>();
|
/// <remarks>Dictionary{TKey, TValue} and List{T} are thread-safe for concurrent reads, which is all that happens after building</remarks>
|
||||||
|
private readonly Dictionary<TypeKey, List<UnboundGeneric>> unboundGenerics = new Dictionary<TypeKey, List<UnboundGeneric>>();
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Maps a type onto a BuilderUpper for that type, which can create an Expresson/Delegate to build up that type.
|
/// Maps a type onto a BuilderUpper for that type, which can create an Expresson/Delegate to build up that type.
|
||||||
|
@ -321,41 +321,37 @@ namespace StyletIoC
|
||||||
if (!this.unboundGenerics.TryGetValue(new TypeKey(unboundGenericType, typeKey.Key), out unboundGenerics))
|
if (!this.unboundGenerics.TryGetValue(new TypeKey(unboundGenericType, typeKey.Key), out unboundGenerics))
|
||||||
return false;
|
return false;
|
||||||
|
|
||||||
// Need to lock this, as someone might modify the underying list by registering a new unbound generic
|
foreach (var unboundGeneric in unboundGenerics)
|
||||||
lock (unboundGenerics)
|
|
||||||
{
|
{
|
||||||
foreach (var unboundGeneric in unboundGenerics)
|
// Consider this scenario:
|
||||||
|
// interface IC<T, U> { } class C<T, U> : IC<U, T> { }
|
||||||
|
// Then they ask for an IC<int, bool>. We need to give them a C<bool, int>
|
||||||
|
// Search the ancestry of C for an IC (called implOfUnboundGenericType), then create a mapping which says that
|
||||||
|
// U is a bool and T is an int by comparing this against 'type' - the IC<T, U> that's registered as the service
|
||||||
|
// Then use this when making the type for C
|
||||||
|
|
||||||
|
Type newType;
|
||||||
|
if (unboundGeneric.Type == unboundGenericType)
|
||||||
{
|
{
|
||||||
// Consider this scenario:
|
newType = type;
|
||||||
// interface IC<T, U> { } class C<T, U> : IC<U, T> { }
|
|
||||||
// Then they ask for an IC<int, bool>. We need to give them a C<bool, int>
|
|
||||||
// Search the ancestry of C for an IC (called implOfUnboundGenericType), then create a mapping which says that
|
|
||||||
// U is a bool and T is an int by comparing this against 'type' - the IC<T, U> that's registered as the service
|
|
||||||
// Then use this when making the type for C
|
|
||||||
|
|
||||||
Type newType;
|
|
||||||
if (unboundGeneric.Type == unboundGenericType)
|
|
||||||
{
|
|
||||||
newType = type;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
var implOfUnboundGenericType = unboundGeneric.Type.GetBaseTypesAndInterfaces().Single(x => x.Name == unboundGenericType.Name);
|
|
||||||
var mapping = implOfUnboundGenericType.GenericTypeArguments.Zip(type.GenericTypeArguments, (n, t) => new { Type = t, Name = n });
|
|
||||||
|
|
||||||
newType = unboundGeneric.Type.MakeGenericType(unboundGeneric.Type.GetTypeInfo().GenericTypeParameters.Select(x => mapping.Single(t => t.Name.Name == x.Name).Type).ToArray());
|
|
||||||
}
|
|
||||||
|
|
||||||
// The binder should have made sure of this
|
|
||||||
Debug.Assert(type.IsAssignableFrom(newType));
|
|
||||||
|
|
||||||
// Right! We've made a new generic type we can use
|
|
||||||
var registration = unboundGeneric.CreateRegistrationForType(newType);
|
|
||||||
|
|
||||||
// AddRegistration returns the IRegistrationCollection which was added/updated, so the one returned from the final
|
|
||||||
// call to AddRegistration is the final IRegistrationCollection for this key
|
|
||||||
registrations = this.AddRegistration(typeKey, registration);
|
|
||||||
}
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
var implOfUnboundGenericType = unboundGeneric.Type.GetBaseTypesAndInterfaces().Single(x => x.Name == unboundGenericType.Name);
|
||||||
|
var mapping = implOfUnboundGenericType.GenericTypeArguments.Zip(type.GenericTypeArguments, (n, t) => new { Type = t, Name = n });
|
||||||
|
|
||||||
|
newType = unboundGeneric.Type.MakeGenericType(unboundGeneric.Type.GetTypeInfo().GenericTypeParameters.Select(x => mapping.Single(t => t.Name.Name == x.Name).Type).ToArray());
|
||||||
|
}
|
||||||
|
|
||||||
|
// The binder should have made sure of this
|
||||||
|
Debug.Assert(type.IsAssignableFrom(newType));
|
||||||
|
|
||||||
|
// Right! We've made a new generic type we can use
|
||||||
|
var registration = unboundGeneric.CreateRegistrationForType(newType);
|
||||||
|
|
||||||
|
// AddRegistration returns the IRegistrationCollection which was added/updated, so the one returned from the final
|
||||||
|
// call to AddRegistration is the final IRegistrationCollection for this key
|
||||||
|
registrations = this.AddRegistration(typeKey, registration);
|
||||||
}
|
}
|
||||||
|
|
||||||
return registrations != null;
|
return registrations != null;
|
||||||
|
@ -408,25 +404,23 @@ namespace StyletIoC
|
||||||
internal void AddUnboundGeneric(TypeKey typeKey, UnboundGeneric unboundGeneric)
|
internal void AddUnboundGeneric(TypeKey typeKey, UnboundGeneric unboundGeneric)
|
||||||
{
|
{
|
||||||
// We're not worried about thread-safety across multiple calls to this function (as it's only called as part of setup, which we're
|
// We're not worried about thread-safety across multiple calls to this function (as it's only called as part of setup, which we're
|
||||||
// not thread-safe about). However someone might be fetching something from this list while we're modifying it, which we need to avoid
|
// not thread-safe about).
|
||||||
var unboundGenerics = this.unboundGenerics.GetOrAdd(typeKey, x => new List<UnboundGeneric>());
|
List<UnboundGeneric> unboundGenerics;
|
||||||
lock (unboundGenerics)
|
if (!this.unboundGenerics.TryGetValue(typeKey, out unboundGenerics))
|
||||||
{
|
{
|
||||||
// Is there an existing registration for this type?
|
unboundGenerics = new List<UnboundGeneric>();
|
||||||
if (unboundGenerics.Any(x => x.Type == unboundGeneric.Type))
|
this.unboundGenerics.Add(typeKey, unboundGenerics);
|
||||||
throw new StyletIoCRegistrationException(String.Format("Multiple registrations for type {0} found", typeKey.Type.Description()));
|
|
||||||
|
|
||||||
unboundGenerics.Add(unboundGeneric);
|
|
||||||
}
|
}
|
||||||
|
// Is there an existing registration for this type?
|
||||||
|
if (unboundGenerics.Any(x => x.Type == unboundGeneric.Type))
|
||||||
|
throw new StyletIoCRegistrationException(String.Format("Multiple registrations for type {0} found", typeKey.Type.Description()));
|
||||||
|
|
||||||
|
unboundGenerics.Add(unboundGeneric);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// </summary>
|
|
||||||
/// <remarks>Not thread-safe, as it's only ever called from the builder</remarks>
|
|
||||||
/// <param name="serviceType"></param>
|
|
||||||
/// <returns></returns>
|
|
||||||
internal Type GetFactoryForType(Type serviceType)
|
internal Type GetFactoryForType(Type serviceType)
|
||||||
{
|
{
|
||||||
|
// Not thread-safe, as it's only called from the builder
|
||||||
if (!serviceType.IsInterface)
|
if (!serviceType.IsInterface)
|
||||||
throw new StyletIoCCreateFactoryException(String.Format("Unable to create a factory implementing type {0}, as it isn't an interface", serviceType.Description()));
|
throw new StyletIoCCreateFactoryException(String.Format("Unable to create a factory implementing type {0}, as it isn't an interface", serviceType.Description()));
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue