mirror of https://github.com/AMT-Cheif/Stylet.git
Fix suspected race condition in SingletonRegistration
This could trigger if the first two requests for a singleton were on different threads, and executed veyr close together
This commit is contained in:
parent
4fa7b9ca1a
commit
24ab49208f
|
@ -12,8 +12,8 @@ namespace StyletIoC.Internal.Registrations
|
|||
protected readonly ICreator Creator;
|
||||
public RuntimeTypeHandle TypeHandle { get { return this.Creator.TypeHandle; } }
|
||||
|
||||
private readonly object generatorLock = new object();
|
||||
private Func<IRegistrationContext, object> generator;
|
||||
protected readonly object lockObject = new object();
|
||||
protected Func<IRegistrationContext, object> generator;
|
||||
|
||||
protected RegistrationBase(ICreator creator)
|
||||
{
|
||||
|
@ -25,9 +25,10 @@ namespace StyletIoC.Internal.Registrations
|
|||
if (this.generator != null)
|
||||
return this.generator;
|
||||
|
||||
lock (this.generatorLock)
|
||||
lock (this.lockObject)
|
||||
{
|
||||
this.generator = this.GetGeneratorInternal();
|
||||
if (this.generator == null)
|
||||
this.generator = this.GetGeneratorInternal();
|
||||
return this.generator;
|
||||
}
|
||||
}
|
||||
|
@ -38,14 +39,6 @@ namespace StyletIoC.Internal.Registrations
|
|||
return Expression.Lambda<Func<IRegistrationContext, object>>(this.GetInstanceExpression(registrationContext), registrationContext).Compile();
|
||||
}
|
||||
|
||||
protected void ClearGenerator()
|
||||
{
|
||||
lock (this.generatorLock)
|
||||
{
|
||||
this.generator = null;
|
||||
}
|
||||
}
|
||||
|
||||
public abstract Expression GetInstanceExpression(ParameterExpression registrationContext);
|
||||
}
|
||||
}
|
||||
|
|
|
@ -11,7 +11,6 @@ namespace StyletIoC.Internal.Registrations
|
|||
internal class SingletonRegistration : RegistrationBase
|
||||
{
|
||||
private readonly IRegistrationContext parentContext;
|
||||
private Expression instanceExpression;
|
||||
private object instance;
|
||||
|
||||
public SingletonRegistration(IRegistrationContext parentContext, ICreator creator)
|
||||
|
@ -20,26 +19,32 @@ namespace StyletIoC.Internal.Registrations
|
|||
this.parentContext = parentContext;
|
||||
this.parentContext.Disposing += (o, e) =>
|
||||
{
|
||||
var disposable = this.instance as IDisposable;
|
||||
IDisposable disposable;
|
||||
lock (this.lockObject)
|
||||
{
|
||||
disposable = this.instance as IDisposable;
|
||||
this.instance = null;
|
||||
this.generator = null;
|
||||
}
|
||||
if (disposable != null)
|
||||
disposable.Dispose();
|
||||
|
||||
this.instance = this.instanceExpression = null;
|
||||
this.ClearGenerator();
|
||||
};
|
||||
}
|
||||
|
||||
public override Expression GetInstanceExpression(ParameterExpression registrationContext)
|
||||
{
|
||||
if (this.instanceExpression != null)
|
||||
return this.instanceExpression;
|
||||
|
||||
this.instance = Expression.Lambda<Func<IRegistrationContext, object>>(this.Creator.GetInstanceExpression(registrationContext), registrationContext).Compile()(this.parentContext);
|
||||
if (this.instance == null)
|
||||
{
|
||||
lock (this.lockObject)
|
||||
{
|
||||
if (this.instance == null)
|
||||
this.instance = Expression.Lambda<Func<IRegistrationContext, object>>(this.Creator.GetInstanceExpression(registrationContext), registrationContext).Compile()(this.parentContext);
|
||||
}
|
||||
}
|
||||
|
||||
// This expression yields the actual type of instance, not 'object'
|
||||
var instanceExpression = Expression.Constant(this.instance);
|
||||
Interlocked.CompareExchange(ref this.instanceExpression, instanceExpression, null);
|
||||
return this.instanceExpression;
|
||||
return instanceExpression;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
|
Loading…
Reference in New Issue