Sort out the thread-safety

This commit is contained in:
Antony Male 2014-05-11 22:35:21 +01:00
parent b89c61a91e
commit af06ee5019
5 changed files with 114 additions and 87 deletions

View File

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

View File

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

View File

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

View File

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

View File

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