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.
209 lines
5.5 KiB
209 lines
5.5 KiB
#if FAT |
|
|
|
using System; |
|
|
|
namespace LinqInternal.Threading.Needles |
|
{ |
|
[Serializable] |
|
[System.Diagnostics.DebuggerNonUserCode] |
|
internal class LazyDisposableNeedle<T> : LazyNeedle<T> |
|
where T : IDisposable |
|
{ |
|
private int _status; |
|
|
|
public LazyDisposableNeedle(Func<T> valueFactory) |
|
: base(valueFactory) |
|
{ |
|
//Empty |
|
} |
|
|
|
public LazyDisposableNeedle(T target) |
|
: base(target) |
|
{ |
|
//Empty |
|
} |
|
|
|
[System.Diagnostics.DebuggerNonUserCode] |
|
~LazyDisposableNeedle() |
|
{ |
|
try |
|
{ |
|
// Empty |
|
} |
|
finally |
|
{ |
|
Dispose(false); |
|
} |
|
} |
|
|
|
public bool IsDisposed |
|
{ |
|
get { return _status == -1; } |
|
} |
|
|
|
[System.Diagnostics.DebuggerNonUserCode] |
|
public void Dispose() |
|
{ |
|
try |
|
{ |
|
Dispose(true); |
|
} |
|
finally |
|
{ |
|
GC.SuppressFinalize(this); |
|
} |
|
} |
|
|
|
[System.Diagnostics.DebuggerNonUserCode] |
|
public void DisposedConditional(Action whenDisposed, Action whenNotDisposed) |
|
{ |
|
if (_status == -1) |
|
{ |
|
if (whenDisposed != null) |
|
{ |
|
whenDisposed.Invoke(); |
|
} |
|
} |
|
else |
|
{ |
|
if (whenNotDisposed != null) |
|
{ |
|
if (ThreadingHelper.SpinWaitRelativeSet(ref _status, 1, -1)) |
|
{ |
|
try |
|
{ |
|
whenNotDisposed.Invoke(); |
|
} |
|
finally |
|
{ |
|
System.Threading.Interlocked.Decrement(ref _status); |
|
} |
|
} |
|
else |
|
{ |
|
if (whenDisposed != null) |
|
{ |
|
whenDisposed.Invoke(); |
|
} |
|
} |
|
} |
|
} |
|
} |
|
|
|
[System.Diagnostics.DebuggerNonUserCode] |
|
public TReturn DisposedConditional<TReturn>(Func<TReturn> whenDisposed, Func<TReturn> whenNotDisposed) |
|
{ |
|
if (_status == -1) |
|
{ |
|
if (whenDisposed == null) |
|
{ |
|
return default(TReturn); |
|
} |
|
return whenDisposed.Invoke(); |
|
} |
|
if (whenNotDisposed == null) |
|
{ |
|
return default(TReturn); |
|
} |
|
if (ThreadingHelper.SpinWaitRelativeSet(ref _status, 1, -1)) |
|
{ |
|
try |
|
{ |
|
return whenNotDisposed.Invoke(); |
|
} |
|
finally |
|
{ |
|
System.Threading.Interlocked.Decrement(ref _status); |
|
} |
|
} |
|
if (whenDisposed == null) |
|
{ |
|
return default(TReturn); |
|
} |
|
return whenDisposed.Invoke(); |
|
} |
|
|
|
public override void Initialize() |
|
{ |
|
Initialize(() => UnDispose()); |
|
} |
|
|
|
public void Reinitialize() |
|
{ |
|
OnDispose(); |
|
Initialize(); |
|
} |
|
|
|
[System.Diagnostics.DebuggerNonUserCode] |
|
internal void Dispose(bool disposeManagedResources) |
|
{ |
|
try |
|
{ |
|
if (TakeDisposalExecution() && disposeManagedResources) |
|
{ |
|
OnDispose(); |
|
} |
|
} |
|
catch (Exception exception) |
|
{ |
|
// Fields may be partially collected |
|
GC.KeepAlive(exception); |
|
} |
|
} |
|
|
|
protected override void Initialize(Action beforeInitialize) |
|
{ |
|
Action beforeInitializeReplacement = () => |
|
{ |
|
try |
|
{ |
|
var waitHandle = WaitHandle.Value; |
|
if (!WaitHandle.IsAlive) |
|
{ |
|
WaitHandle.Value = new System.Threading.ManualResetEventSlim(false); |
|
GC.KeepAlive(waitHandle); |
|
} |
|
if (beforeInitialize != null) |
|
{ |
|
beforeInitialize.Invoke(); |
|
} |
|
} |
|
finally |
|
{ |
|
UnDispose(); |
|
} |
|
}; |
|
base.Initialize(beforeInitializeReplacement); |
|
} |
|
|
|
private void OnDispose() |
|
{ |
|
var target = Value; |
|
target.Dispose(); |
|
Free(); |
|
ReleaseWaitHandle(false); |
|
} |
|
|
|
private bool TakeDisposalExecution() |
|
{ |
|
if (_status == -1) |
|
{ |
|
return false; |
|
} |
|
return ThreadingHelper.SpinWaitSetUnless(ref _status, -1, 0, -1); |
|
} |
|
|
|
[System.Diagnostics.DebuggerNonUserCode] |
|
private bool UnDispose() |
|
{ |
|
if (System.Threading.Volatile.Read(ref _status) == -1) |
|
{ |
|
System.Threading.Volatile.Write(ref _status, 0); |
|
return true; |
|
} |
|
return false; |
|
} |
|
} |
|
} |
|
|
|
#endif |