less locking for screen connection, force more concurrency

This commit is contained in:
Vinzenz Schroeter 2024-04-30 23:49:39 +02:00 committed by RobbersDaughter
parent 0b10695e07
commit 0e93b1356f
10 changed files with 143 additions and 93 deletions

View file

@ -48,14 +48,18 @@ internal sealed class Endpoints(
return TypedResults.Empty; return TypedResults.Empty;
} }
private async Task<Results<BadRequest, EmptyHttpResult>> ConnectScreenAsync(HttpContext context, private async Task<Results<BadRequest, EmptyHttpResult, NotFound>> ConnectScreenAsync(HttpContext context,
[FromQuery] string? playerName) [FromQuery] string? playerName)
{ {
if (!context.WebSockets.IsWebSocketRequest) if (!context.WebSockets.IsWebSocketRequest)
return TypedResults.BadRequest(); return TypedResults.BadRequest();
Player? player = null;
if (!string.IsNullOrWhiteSpace(playerName) && !playerService.TryGet(playerName, out player))
return TypedResults.NotFound();
using var ws = await context.WebSockets.AcceptWebSocketAsync(); using var ws = await context.WebSockets.AcceptWebSocketAsync();
await clientScreenServer.HandleClientAsync(ws, playerName); await clientScreenServer.HandleClientAsync(ws, player);
return TypedResults.Empty; return TypedResults.Empty;
} }

View file

@ -7,37 +7,19 @@ internal sealed class GameTickWorker(
IEnumerable<ITickStep> steps, IEnumerable<ITickStep> steps,
IHostApplicationLifetime lifetime, IHostApplicationLifetime lifetime,
ILogger<GameTickWorker> logger ILogger<GameTickWorker> logger
) : IHostedService, IDisposable ) : IHostedLifecycleService, IDisposable
{ {
private readonly CancellationTokenSource _cancellation = new(); private readonly CancellationTokenSource _cancellation = new();
private readonly TaskCompletionSource _shutdownCompletion = new();
private readonly List<ITickStep> _steps = steps.ToList(); private readonly List<ITickStep> _steps = steps.ToList();
private Task? _run;
public void Dispose() public async Task StartedAsync(CancellationToken cancellationToken)
{ {
_cancellation.Dispose(); await Task.Yield();
_run?.Dispose();
}
public Task StartAsync(CancellationToken cancellationToken)
{
_run = RunAsync();
return Task.CompletedTask;
}
public async Task StopAsync(CancellationToken cancellationToken)
{
await _cancellation.CancelAsync();
if (_run != null) await _run;
}
private async Task RunAsync()
{
// the first tick is really short (< 0.01ms) if this line is directly above the while // the first tick is really short (< 0.01ms) if this line is directly above the while
var sw = Stopwatch.StartNew(); var sw = Stopwatch.StartNew();
await Task.Delay(1, CancellationToken.None).ConfigureAwait(false);
// do not block in StartAsync
await Task.Delay(1).ConfigureAwait(false);
try try
{ {
@ -55,5 +37,19 @@ internal sealed class GameTickWorker(
logger.LogError(ex, "game tick service crashed"); logger.LogError(ex, "game tick service crashed");
lifetime.StopApplication(); lifetime.StopApplication();
} }
_shutdownCompletion.SetResult();
} }
public Task StoppingAsync(CancellationToken cancellationToken) => _cancellation.CancelAsync();
public Task StopAsync(CancellationToken cancellationToken) => _shutdownCompletion.Task;
public void Dispose() => _cancellation.Dispose();
public Task StartAsync(CancellationToken cancellationToken) => Task.CompletedTask;
public Task StartingAsync(CancellationToken cancellationToken) => Task.CompletedTask;
public Task StoppedAsync(CancellationToken cancellationToken) => Task.CompletedTask;
} }

View file

@ -8,26 +8,38 @@ internal sealed class GeneratePixelsTickStep(
IEnumerable<IFrameConsumer> consumers IEnumerable<IFrameConsumer> consumers
) : ITickStep ) : ITickStep
{ {
private readonly GamePixelGrid _gamePixelGrid = new(MapService.PixelsPerRow, MapService.PixelsPerColumn); private GamePixelGrid _lastGamePixelGrid = new(MapService.PixelsPerRow, MapService.PixelsPerColumn);
private PixelGrid _lastObserverPixelGrid = new(MapService.PixelsPerRow, MapService.PixelsPerColumn);
private GamePixelGrid _gamePixelGrid = new(MapService.PixelsPerRow, MapService.PixelsPerColumn);
private PixelGrid _observerPixelGrid = new(MapService.PixelsPerRow, MapService.PixelsPerColumn);
private readonly List<IDrawStep> _drawSteps = drawSteps.ToList(); private readonly List<IDrawStep> _drawSteps = drawSteps.ToList();
private readonly List<IFrameConsumer> _consumers = consumers.ToList(); private readonly List<IFrameConsumer> _consumers = consumers.ToList();
public async ValueTask TickAsync(TimeSpan _) public async ValueTask TickAsync(TimeSpan _)
{ {
PixelGrid observerPixelGrid = new(MapService.PixelsPerRow, MapService.PixelsPerColumn); Draw(_gamePixelGrid, _observerPixelGrid);
if (_observerPixelGrid.Data.Span.SequenceEqual(_lastObserverPixelGrid.Data.Span))
return;
_gamePixelGrid.Clear(); await Task.WhenAll(_consumers.Select(c => c.OnFrameDoneAsync(_gamePixelGrid, _observerPixelGrid)));
(_lastGamePixelGrid, _gamePixelGrid) = (_gamePixelGrid, _lastGamePixelGrid);
(_lastObserverPixelGrid, _observerPixelGrid) = (_observerPixelGrid, _lastObserverPixelGrid);
}
private void Draw(GamePixelGrid gamePixelGrid, PixelGrid observerPixelGrid)
{
gamePixelGrid.Clear();
foreach (var step in _drawSteps) foreach (var step in _drawSteps)
step.Draw(_gamePixelGrid); step.Draw(gamePixelGrid);
observerPixelGrid.Clear();
for (var y = 0; y < MapService.PixelsPerColumn; y++) for (var y = 0; y < MapService.PixelsPerColumn; y++)
for (var x = 0; x < MapService.PixelsPerRow; x++) for (var x = 0; x < MapService.PixelsPerRow; x++)
{ {
if (_gamePixelGrid[x, y].EntityType.HasValue) if (gamePixelGrid[x, y].EntityType.HasValue)
observerPixelGrid[(ushort)x, (ushort)y] = true; observerPixelGrid[(ushort)x, (ushort)y] = true;
} }
foreach (var consumer in _consumers)
await consumer.OnFrameDoneAsync(_gamePixelGrid, observerPixelGrid);
} }
} }

