less locking for screen connection, force more concurrency
This commit is contained in:
parent
0b10695e07
commit
0e93b1356f
|
@ -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;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -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;
|
||||||
}
|
}
|
||||||
|
|
|
@ -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);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -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);
|
||||||
}
|
}
|
||||||
|
|
|
@ -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);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -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());
|
||||||
}
|
}
|
||||||
|
|
|
@ -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;
|
||||||
_wantsFrameOnTick = true;
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
await SendNowAsync();
|
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 ValueTask OnGameTickAsync(PixelGrid pixels, GamePixelGrid gamePixelGrid) => LockedAsync(async () =>
|
public async ValueTask OnGameTickAsync(PixelGrid pixels, GamePixelGrid gamePixelGrid)
|
||||||
{
|
{
|
||||||
if (pixels == _lastSentPixels)
|
await Task.Yield();
|
||||||
return;
|
|
||||||
|
|
||||||
if (_nextPlayerData != null)
|
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)
|
||||||
{
|
{
|
||||||
_nextPlayerData.Clear();
|
var data = _playerDataBuilder.Build(gamePixelGrid);
|
||||||
foreach (var gamePixel in gamePixelGrid)
|
nextPlayerDataOwner = _memoryPool.Rent(data.Length);
|
||||||
{
|
nextPlayerData = nextPlayerDataOwner.Memory[..data.Length];
|
||||||
if (!gamePixel.EntityType.HasValue)
|
data.CopyTo(nextPlayerData.Value);
|
||||||
continue;
|
|
||||||
_nextPlayerData.Add(gamePixel.EntityType.Value, gamePixel.BelongsTo?.Name == playerName);
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
_nextPixels = pixels;
|
var next = new Package(nextPixelsOwner, nextPixels, nextPlayerDataOwner, nextPlayerData);
|
||||||
if (_wantsFrameOnTick)
|
if (Interlocked.Exchange(ref _wantsFrameOnTick, 0) != 0)
|
||||||
_ = await SendNowAsync();
|
{
|
||||||
});
|
await SendAndDisposeAsync(next);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
private async ValueTask<bool> SendNowAsync()
|
var oldNext = Interlocked.Exchange(ref _next, next);
|
||||||
|
oldNext?.PixelsOwner.Dispose();
|
||||||
|
oldNext?.PlayerDataOwner?.Dispose();
|
||||||
|
}
|
||||||
|
|
||||||
|
private async ValueTask SendAndDisposeAsync(Package package)
|
||||||
{
|
{
|
||||||
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();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -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);
|
||||||
logger.LogTrace("packet length: {} (count={})", index, _count);
|
|
||||||
|
if (logger.IsEnabled(LogLevel.Trace))
|
||||||
|
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();
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -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();
|
||||||
|
|
||||||
|
|
|
@ -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)
|
||||||
|
|
Loading…
Reference in a new issue