269 lines
7.6 KiB
C#
Raw Normal View History

2019-06-12 21:56:55 +02:00
using System;
using System.Diagnostics;
using System.Threading.Tasks;
namespace Capnp.Rpc
{
class PromisedCapability : RemoteResolvingCapability
{
readonly uint _remoteId;
readonly object _reentrancyBlocker = new object();
2020-03-10 21:55:34 +01:00
readonly TaskCompletionSource<ConsumedCapability?> _resolvedCap = new TaskCompletionSource<ConsumedCapability?>();
readonly Task<Proxy> _whenResolvedProxy;
2019-06-12 21:56:55 +02:00
bool _released;
public PromisedCapability(IRpcEndpoint ep, uint remoteId): base(ep)
{
_remoteId = remoteId;
2020-03-10 21:55:34 +01:00
async Task<Proxy> AwaitProxy() => new Proxy(await WhenResolved);
_whenResolvedProxy = AwaitProxy();
2019-06-12 21:56:55 +02:00
}
2020-03-10 21:55:34 +01:00
public override Task<ConsumedCapability?> WhenResolved => _resolvedCap.Task;
2019-06-12 21:56:55 +02:00
2020-01-11 17:21:31 +01:00
internal override void Freeze(out IRpcEndpoint? boundEndpoint)
2019-06-12 21:56:55 +02:00
{
lock (_reentrancyBlocker)
{
if (_resolvedCap.Task.IsCompleted && _pendingCallsOnPromise == 0)
{
2020-03-10 21:55:34 +01:00
boundEndpoint = null;
2019-06-12 21:56:55 +02:00
try
{
2020-03-10 21:55:34 +01:00
_resolvedCap.Task.Result?.Freeze(out boundEndpoint);
2019-06-12 21:56:55 +02:00
}
catch (AggregateException exception)
{
2020-01-11 17:21:31 +01:00
throw exception.InnerException!;
2019-06-12 21:56:55 +02:00
}
}
else
{
Debug.Assert(!_released);
++_pendingCallsOnPromise;
boundEndpoint = _ep;
}
}
}
internal override void Unfreeze()
{
bool release = false;
lock (_reentrancyBlocker)
{
if (_pendingCallsOnPromise == 0)
{
2020-03-10 21:55:34 +01:00
_resolvedCap.Task.Result?.Unfreeze();
2019-06-12 21:56:55 +02:00
}
else
{
Debug.Assert(_pendingCallsOnPromise > 0);
Debug.Assert(!_released);
if (--_pendingCallsOnPromise == 0 && _resolvedCap.Task.IsCompleted)
{
release = true;
_released = true;
}
}
}
if (release)
{
_ep.ReleaseImport(_remoteId);
}
}
internal override Action? Export(IRpcEndpoint endpoint, CapDescriptor.WRITER writer)
2019-06-12 21:56:55 +02:00
{
lock (_reentrancyBlocker)
{
2019-06-22 18:43:30 -04:00
if (_resolvedCap.Task.ReplacementTaskIsCompletedSuccessfully())
2019-06-12 21:56:55 +02:00
{
2020-03-10 21:55:34 +01:00
using var proxy = new Proxy(_resolvedCap.Task.Result);
proxy.Export(endpoint, writer);
2019-06-12 21:56:55 +02:00
}
else
{
if (_ep == endpoint)
{
writer.which = CapDescriptor.WHICH.ReceiverHosted;
writer.ReceiverHosted = _remoteId;
Debug.Assert(!_released);
++_pendingCallsOnPromise;
return () =>
2019-06-12 21:56:55 +02:00
{
bool release = false;
lock (_reentrancyBlocker)
{
if (--_pendingCallsOnPromise == 0 && _resolvedCap.Task.IsCompleted)
{
_released = true;
release = true;
}
}
if (release)
{
_ep.ReleaseImport(_remoteId);
}
};
2019-06-12 21:56:55 +02:00
}
else
{
this.ExportAsSenderPromise(endpoint, writer);
}
}
}
return null;
2019-06-12 21:56:55 +02:00
}
async void TrackCall(Task call)
{
try
{
await call;
}
catch
{
}
finally
{
bool release = false;
lock (_reentrancyBlocker)
{
if (--_pendingCallsOnPromise == 0 && _resolvedCap.Task.IsCompleted)
{
release = true;
_released = true;
}
}
if (release)
{
_ep.ReleaseImport(_remoteId);
}
}
}
2020-03-10 21:55:34 +01:00
protected override ConsumedCapability? ResolvedCap
2019-06-12 21:56:55 +02:00
{
get
{
try
{
return _resolvedCap.Task.IsCompleted ? _resolvedCap.Task.Result : null;
}
catch (AggregateException exception)
{
2020-01-11 17:21:31 +01:00
throw exception.InnerException!;
2019-06-12 21:56:55 +02:00
}
}
}
protected override void GetMessageTarget(MessageTarget.WRITER wr)
{
wr.which = MessageTarget.WHICH.ImportedCap;
wr.ImportedCap = _remoteId;
}
internal override IPromisedAnswer DoCall(ulong interfaceId, ushort methodId, DynamicSerializerState args)
2019-06-12 21:56:55 +02:00
{
lock (_reentrancyBlocker)
{
if (_resolvedCap.Task.IsCompleted)
{
return CallOnResolution(interfaceId, methodId, args);
2019-06-12 21:56:55 +02:00
}
else
{
Debug.Assert(!_released);
++_pendingCallsOnPromise;
}
}
var promisedAnswer = base.DoCall(interfaceId, methodId, args);
2019-06-12 21:56:55 +02:00
TrackCall(promisedAnswer.WhenReturned);
return promisedAnswer;
}
public void ResolveTo(ConsumedCapability resolvedCap)
{
bool release = false;
lock (_reentrancyBlocker)
{
2020-03-10 21:55:34 +01:00
_resolvedCap.SetResult(resolvedCap);
2019-06-12 21:56:55 +02:00
if (_pendingCallsOnPromise == 0)
{
release = true;
_released = true;
}
}
if (release)
{
_ep.ReleaseImport(_remoteId);
}
}
public void Break(string message)
{
bool release = false;
lock (_reentrancyBlocker)
{
#if false
_resolvedCap.SetException(new RpcException(message));
#else
2020-03-10 21:55:34 +01:00
_resolvedCap.SetResult(LazyCapability.CreateBrokenCap(message));
2019-06-12 21:56:55 +02:00
#endif
if (_pendingCallsOnPromise == 0)
{
release = true;
_released = true;
}
}
if (release)
{
_ep.ReleaseImport(_remoteId);
}
}
2020-03-10 21:55:34 +01:00
protected async override void ReleaseRemotely()
2019-06-12 21:56:55 +02:00
{
if (!_released)
{
_ep.ReleaseImport(_remoteId);
}
2019-08-26 20:03:56 +02:00
_ep.ReleaseImport(_remoteId);
2019-06-12 21:56:55 +02:00
2020-03-10 21:55:34 +01:00
try { using var _ = await _whenResolvedProxy; }
catch { }
2019-06-12 21:56:55 +02:00
}
protected override Call.WRITER SetupMessage(DynamicSerializerState args, ulong interfaceId, ushort methodId)
{
var call = base.SetupMessage(args, interfaceId, methodId);
call.Target.which = MessageTarget.WHICH.ImportedCap;
call.Target.ImportedCap = _remoteId;
return call;
}
}
2020-01-11 17:56:12 +01:00
}