View file

@ -4,5 +4,5 @@ namespace TanksServer.Graphics;
internal interface IFrameConsumer internal interface IFrameConsumer
{ {
ValueTask OnFrameDoneAsync(GamePixelGrid gamePixelGrid, PixelGrid observerPixels); Task OnFrameDoneAsync(GamePixelGrid gamePixelGrid, PixelGrid observerPixels);
} }

View file

@ -16,20 +16,20 @@ internal sealed class ByteChannelWebSocket(WebSocket socket, ILogger logger, int
catch (WebSocketException e) catch (WebSocketException e)
{ {
logger.LogError(e, "could not send binary message"); logger.LogError(e, "could not send binary message");
await CloseAsync(); await CloseWithErrorAsync(e.Message);
} }
} }
public async ValueTask SendTextAsync(ReadOnlyMemory<byte> data, bool endOfMessage = true) public async ValueTask SendTextAsync(ReadOnlyMemory<byte> utf8Data, bool endOfMessage = true)
{ {
try try
{ {
await socket.SendAsync(data, WebSocketMessageType.Text, endOfMessage, CancellationToken.None); await socket.SendAsync(utf8Data, WebSocketMessageType.Text, endOfMessage, CancellationToken.None);
} }
catch (WebSocketException e) catch (WebSocketException e)
{ {
logger.LogError(e, "could not send text message"); logger.LogError(e, "could not send text message");
await CloseAsync(); await CloseWithErrorAsync(e.Message);
} }
} }

View file

@ -9,13 +9,13 @@ internal sealed class ClientScreenServer(
ILoggerFactory loggerFactory ILoggerFactory loggerFactory
) : WebsocketServer<ClientScreenServerConnection>(logger), IFrameConsumer ) : WebsocketServer<ClientScreenServerConnection>(logger), IFrameConsumer
{ {
public Task HandleClientAsync(WebSocket socket, string? player) public Task HandleClientAsync(WebSocket socket, Player? player)
=> base.HandleClientAsync(new( => base.HandleClientAsync(new ClientScreenServerConnection(
socket, socket,
loggerFactory.CreateLogger<ClientScreenServerConnection>(), loggerFactory.CreateLogger<ClientScreenServerConnection>(),
player player
)); ));
public ValueTask OnFrameDoneAsync(GamePixelGrid gamePixelGrid, PixelGrid observerPixels) public async Task OnFrameDoneAsync(GamePixelGrid gamePixelGrid, PixelGrid observerPixels)
=> ParallelForEachConnectionAsync(c => c.OnGameTickAsync(observerPixels, gamePixelGrid).AsTask()); => await ParallelForEachConnectionAsync(c => c.OnGameTickAsync(observerPixels, gamePixelGrid).AsTask());
} }

