-
Notifications
You must be signed in to change notification settings - Fork 155
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Made small improvements to the protocol layer of Bedrock.Framework #160
base: main
Are you sure you want to change the base?
Conversation
thenameless314159
commented
Apr 5, 2023
- ProtocolWriter has been refactored to try and get the conch from the semaphore synchronously first if possible
- ProtocolWriter has an additional method to write an array of messages
- Made benchmarks for ProtocolWriter
- Made unit-tests for ProtocolWriter
- Added unit-tests for ProtocolReader
- ProtocolWriter has been refactored to try and get the conch from the semaphore synchronously first if possible - ProtocolWriter has an additional method to write an array of messages - Made benchmarks for ProtocolWriter - Made unit-tests for ProtocolWriter - Added unit-tests for ProtocolReader
I'm wondering about the ConcurrentPipeWriter.cs from Kestrel sources, wouldn't it be "better" to rely on it instead of a SemaphoreSlim ? And in which ways ? |
I am actually not sure whether the ProtocolWriter should noop on write complete/cancel, because it may allow the execution of logic that shouldn't have been executed because of a failed send or something, what's your thought @davidfowl ? |
I think it should throw |
Alright done @davidfowl ! This change allowed me to greatly improve the clarity of the fast sync path, it's a lot cleaner now but it still require some review, here is how the fast path looks like now (with .NET 7) : public ValueTask WriteAsync<TMessage>(IMessageWriter<TMessage> writer, in TMessage message, CancellationToken cancellationToken = default)
{
// This will always finish synchronously so we do not need to bother with cancel
if (!TryWaitForSingleWriter(cancellationToken: CancellationToken.None))
return WriteAsyncSlow(writer, message, cancellationToken);
bool release = true, hasWritten = false;
try
{
if (_disposed) throw new ObjectDisposedException(nameof(ProtocolWriter));
var task = WriteCore(writer, in message, cancellationToken);
if (task.IsCompletedSuccessfully)
{
// If it's a IValueTaskSource backed ValueTask,
// inform it its result has been read so it can reset
var result = task.GetAwaiter().GetResult();
if (result.IsCanceled)
throw new OperationCanceledException(cancellationToken);
hasWritten = !result.IsCompleted;
return hasWritten
? ValueTask.CompletedTask
// REVIEW : could DisposeAsyncCore(false) here if we add a !_dispose check between disposing && !TryWaitForSingleWriter()
// but it'd require to implement a ThrowAfter extension method for ValueTask
: throw new ObjectDisposedException(nameof(ProtocolWriter));
}
else
{
release = false; // do not release if we need to go async to complete the write
return new ValueTask(CompleteWriteAsync(task, messagesWritten: 1));
}
}
finally { if (release) ReleaseSingleWriter(hasWritten ? 1 : 0); }
} And this is the extension method that'd be required in order to public static ValueTask ThrowAfter(this in ValueTask valueTask, Exception exception)
{
if (valueTask.IsCompletedSuccessfully)
{
// Signal consumption to the IValueTaskSource
valueTask.GetAwaiter().GetResult();
throw exception;
}
else return AwaitAndThrow(valueTask.AsTask(), exception);
static async ValueTask AwaitAndThrow(Task task, Exception exception)
{
await task.ConfigureAwait(false);
throw exception;
}
} |