View file

@ -1,3 +1,5 @@
using System.Buffers;
using System.Diagnostics;
using System.Net.WebSockets; using System.Net.WebSockets;
using DisplayCommands; using DisplayCommands;
using TanksServer.Graphics; using TanksServer.Graphics;
@ -7,67 +9,87 @@ namespace TanksServer.Interactivity;
internal sealed class ClientScreenServerConnection( internal sealed class ClientScreenServerConnection(
WebSocket webSocket, WebSocket webSocket,
ILogger<ClientScreenServerConnection> logger, ILogger<ClientScreenServerConnection> logger,
string? playerName = null Player? player
) : WebsocketServerConnection(logger, new ByteChannelWebSocket(webSocket, logger, 0)) ) : WebsocketServerConnection(logger, new ByteChannelWebSocket(webSocket, logger, 0))
{ {
private bool _wantsFrameOnTick = true; private sealed record class Package(
IMemoryOwner<byte> PixelsOwner,
Memory<byte> Pixels,
IMemoryOwner<byte>? PlayerDataOwner,
Memory<byte>? PlayerData
);
private PixelGrid? _lastSentPixels; private readonly MemoryPool<byte> _memoryPool = MemoryPool<byte>.Shared;
private PixelGrid? _nextPixels; private int _wantsFrameOnTick = 1;
private readonly PlayerScreenData? _nextPlayerData = playerName != null ? new PlayerScreenData(logger) : null; private Package? _next;
protected override async ValueTask HandleMessageLockedAsync(Memory<byte> _) private readonly PlayerScreenData? _playerDataBuilder = player == null
? null
: new PlayerScreenData(logger, player);
protected override ValueTask HandleMessageLockedAsync(Memory<byte> buffer) => throw new UnreachableException();
protected override ValueTask HandleMessageAsync(Memory<byte> _)
{ {
if (_nextPixels == null) if (_wantsFrameOnTick != 0)
return ValueTask.CompletedTask;
var package = Interlocked.Exchange(ref _next, null);
if (package != null)
return SendAndDisposeAsync(package);
// the delay between one exchange and this set could be enough for another frame to complete
// this would mean the client simply drops a frame, so this should be fine
_wantsFrameOnTick = 1;
return ValueTask.CompletedTask;
}
public async ValueTask OnGameTickAsync(PixelGrid pixels, GamePixelGrid gamePixelGrid)
{ {
_wantsFrameOnTick = true; await Task.Yield();
var nextPixelsOwner = _memoryPool.Rent(pixels.Data.Length);
var nextPixels = nextPixelsOwner.Memory[..pixels.Data.Length];
pixels.Data.CopyTo(nextPixels);
IMemoryOwner<byte>? nextPlayerDataOwner = null;
Memory<byte>? nextPlayerData = null;
if (_playerDataBuilder != null)
{
var data = _playerDataBuilder.Build(gamePixelGrid);
nextPlayerDataOwner = _memoryPool.Rent(data.Length);
nextPlayerData = nextPlayerDataOwner.Memory[..data.Length];
data.CopyTo(nextPlayerData.Value);
}
var next = new Package(nextPixelsOwner, nextPixels, nextPlayerDataOwner, nextPlayerData);
if (Interlocked.Exchange(ref _wantsFrameOnTick, 0) != 0)
{
await SendAndDisposeAsync(next);
return; return;
} }
await SendNowAsync(); var oldNext = Interlocked.Exchange(ref _next, next);
oldNext?.PixelsOwner.Dispose();
oldNext?.PlayerDataOwner?.Dispose();
} }
public ValueTask OnGameTickAsync(PixelGrid pixels, GamePixelGrid gamePixelGrid) => LockedAsync(async () => private async ValueTask SendAndDisposeAsync(Package package)
{ {
if (pixels == _lastSentPixels)
return;
if (_nextPlayerData != null)
{
_nextPlayerData.Clear();
foreach (var gamePixel in gamePixelGrid)
{
if (!gamePixel.EntityType.HasValue)
continue;
_nextPlayerData.Add(gamePixel.EntityType.Value, gamePixel.BelongsTo?.Name == playerName);
}
}
_nextPixels = pixels;
if (_wantsFrameOnTick)
_ = await SendNowAsync();
});
private async ValueTask<bool> SendNowAsync()
{
var pixels = _nextPixels
?? throw new InvalidOperationException("next pixels not set");
try try
{ {
await Socket.SendBinaryAsync(pixels.Data, _nextPlayerData == null); await Socket.SendBinaryAsync(package.Pixels, package.PlayerData == null);
if (_nextPlayerData != null) if (package.PlayerData != null)
await Socket.SendBinaryAsync(_nextPlayerData.GetPacket()); await Socket.SendBinaryAsync(package.PlayerData.Value);
_lastSentPixels = _nextPixels;
_nextPixels = null;
_wantsFrameOnTick = false;
return true;
} }
catch (WebSocketException ex) catch (WebSocketException ex)
{ {
Logger.LogWarning(ex, "send failed"); Logger.LogWarning(ex, "send failed");
return false; }
finally
{
package.PixelsOwner.Dispose();
package.PlayerDataOwner?.Dispose();
} }
} }
} }

View file

@ -4,25 +4,30 @@ using TanksServer.Graphics;
namespace TanksServer.Interactivity; namespace TanksServer.Interactivity;
internal sealed class PlayerScreenData(ILogger logger) internal sealed class PlayerScreenData(ILogger logger, Player player)
{ {
private readonly Memory<byte> _data = new byte[MapService.PixelsPerRow * MapService.PixelsPerColumn / 2]; private readonly Memory<byte> _data = new byte[MapService.PixelsPerRow * MapService.PixelsPerColumn / 2];
private int _count; private int _count;
public void Clear() public ReadOnlyMemory<byte> Build(GamePixelGrid gamePixelGrid)
{ {
_count = 0; Clear();
_data.Span.Clear(); foreach (var gamePixel in gamePixelGrid)
{
if (!gamePixel.EntityType.HasValue)
continue;
Add(gamePixel.EntityType.Value, gamePixel.BelongsTo == player);
} }
public ReadOnlyMemory<byte> GetPacket()
{
var index = _count / 2 + (_count % 2 == 0 ? 0 : 1); var index = _count / 2 + (_count % 2 == 0 ? 0 : 1);
if (logger.IsEnabled(LogLevel.Trace))
logger.LogTrace("packet length: {} (count={})", index, _count); logger.LogTrace("packet length: {} (count={})", index, _count);
return _data[..index]; return _data[..index];
} }
public void Add(GamePixelEntityType entityKind, bool isCurrentPlayer) private void Add(GamePixelEntityType entityKind, bool isCurrentPlayer)
{ {
var result = (byte)(isCurrentPlayer ? 0x1 : 0x0); var result = (byte)(isCurrentPlayer ? 0x1 : 0x0);
var kind = (byte)entityKind; var kind = (byte)entityKind;
@ -36,4 +41,10 @@ internal sealed class PlayerScreenData(ILogger logger)
_data.Span[index] = result; _data.Span[index] = result;
_count++; _count++;
} }
private void Clear()
{
_count = 0;
_data.Span.Clear();
}
} }

View file

@ -48,11 +48,13 @@ internal sealed class SendToServicePointDisplay : IFrameConsumer
}; };
} }
public async ValueTask OnFrameDoneAsync(GamePixelGrid gamePixelGrid, PixelGrid observerPixels) public async Task OnFrameDoneAsync(GamePixelGrid gamePixelGrid, PixelGrid observerPixels)
{ {
if (DateTime.Now < _nextFrameAfter) if (DateTime.Now < _nextFrameAfter)
return; return;
_nextFrameAfter = DateTime.Now + _minFrameTime; _nextFrameAfter = DateTime.Now + _minFrameTime;
await Task.Yield();
RefreshScores(); RefreshScores();

View file

@ -18,10 +18,13 @@ internal abstract class WebsocketServerConnection(
public async Task ReceiveAsync() public async Task ReceiveAsync()
{ {
await foreach (var buffer in Socket.ReadAllAsync()) await foreach (var buffer in Socket.ReadAllAsync())
await LockedAsync(() => HandleMessageLockedAsync(buffer)); await HandleMessageAsync(buffer);
Logger.LogTrace("done receiving"); Logger.LogTrace("done receiving");
} }
protected virtual ValueTask HandleMessageAsync(Memory<byte> buffer)
=> LockedAsync(() => HandleMessageLockedAsync(buffer));
protected abstract ValueTask HandleMessageLockedAsync(Memory<byte> buffer); protected abstract ValueTask HandleMessageLockedAsync(Memory<byte> buffer);
protected async ValueTask LockedAsync(Func<ValueTask> action) protected async ValueTask LockedAsync(Func<ValueTask> action)