Compare commits
98 Commits
v1.0.0-rc
...
v1.0.30-rc
| Author | SHA1 | Date | |
|---|---|---|---|
| a5a67963be | |||
| ab85578ccf | |||
| 13044357a5 | |||
| a9be230960 | |||
| 5a4bf0c24e | |||
| fa6ace4dc8 | |||
| 4d8ed1105a | |||
| e2ef761bd7 | |||
| 828112855f | |||
| 06ff76fe0b | |||
| c92b5a5bc4 | |||
| f83d3ea0c7 | |||
| 3564eb2adc | |||
| 3531432758 | |||
| 6bda468607 | |||
| 7ddd52bf9d | |||
| ecf3631163 | |||
| 8a149eb3b4 | |||
| 14ae5e8189 | |||
| b7e8327ca8 | |||
| b9e79af9d8 | |||
| 545ec02ecc | |||
| 73feb77169 | |||
| cbcf1773aa | |||
| 5519ae7679 | |||
| 1558b5eefb | |||
| aaa0e4a317 | |||
| ff712dc094 | |||
| c2c75cb513 | |||
| 9a22566f79 | |||
| 783d2ce922 | |||
| 5771ec738b | |||
| 85081e1da7 | |||
| dbaa5d9d92 | |||
| 54786c065d | |||
| 0b3a0dd1ac | |||
| 72ff37e94a | |||
| 75dab9d16f | |||
| 25b0e54d13 | |||
| e4f664b557 | |||
| 174a4c4422 | |||
| a51d7c73cd | |||
| 957622a170 | |||
| 4a07424293 | |||
| 568a3282c3 | |||
| 6a71fe5fe0 | |||
| 082e183989 | |||
| 51c0482a40 | |||
| 9e16c53cad | |||
| 6ec29e5dcd | |||
| 65c1d9c6d4 | |||
| f2934bc8ee | |||
| 81879d4fe2 | |||
| 2d89d267f5 | |||
| 34107cbd5f | |||
| ba2ce25aec | |||
| ca17e7de83 | |||
| dff6dbcf97 | |||
| ae485f96d4 | |||
| 74904de16b | |||
| 655cf3503d | |||
| 8fd8558323 | |||
| 944cb65733 | |||
| 877ebdcde2 | |||
| e2032f381a | |||
| 85336f998e | |||
| 16b8d3a062 | |||
| f72b4c0904 | |||
| 19272609ef | |||
| 5f3b2d7ed8 | |||
| 773adeefb2 | |||
| e674600308 | |||
| 4f587fa59c | |||
| 76caa840bd | |||
| 1bff47e56b | |||
| 1e41b9f2eb | |||
| 679608bc48 | |||
| e9c129e2f2 | |||
| 66181c612a | |||
| 189278e17c | |||
| 08c399a030 | |||
| 05c8904601 | |||
| 04b1b16b11 | |||
| 530c6109ea | |||
| 3efd73d8cb | |||
| a85ac99a3c | |||
| e295e9f7db | |||
| 06dd23ee8d | |||
| 62d3f7acdd | |||
| e2d07eb396 | |||
| 4f00c36cd9 | |||
| 8c1945e352 | |||
| 2ef1da5c90 | |||
| ec65b9f305 | |||
| 35ca016520 | |||
| 8481cb89ad | |||
| bcec99cff1 | |||
| bb7cccb61a |
@@ -19,7 +19,8 @@ jobs:
|
|||||||
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||||||
with:
|
with:
|
||||||
tag_name: ${{ github.ref }}
|
tag_name: ${{ github.ref }}
|
||||||
release_name: SimpleServer-${{ github.ref }}
|
release_name: Ragon.Relay-${{ github.ref }}
|
||||||
|
prerelease: true
|
||||||
|
|
||||||
build:
|
build:
|
||||||
name: Release
|
name: Release
|
||||||
@@ -52,10 +53,10 @@ jobs:
|
|||||||
- name: Build
|
- name: Build
|
||||||
shell: bash
|
shell: bash
|
||||||
run: |
|
run: |
|
||||||
release_name="SimpleServer-${{ env.TAG }}-${{ matrix.target }}"
|
release_name="Ragon.Relay-${{ env.TAG }}-${{ matrix.target }}"
|
||||||
|
|
||||||
# Build everything
|
# Build everything
|
||||||
dotnet publish SimpleServer/SimpleServer.csproj -c Release --runtime "${{ matrix.target }}" -p:PublishSingleFile=true -o "$release_name"
|
dotnet publish Ragon.Relay/Ragon.Relay.csproj -c Release --runtime "${{ matrix.target }}" -p:PublishSingleFile=true -o "$release_name"
|
||||||
|
|
||||||
# Pack files
|
# Pack files
|
||||||
7z a -tzip "${release_name}.zip" "./${release_name}/*"
|
7z a -tzip "${release_name}.zip" "./${release_name}/*"
|
||||||
@@ -68,6 +69,6 @@ jobs:
|
|||||||
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||||||
with:
|
with:
|
||||||
upload_url: ${{ needs.release.outputs.upload_url }}
|
upload_url: ${{ needs.release.outputs.upload_url }}
|
||||||
asset_path: SimpleServer-${{ env.TAG }}-${{ matrix.target }}.zip
|
asset_path: Ragon.Relay-${{ env.TAG }}-${{ matrix.target }}.zip
|
||||||
asset_name: SimpleServer-${{ env.TAG }}-${{ matrix.target }}.zip
|
asset_name: Ragon.Relay-${{ env.TAG }}-${{ matrix.target }}.zip
|
||||||
asset_content_type: application/zip
|
asset_content_type: application/zip
|
||||||
|
|||||||
Binary file not shown.
|
Before Width: | Height: | Size: 11 KiB |
Binary file not shown.
|
After Width: | Height: | Size: 6.0 KiB |
@@ -1,275 +0,0 @@
|
|||||||
using System.Runtime.CompilerServices;
|
|
||||||
using System.Runtime.InteropServices;
|
|
||||||
using System.Threading;
|
|
||||||
|
|
||||||
namespace DisruptorUnity3d
|
|
||||||
{
|
|
||||||
/// <summary>
|
|
||||||
/// Implementation of the Disruptor pattern
|
|
||||||
/// </summary>
|
|
||||||
/// <typeparam name="T">the type of item to be stored</typeparam>
|
|
||||||
public class RingBuffer<T>
|
|
||||||
{
|
|
||||||
private readonly T[] _entries;
|
|
||||||
private readonly int _modMask;
|
|
||||||
private Volatile.PaddedLong _consumerCursor = new Volatile.PaddedLong();
|
|
||||||
private Volatile.PaddedLong _producerCursor = new Volatile.PaddedLong();
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Creates a new RingBuffer with the given capacity
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="capacity">The capacity of the buffer</param>
|
|
||||||
/// <remarks>Only a single thread may attempt to consume at any one time</remarks>
|
|
||||||
public RingBuffer(int capacity)
|
|
||||||
{
|
|
||||||
capacity = NextPowerOfTwo(capacity);
|
|
||||||
_modMask = capacity - 1;
|
|
||||||
_entries = new T[capacity];
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// The maximum number of items that can be stored
|
|
||||||
/// </summary>
|
|
||||||
public int Capacity
|
|
||||||
{
|
|
||||||
get { return _entries.Length; }
|
|
||||||
}
|
|
||||||
|
|
||||||
public T this[long index]
|
|
||||||
{
|
|
||||||
get { unchecked { return _entries[index & _modMask]; } }
|
|
||||||
set { unchecked { _entries[index & _modMask] = value; } }
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Removes an item from the buffer.
|
|
||||||
/// </summary>
|
|
||||||
/// <returns>The next available item</returns>
|
|
||||||
public T Dequeue()
|
|
||||||
{
|
|
||||||
var next = _consumerCursor.ReadAcquireFence() + 1;
|
|
||||||
while (_producerCursor.ReadAcquireFence() < next) // makes sure we read the data from _entries after we have read the producer cursor
|
|
||||||
{
|
|
||||||
Thread.SpinWait(1);
|
|
||||||
}
|
|
||||||
var result = this[next];
|
|
||||||
_consumerCursor.WriteReleaseFence(next); // makes sure we read the data from _entries before we update the consumer cursor
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Attempts to remove an items from the queue
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="obj">the items</param>
|
|
||||||
/// <returns>True if successful</returns>
|
|
||||||
public bool TryDequeue(out T obj)
|
|
||||||
{
|
|
||||||
var next = _consumerCursor.ReadAcquireFence() + 1;
|
|
||||||
|
|
||||||
if (_producerCursor.ReadAcquireFence() < next)
|
|
||||||
{
|
|
||||||
obj = default(T);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
obj = Dequeue();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Add an item to the buffer
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="item"></param>
|
|
||||||
public void Enqueue(T item)
|
|
||||||
{
|
|
||||||
var next = _producerCursor.ReadAcquireFence() + 1;
|
|
||||||
|
|
||||||
long wrapPoint = next - _entries.Length;
|
|
||||||
long min = _consumerCursor.ReadAcquireFence();
|
|
||||||
|
|
||||||
while (wrapPoint > min)
|
|
||||||
{
|
|
||||||
min = _consumerCursor.ReadAcquireFence();
|
|
||||||
Thread.SpinWait(1);
|
|
||||||
}
|
|
||||||
|
|
||||||
this[next] = item;
|
|
||||||
_producerCursor.WriteReleaseFence(next); // makes sure we write the data in _entries before we update the producer cursor
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// The number of items in the buffer
|
|
||||||
/// </summary>
|
|
||||||
/// <remarks>for indicative purposes only, may contain stale data</remarks>
|
|
||||||
public int Count { get { return (int)(_producerCursor.ReadFullFence() - _consumerCursor.ReadFullFence()); } }
|
|
||||||
|
|
||||||
private static int NextPowerOfTwo(int x)
|
|
||||||
{
|
|
||||||
var result = 2;
|
|
||||||
while (result < x)
|
|
||||||
{
|
|
||||||
result <<= 1;
|
|
||||||
}
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
}
|
|
||||||
public static class Volatile
|
|
||||||
{
|
|
||||||
private const int CacheLineSize = 64;
|
|
||||||
|
|
||||||
[StructLayout(LayoutKind.Explicit, Size = CacheLineSize * 2)]
|
|
||||||
public struct PaddedLong
|
|
||||||
{
|
|
||||||
[FieldOffset(CacheLineSize)]
|
|
||||||
private long _value;
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Create a new <see cref="PaddedLong"/> with the given initial value.
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="value">Initial value</param>
|
|
||||||
public PaddedLong(long value)
|
|
||||||
{
|
|
||||||
_value = value;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Read the value without applying any fence
|
|
||||||
/// </summary>
|
|
||||||
/// <returns>The current value</returns>
|
|
||||||
public long ReadUnfenced()
|
|
||||||
{
|
|
||||||
return _value;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Read the value applying acquire fence semantic
|
|
||||||
/// </summary>
|
|
||||||
/// <returns>The current value</returns>
|
|
||||||
public long ReadAcquireFence()
|
|
||||||
{
|
|
||||||
var value = _value;
|
|
||||||
Thread.MemoryBarrier();
|
|
||||||
return value;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Read the value applying full fence semantic
|
|
||||||
/// </summary>
|
|
||||||
/// <returns>The current value</returns>
|
|
||||||
public long ReadFullFence()
|
|
||||||
{
|
|
||||||
Thread.MemoryBarrier();
|
|
||||||
return _value;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Read the value applying a compiler only fence, no CPU fence is applied
|
|
||||||
/// </summary>
|
|
||||||
/// <returns>The current value</returns>
|
|
||||||
[MethodImpl(MethodImplOptions.NoOptimization)]
|
|
||||||
public long ReadCompilerOnlyFence()
|
|
||||||
{
|
|
||||||
return _value;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Write the value applying release fence semantic
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="newValue">The new value</param>
|
|
||||||
public void WriteReleaseFence(long newValue)
|
|
||||||
{
|
|
||||||
Thread.MemoryBarrier();
|
|
||||||
_value = newValue;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Write the value applying full fence semantic
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="newValue">The new value</param>
|
|
||||||
public void WriteFullFence(long newValue)
|
|
||||||
{
|
|
||||||
Thread.MemoryBarrier();
|
|
||||||
_value = newValue;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Write the value applying a compiler fence only, no CPU fence is applied
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="newValue">The new value</param>
|
|
||||||
[MethodImpl(MethodImplOptions.NoOptimization)]
|
|
||||||
public void WriteCompilerOnlyFence(long newValue)
|
|
||||||
{
|
|
||||||
_value = newValue;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Write without applying any fence
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="newValue">The new value</param>
|
|
||||||
public void WriteUnfenced(long newValue)
|
|
||||||
{
|
|
||||||
_value = newValue;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Atomically set the value to the given updated value if the current value equals the comparand
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="newValue">The new value</param>
|
|
||||||
/// <param name="comparand">The comparand (expected value)</param>
|
|
||||||
/// <returns></returns>
|
|
||||||
public bool AtomicCompareExchange(long newValue, long comparand)
|
|
||||||
{
|
|
||||||
return Interlocked.CompareExchange(ref _value, newValue, comparand) == comparand;
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Atomically set the value to the given updated value
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="newValue">The new value</param>
|
|
||||||
/// <returns>The original value</returns>
|
|
||||||
public long AtomicExchange(long newValue)
|
|
||||||
{
|
|
||||||
return Interlocked.Exchange(ref _value, newValue);
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Atomically add the given value to the current value and return the sum
|
|
||||||
/// </summary>
|
|
||||||
/// <param name="delta">The value to be added</param>
|
|
||||||
/// <returns>The sum of the current value and the given value</returns>
|
|
||||||
public long AtomicAddAndGet(long delta)
|
|
||||||
{
|
|
||||||
return Interlocked.Add(ref _value, delta);
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Atomically increment the current value and return the new value
|
|
||||||
/// </summary>
|
|
||||||
/// <returns>The incremented value.</returns>
|
|
||||||
public long AtomicIncrementAndGet()
|
|
||||||
{
|
|
||||||
return Interlocked.Increment(ref _value);
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Atomically increment the current value and return the new value
|
|
||||||
/// </summary>
|
|
||||||
/// <returns>The decremented value.</returns>
|
|
||||||
public long AtomicDecrementAndGet()
|
|
||||||
{
|
|
||||||
return Interlocked.Decrement(ref _value);
|
|
||||||
}
|
|
||||||
|
|
||||||
/// <summary>
|
|
||||||
/// Returns the string representation of the current value.
|
|
||||||
/// </summary>
|
|
||||||
/// <returns>the string representation of the current value.</returns>
|
|
||||||
public override string ToString()
|
|
||||||
{
|
|
||||||
var value = ReadFullFence();
|
|
||||||
return value.ToString();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,69 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) 2018 Stanislav Denisov
|
|
||||||
*
|
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
* of this software and associated documentation files (the "Software"), to deal
|
|
||||||
* in the Software without restriction, including without limitation the rights
|
|
||||||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
* copies of the Software, and to permit persons to whom the Software is
|
|
||||||
* furnished to do so, subject to the following conditions:
|
|
||||||
*
|
|
||||||
* The above copyright notice and this permission notice shall be included in all
|
|
||||||
* copies or substantial portions of the Software.
|
|
||||||
*
|
|
||||||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
* SOFTWARE.
|
|
||||||
*/
|
|
||||||
|
|
||||||
using System.Threading;
|
|
||||||
using System.Runtime.CompilerServices;
|
|
||||||
|
|
||||||
namespace NetStack.Buffers {
|
|
||||||
public abstract class ArrayPool<T> {
|
|
||||||
#if NET_4_6 || NET_STANDARD_2_0
|
|
||||||
private static ArrayPool<T> s_sharedInstance = null;
|
|
||||||
#else
|
|
||||||
private static volatile ArrayPool<T> s_sharedInstance = null;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
public static ArrayPool<T> Shared {
|
|
||||||
#if NET_4_6 || NET_STANDARD_2_0
|
|
||||||
[MethodImpl(256)]
|
|
||||||
get {
|
|
||||||
return Volatile.Read(ref s_sharedInstance) ?? EnsureSharedCreated();
|
|
||||||
}
|
|
||||||
#else
|
|
||||||
[MethodImpl(256)]
|
|
||||||
get {
|
|
||||||
return s_sharedInstance ?? EnsureSharedCreated();
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
#pragma warning disable 420
|
|
||||||
|
|
||||||
[MethodImpl(MethodImplOptions.NoInlining)]
|
|
||||||
private static ArrayPool<T> EnsureSharedCreated() {
|
|
||||||
Interlocked.CompareExchange(ref s_sharedInstance, Create(), null);
|
|
||||||
|
|
||||||
return s_sharedInstance;
|
|
||||||
}
|
|
||||||
|
|
||||||
public static ArrayPool<T> Create() {
|
|
||||||
return new DefaultArrayPool<T>();
|
|
||||||
}
|
|
||||||
|
|
||||||
public static ArrayPool<T> Create(int maxArrayLength, int maxArraysPerBucket) {
|
|
||||||
return new DefaultArrayPool<T>(maxArrayLength, maxArraysPerBucket);
|
|
||||||
}
|
|
||||||
|
|
||||||
public abstract T[] Rent(int minimumLength);
|
|
||||||
|
|
||||||
public abstract void Return(T[] array, bool clearArray = false);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,82 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) 2018 Stanislav Denisov
|
|
||||||
*
|
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
* of this software and associated documentation files (the "Software"), to deal
|
|
||||||
* in the Software without restriction, including without limitation the rights
|
|
||||||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
* copies of the Software, and to permit persons to whom the Software is
|
|
||||||
* furnished to do so, subject to the following conditions:
|
|
||||||
*
|
|
||||||
* The above copyright notice and this permission notice shall be included in all
|
|
||||||
* copies or substantial portions of the Software.
|
|
||||||
*
|
|
||||||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
* SOFTWARE.
|
|
||||||
*/
|
|
||||||
|
|
||||||
using System;
|
|
||||||
|
|
||||||
#if ENABLE_MONO || ENABLE_IL2CPP
|
|
||||||
using UnityEngine;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
namespace NetStack.Buffers {
|
|
||||||
internal sealed class ArrayPoolEventSource {
|
|
||||||
#if NETSTACK_BUFFERS_LOG
|
|
||||||
internal static readonly ArrayPoolEventSource EventLog = new ArrayPoolEventSource();
|
|
||||||
|
|
||||||
internal enum BufferAllocatedReason : int {
|
|
||||||
Pooled,
|
|
||||||
OverMaximumSize,
|
|
||||||
PoolExhausted
|
|
||||||
}
|
|
||||||
|
|
||||||
internal void BufferAllocated(int bufferId, int bufferSize, int poolId, int bucketId, BufferAllocatedReason reason) {
|
|
||||||
var message = "Buffer allocated (Buffer ID: " + bufferId + ", Buffer size: " + bufferSize + ", Pool ID: " + poolId + ", Bucket ID: " + bucketId + ", Reason: " + reason + ")";
|
|
||||||
|
|
||||||
if (reason == BufferAllocatedReason.Pooled)
|
|
||||||
Log.Info("Buffers", message);
|
|
||||||
else
|
|
||||||
Log.Warning("Buffers", message);
|
|
||||||
}
|
|
||||||
|
|
||||||
internal void BufferRented(int bufferId, int bufferSize, int poolId, int bucketId) {
|
|
||||||
Log.Info("Buffers", "Buffer rented (Buffer ID: " + bufferId + ", Buffer size: " + bufferSize + ", Pool ID: " + poolId + ", Bucket ID: " + bucketId + ")");
|
|
||||||
}
|
|
||||||
|
|
||||||
internal void BufferReturned(int bufferId, int bufferSize, int poolId) {
|
|
||||||
Log.Info("Buffers", "Buffer returned (Buffer ID: " + bufferId + ", Buffer size: " + bufferSize + ", Pool ID: " + poolId + ")");
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
internal static class Log {
|
|
||||||
private static string Output(string module, string message) {
|
|
||||||
return DateTime.Now.ToString("[HH:mm:ss]") + " [NetStack." + module + "] " + message;
|
|
||||||
}
|
|
||||||
|
|
||||||
public static void Info(string module, string message) {
|
|
||||||
#if ENABLE_MONO || ENABLE_IL2CPP
|
|
||||||
Debug.Log(Output(module, message));
|
|
||||||
#else
|
|
||||||
Console.WriteLine(Output(module, message));
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
public static void Warning(string module, string message) {
|
|
||||||
#if ENABLE_MONO || ENABLE_IL2CPP
|
|
||||||
Debug.LogWarning(Output(module, message));
|
|
||||||
#else
|
|
||||||
Console.ForegroundColor = ConsoleColor.Yellow;
|
|
||||||
Console.WriteLine(Output(module, message));
|
|
||||||
Console.ResetColor();
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,134 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) 2018 Stanislav Denisov
|
|
||||||
*
|
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
* of this software and associated documentation files (the "Software"), to deal
|
|
||||||
* in the Software without restriction, including without limitation the rights
|
|
||||||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
* copies of the Software, and to permit persons to whom the Software is
|
|
||||||
* furnished to do so, subject to the following conditions:
|
|
||||||
*
|
|
||||||
* The above copyright notice and this permission notice shall be included in all
|
|
||||||
* copies or substantial portions of the Software.
|
|
||||||
*
|
|
||||||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
* SOFTWARE.
|
|
||||||
*/
|
|
||||||
|
|
||||||
using System;
|
|
||||||
|
|
||||||
namespace NetStack.Buffers {
|
|
||||||
internal sealed partial class DefaultArrayPool<T> : ArrayPool<T> {
|
|
||||||
private const int DefaultMaxArrayLength = 1024 * 1024;
|
|
||||||
private const int DefaultMaxNumberOfArraysPerBucket = 50;
|
|
||||||
private static T[] s_emptyArray;
|
|
||||||
private readonly Bucket[] _buckets;
|
|
||||||
|
|
||||||
internal DefaultArrayPool() : this(DefaultMaxArrayLength, DefaultMaxNumberOfArraysPerBucket) { }
|
|
||||||
|
|
||||||
internal DefaultArrayPool(int maxArrayLength, int maxArraysPerBucket) {
|
|
||||||
if (maxArrayLength <= 0)
|
|
||||||
throw new ArgumentOutOfRangeException("maxArrayLength");
|
|
||||||
|
|
||||||
if (maxArraysPerBucket <= 0)
|
|
||||||
throw new ArgumentOutOfRangeException("maxArraysPerBucket");
|
|
||||||
|
|
||||||
const int MinimumArrayLength = 0x10, MaximumArrayLength = 0x40000000;
|
|
||||||
|
|
||||||
if (maxArrayLength > MaximumArrayLength)
|
|
||||||
maxArrayLength = MaximumArrayLength;
|
|
||||||
else if (maxArrayLength < MinimumArrayLength)
|
|
||||||
maxArrayLength = MinimumArrayLength;
|
|
||||||
|
|
||||||
int poolId = Id;
|
|
||||||
int maxBuckets = Utilities.SelectBucketIndex(maxArrayLength);
|
|
||||||
var buckets = new Bucket[maxBuckets + 1];
|
|
||||||
|
|
||||||
for (int i = 0; i < buckets.Length; i++) {
|
|
||||||
buckets[i] = new Bucket(Utilities.GetMaxSizeForBucket(i), maxArraysPerBucket, poolId);
|
|
||||||
}
|
|
||||||
|
|
||||||
_buckets = buckets;
|
|
||||||
}
|
|
||||||
|
|
||||||
private int Id {
|
|
||||||
get {
|
|
||||||
return GetHashCode();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public override T[] Rent(int minimumLength) {
|
|
||||||
if (minimumLength < 0)
|
|
||||||
throw new ArgumentOutOfRangeException("minimumLength");
|
|
||||||
else if (minimumLength == 0)
|
|
||||||
return s_emptyArray ?? (s_emptyArray = new T[0]);
|
|
||||||
|
|
||||||
#if NETSTACK_BUFFERS_LOG
|
|
||||||
var log = ArrayPoolEventSource.EventLog;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
T[] buffer = null;
|
|
||||||
int index = Utilities.SelectBucketIndex(minimumLength);
|
|
||||||
|
|
||||||
if (index < _buckets.Length) {
|
|
||||||
const int MaxBucketsToTry = 2;
|
|
||||||
|
|
||||||
int i = index;
|
|
||||||
|
|
||||||
do {
|
|
||||||
buffer = _buckets[i].Rent();
|
|
||||||
|
|
||||||
if (buffer != null) {
|
|
||||||
#if NETSTACK_BUFFERS_LOG
|
|
||||||
log.BufferRented(buffer.GetHashCode(), buffer.Length, Id, _buckets[i].Id);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
return buffer;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
while (++i < _buckets.Length && i != index + MaxBucketsToTry);
|
|
||||||
|
|
||||||
buffer = new T[_buckets[index]._bufferLength];
|
|
||||||
} else {
|
|
||||||
buffer = new T[minimumLength];
|
|
||||||
}
|
|
||||||
|
|
||||||
#if NETSTACK_BUFFERS_LOG
|
|
||||||
int bufferId = buffer.GetHashCode(), bucketId = -1;
|
|
||||||
|
|
||||||
log.BufferRented(bufferId, buffer.Length, Id, bucketId);
|
|
||||||
log.BufferAllocated(bufferId, buffer.Length, Id, bucketId, index >= _buckets.Length ? ArrayPoolEventSource.BufferAllocatedReason.OverMaximumSize : ArrayPoolEventSource.BufferAllocatedReason.PoolExhausted);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
return buffer;
|
|
||||||
}
|
|
||||||
|
|
||||||
public override void Return(T[] array, bool clearArray = false) {
|
|
||||||
if (array == null)
|
|
||||||
throw new ArgumentNullException("array");
|
|
||||||
else if (array.Length == 0)
|
|
||||||
return;
|
|
||||||
|
|
||||||
int bucket = Utilities.SelectBucketIndex(array.Length);
|
|
||||||
|
|
||||||
if (bucket < _buckets.Length) {
|
|
||||||
if (clearArray)
|
|
||||||
Array.Clear(array, 0, array.Length);
|
|
||||||
|
|
||||||
_buckets[bucket].Return(array);
|
|
||||||
}
|
|
||||||
|
|
||||||
#if NETSTACK_BUFFERS_LOG
|
|
||||||
var log = ArrayPoolEventSource.EventLog;
|
|
||||||
|
|
||||||
log.BufferReturned(array.GetHashCode(), array.Length, Id);
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,151 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) 2018 Stanislav Denisov
|
|
||||||
*
|
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
* of this software and associated documentation files (the "Software"), to deal
|
|
||||||
* in the Software without restriction, including without limitation the rights
|
|
||||||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
* copies of the Software, and to permit persons to whom the Software is
|
|
||||||
* furnished to do so, subject to the following conditions:
|
|
||||||
*
|
|
||||||
* The above copyright notice and this permission notice shall be included in all
|
|
||||||
* copies or substantial portions of the Software.
|
|
||||||
*
|
|
||||||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
* SOFTWARE.
|
|
||||||
*/
|
|
||||||
|
|
||||||
using System;
|
|
||||||
using System.Diagnostics;
|
|
||||||
using System.Threading;
|
|
||||||
|
|
||||||
namespace NetStack.Buffers {
|
|
||||||
internal sealed partial class DefaultArrayPool<T> : ArrayPool<T> {
|
|
||||||
private sealed class Bucket {
|
|
||||||
internal readonly int _bufferLength;
|
|
||||||
private readonly T[][] _buffers;
|
|
||||||
#if NETSTACK_BUFFERS_LOG
|
|
||||||
private readonly int _poolId;
|
|
||||||
#endif
|
|
||||||
#if NET_4_6 || NET_STANDARD_2_0
|
|
||||||
private SpinLock _lock;
|
|
||||||
#else
|
|
||||||
private object _lock;
|
|
||||||
#endif
|
|
||||||
private int _index;
|
|
||||||
|
|
||||||
internal Bucket(int bufferLength, int numberOfBuffers, int poolId) {
|
|
||||||
#if NET_4_6 || NET_STANDARD_2_0
|
|
||||||
_lock = new SpinLock();
|
|
||||||
#else
|
|
||||||
_lock = new Object();
|
|
||||||
#endif
|
|
||||||
_buffers = new T[numberOfBuffers][];
|
|
||||||
_bufferLength = bufferLength;
|
|
||||||
|
|
||||||
#if NETSTACK_BUFFERS_LOG
|
|
||||||
_poolId = poolId;
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
#if NET_4_6 || NET_STANDARD_2_0
|
|
||||||
internal int Id => GetHashCode();
|
|
||||||
#else
|
|
||||||
internal int Id {
|
|
||||||
get {
|
|
||||||
return GetHashCode();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
internal T[] Rent() {
|
|
||||||
T[][] buffers = _buffers;
|
|
||||||
T[] buffer = null;
|
|
||||||
bool allocateBuffer = false;
|
|
||||||
|
|
||||||
#if NET_4_6 || NET_STANDARD_2_0
|
|
||||||
bool lockTaken = false;
|
|
||||||
|
|
||||||
try {
|
|
||||||
_lock.Enter(ref lockTaken);
|
|
||||||
|
|
||||||
if (_index < buffers.Length) {
|
|
||||||
buffer = buffers[_index];
|
|
||||||
buffers[_index++] = null;
|
|
||||||
allocateBuffer = buffer == null;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
finally {
|
|
||||||
if (lockTaken)
|
|
||||||
_lock.Exit(false);
|
|
||||||
}
|
|
||||||
#else
|
|
||||||
try {
|
|
||||||
Monitor.Enter(_lock);
|
|
||||||
|
|
||||||
if (_index < buffers.Length) {
|
|
||||||
buffer = buffers[_index];
|
|
||||||
buffers[_index++] = null;
|
|
||||||
allocateBuffer = buffer == null;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
finally {
|
|
||||||
Monitor.Exit(_lock);
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
if (allocateBuffer) {
|
|
||||||
buffer = new T[_bufferLength];
|
|
||||||
|
|
||||||
#if NETSTACK_BUFFERS_LOG
|
|
||||||
var log = ArrayPoolEventSource.EventLog;
|
|
||||||
|
|
||||||
log.BufferAllocated(buffer.GetHashCode(), _bufferLength, _poolId, Id, ArrayPoolEventSource.BufferAllocatedReason.Pooled);
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
return buffer;
|
|
||||||
}
|
|
||||||
|
|
||||||
internal void Return(T[] array) {
|
|
||||||
if (array.Length != _bufferLength)
|
|
||||||
throw new ArgumentException("BufferNotFromPool", "array");
|
|
||||||
|
|
||||||
#if NET_4_6 || NET_STANDARD_2_0
|
|
||||||
bool lockTaken = false;
|
|
||||||
|
|
||||||
try {
|
|
||||||
_lock.Enter(ref lockTaken);
|
|
||||||
|
|
||||||
if (_index != 0)
|
|
||||||
_buffers[--_index] = array;
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
finally {
|
|
||||||
if (lockTaken)
|
|
||||||
_lock.Exit(false);
|
|
||||||
}
|
|
||||||
#else
|
|
||||||
try {
|
|
||||||
Monitor.Enter(_lock);
|
|
||||||
|
|
||||||
if (_index != 0)
|
|
||||||
_buffers[--_index] = array;
|
|
||||||
}
|
|
||||||
|
|
||||||
finally {
|
|
||||||
Monitor.Exit(_lock);
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,84 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) 2018 Stanislav Denisov
|
|
||||||
*
|
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
* of this software and associated documentation files (the "Software"), to deal
|
|
||||||
* in the Software without restriction, including without limitation the rights
|
|
||||||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
* copies of the Software, and to permit persons to whom the Software is
|
|
||||||
* furnished to do so, subject to the following conditions:
|
|
||||||
*
|
|
||||||
* The above copyright notice and this permission notice shall be included in all
|
|
||||||
* copies or substantial portions of the Software.
|
|
||||||
*
|
|
||||||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
* SOFTWARE.
|
|
||||||
*/
|
|
||||||
|
|
||||||
using System.Diagnostics;
|
|
||||||
using System.Runtime.CompilerServices;
|
|
||||||
|
|
||||||
#if ENABLE_MONO || ENABLE_IL2CPP
|
|
||||||
using UnityEngine.Assertions;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
namespace NetStack.Buffers {
|
|
||||||
internal static class Utilities {
|
|
||||||
[MethodImpl(256)]
|
|
||||||
internal static int SelectBucketIndex(int bufferSize) {
|
|
||||||
#if ENABLE_MONO || ENABLE_IL2CPP
|
|
||||||
Assert.IsTrue(bufferSize > 0);
|
|
||||||
#else
|
|
||||||
Debug.Assert(bufferSize > 0);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
uint bitsRemaining = ((uint)bufferSize - 1) >> 4;
|
|
||||||
int poolIndex = 0;
|
|
||||||
|
|
||||||
if (bitsRemaining > 0xFFFF) {
|
|
||||||
bitsRemaining >>= 16;
|
|
||||||
poolIndex = 16;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (bitsRemaining > 0xFF) {
|
|
||||||
bitsRemaining >>= 8;
|
|
||||||
poolIndex += 8;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (bitsRemaining > 0xF) {
|
|
||||||
bitsRemaining >>= 4;
|
|
||||||
poolIndex += 4;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (bitsRemaining > 0x3) {
|
|
||||||
bitsRemaining >>= 2;
|
|
||||||
poolIndex += 2;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (bitsRemaining > 0x1) {
|
|
||||||
bitsRemaining >>= 1;
|
|
||||||
poolIndex += 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
return poolIndex + (int)bitsRemaining;
|
|
||||||
}
|
|
||||||
|
|
||||||
[MethodImpl(256)]
|
|
||||||
internal static int GetMaxSizeForBucket(int binIndex) {
|
|
||||||
int maxSize = 16 << binIndex;
|
|
||||||
|
|
||||||
#if ENABLE_MONO || ENABLE_IL2CPP
|
|
||||||
Assert.IsTrue(maxSize >= 0);
|
|
||||||
#else
|
|
||||||
Debug.Assert(maxSize >= 0);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
return maxSize;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,184 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) 2018 Stanislav Denisov
|
|
||||||
*
|
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
* of this software and associated documentation files (the "Software"), to deal
|
|
||||||
* in the Software without restriction, including without limitation the rights
|
|
||||||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
* copies of the Software, and to permit persons to whom the Software is
|
|
||||||
* furnished to do so, subject to the following conditions:
|
|
||||||
*
|
|
||||||
* The above copyright notice and this permission notice shall be included in all
|
|
||||||
* copies or substantial portions of the Software.
|
|
||||||
*
|
|
||||||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
* SOFTWARE.
|
|
||||||
*/
|
|
||||||
|
|
||||||
using System;
|
|
||||||
using System.Runtime.CompilerServices;
|
|
||||||
|
|
||||||
#if !(ENABLE_MONO || ENABLE_IL2CPP)
|
|
||||||
using System.Numerics;
|
|
||||||
#else
|
|
||||||
using UnityEngine;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
namespace NetStack.Quantization {
|
|
||||||
public struct QuantizedVector2 {
|
|
||||||
public uint x;
|
|
||||||
public uint y;
|
|
||||||
|
|
||||||
public QuantizedVector2(uint x, uint y) {
|
|
||||||
this.x = x;
|
|
||||||
this.y = y;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public struct QuantizedVector3 {
|
|
||||||
public uint x;
|
|
||||||
public uint y;
|
|
||||||
public uint z;
|
|
||||||
|
|
||||||
public QuantizedVector3(uint x, uint y, uint z) {
|
|
||||||
this.x = x;
|
|
||||||
this.y = y;
|
|
||||||
this.z = z;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public struct QuantizedVector4 {
|
|
||||||
public uint x;
|
|
||||||
public uint y;
|
|
||||||
public uint z;
|
|
||||||
public uint w;
|
|
||||||
|
|
||||||
public QuantizedVector4(uint x, uint y, uint z, uint w) {
|
|
||||||
this.x = x;
|
|
||||||
this.y = y;
|
|
||||||
this.z = z;
|
|
||||||
this.w = w;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public static class DeBruijn {
|
|
||||||
public static readonly int[] Lookup = new int[32] {
|
|
||||||
0, 9, 1, 10, 13, 21, 2, 29, 11, 14, 16, 18, 22, 25, 3, 30,
|
|
||||||
8, 12, 20, 28, 15, 17, 24, 7, 19, 27, 23, 6, 26, 5, 4, 31
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
public class BoundedRange {
|
|
||||||
private readonly float minValue;
|
|
||||||
private readonly float maxValue;
|
|
||||||
private readonly float precision;
|
|
||||||
private readonly int requiredBits;
|
|
||||||
private readonly uint mask;
|
|
||||||
|
|
||||||
public BoundedRange(float minValue, float maxValue, float precision) {
|
|
||||||
this.minValue = minValue;
|
|
||||||
this.maxValue = maxValue;
|
|
||||||
this.precision = precision;
|
|
||||||
|
|
||||||
requiredBits = Log2((uint)((maxValue - minValue) * (1.0f / precision) + 0.5f)) + 1;
|
|
||||||
mask = (uint)((1L << requiredBits) - 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
private int Log2(uint value) {
|
|
||||||
value |= value >> 1;
|
|
||||||
value |= value >> 2;
|
|
||||||
value |= value >> 4;
|
|
||||||
value |= value >> 8;
|
|
||||||
value |= value >> 16;
|
|
||||||
|
|
||||||
return DeBruijn.Lookup[(value * 0x07C4ACDDU) >> 27];
|
|
||||||
}
|
|
||||||
|
|
||||||
[MethodImpl(256)]
|
|
||||||
public uint Quantize(float value) {
|
|
||||||
if (value < minValue)
|
|
||||||
value = minValue;
|
|
||||||
else if (value > maxValue)
|
|
||||||
value = maxValue;
|
|
||||||
|
|
||||||
return (uint)((float)((value - minValue) * (1f / precision)) + 0.5f) & mask;
|
|
||||||
}
|
|
||||||
|
|
||||||
[MethodImpl(256)]
|
|
||||||
public float Dequantize(uint data) {
|
|
||||||
float adjusted = ((float)data * precision) + minValue;
|
|
||||||
|
|
||||||
if (adjusted < minValue)
|
|
||||||
adjusted = minValue;
|
|
||||||
else if (adjusted > maxValue)
|
|
||||||
adjusted = maxValue;
|
|
||||||
|
|
||||||
return adjusted;
|
|
||||||
}
|
|
||||||
|
|
||||||
public static QuantizedVector2 Quantize(Vector2 vector2, BoundedRange[] boundedRange) {
|
|
||||||
QuantizedVector2 data = default(QuantizedVector2);
|
|
||||||
|
|
||||||
#if ENABLE_MONO || ENABLE_IL2CPP
|
|
||||||
data.x = boundedRange[0].Quantize(vector2.x);
|
|
||||||
data.y = boundedRange[1].Quantize(vector2.y);
|
|
||||||
#else
|
|
||||||
data.x = boundedRange[0].Quantize(vector2.X);
|
|
||||||
data.y = boundedRange[1].Quantize(vector2.Y);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
return data;
|
|
||||||
}
|
|
||||||
|
|
||||||
public static QuantizedVector3 Quantize(Vector3 vector3, BoundedRange[] boundedRange) {
|
|
||||||
QuantizedVector3 data = default(QuantizedVector3);
|
|
||||||
|
|
||||||
#if ENABLE_MONO || ENABLE_IL2CPP
|
|
||||||
data.x = boundedRange[0].Quantize(vector3.x);
|
|
||||||
data.y = boundedRange[1].Quantize(vector3.y);
|
|
||||||
data.z = boundedRange[2].Quantize(vector3.z);
|
|
||||||
#else
|
|
||||||
data.x = boundedRange[0].Quantize(vector3.X);
|
|
||||||
data.y = boundedRange[1].Quantize(vector3.Y);
|
|
||||||
data.z = boundedRange[2].Quantize(vector3.Z);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
return data;
|
|
||||||
}
|
|
||||||
|
|
||||||
public static QuantizedVector4 Quantize(Vector4 vector4, BoundedRange[] boundedRange) {
|
|
||||||
QuantizedVector4 data = default(QuantizedVector4);
|
|
||||||
|
|
||||||
#if ENABLE_MONO || ENABLE_IL2CPP
|
|
||||||
data.x = boundedRange[0].Quantize(vector4.x);
|
|
||||||
data.y = boundedRange[1].Quantize(vector4.y);
|
|
||||||
data.z = boundedRange[2].Quantize(vector4.z);
|
|
||||||
data.w = boundedRange[3].Quantize(vector4.w);
|
|
||||||
#else
|
|
||||||
data.x = boundedRange[0].Quantize(vector4.X);
|
|
||||||
data.y = boundedRange[1].Quantize(vector4.Y);
|
|
||||||
data.z = boundedRange[2].Quantize(vector4.Z);
|
|
||||||
data.w = boundedRange[3].Quantize(vector4.W);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
return data;
|
|
||||||
}
|
|
||||||
|
|
||||||
public static Vector2 Dequantize(QuantizedVector2 data, BoundedRange[] boundedRange) {
|
|
||||||
return new Vector2(boundedRange[0].Dequantize(data.x), boundedRange[1].Dequantize(data.y));
|
|
||||||
}
|
|
||||||
|
|
||||||
public static Vector3 Dequantize(QuantizedVector3 data, BoundedRange[] boundedRange) {
|
|
||||||
return new Vector3(boundedRange[0].Dequantize(data.x), boundedRange[1].Dequantize(data.y), boundedRange[2].Dequantize(data.z));
|
|
||||||
}
|
|
||||||
|
|
||||||
public static Vector4 Dequantize(QuantizedVector4 data, BoundedRange[] boundedRange) {
|
|
||||||
return new Vector4(boundedRange[0].Dequantize(data.x), boundedRange[1].Dequantize(data.y), boundedRange[2].Dequantize(data.z), boundedRange[3].Dequantize(data.w));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,117 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) 2018 Stanislav Denisov
|
|
||||||
*
|
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
* of this software and associated documentation files (the "Software"), to deal
|
|
||||||
* in the Software without restriction, including without limitation the rights
|
|
||||||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
* copies of the Software, and to permit persons to whom the Software is
|
|
||||||
* furnished to do so, subject to the following conditions:
|
|
||||||
*
|
|
||||||
* The above copyright notice and this permission notice shall be included in all
|
|
||||||
* copies or substantial portions of the Software.
|
|
||||||
*
|
|
||||||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
* SOFTWARE.
|
|
||||||
*/
|
|
||||||
|
|
||||||
using System.Runtime.CompilerServices;
|
|
||||||
using System.Runtime.InteropServices;
|
|
||||||
|
|
||||||
namespace NetStack.Quantization {
|
|
||||||
public static class HalfPrecision {
|
|
||||||
[StructLayout(LayoutKind.Explicit)]
|
|
||||||
private struct Values {
|
|
||||||
[FieldOffset(0)]
|
|
||||||
public float f;
|
|
||||||
[FieldOffset(0)]
|
|
||||||
public int i;
|
|
||||||
[FieldOffset(0)]
|
|
||||||
public uint u;
|
|
||||||
}
|
|
||||||
|
|
||||||
[MethodImpl(256)]
|
|
||||||
public static ushort Quantize(float value) {
|
|
||||||
var values = new Values {
|
|
||||||
f = value
|
|
||||||
};
|
|
||||||
|
|
||||||
return Quantize(values.i);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static ushort Quantize(int value) {
|
|
||||||
int s = (value >> 16) & 0x00008000;
|
|
||||||
int e = ((value >> 23) & 0X000000FF) - (127 - 15);
|
|
||||||
int m = value & 0X007FFFFF;
|
|
||||||
|
|
||||||
if (e <= 0) {
|
|
||||||
if (e < -10)
|
|
||||||
return (ushort)s;
|
|
||||||
|
|
||||||
m = m | 0x00800000;
|
|
||||||
|
|
||||||
int t = 14 - e;
|
|
||||||
int a = (1 << (t - 1)) - 1;
|
|
||||||
int b = (m >> t) & 1;
|
|
||||||
|
|
||||||
m = (m + a + b) >> t;
|
|
||||||
|
|
||||||
return (ushort)(s | m);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (e == 0XFF - (127 - 15)) {
|
|
||||||
if (m == 0)
|
|
||||||
return (ushort)(s | 0X7C00);
|
|
||||||
|
|
||||||
m >>= 13;
|
|
||||||
|
|
||||||
return (ushort)(s | 0X7C00 | m | ((m == 0) ? 1 : 0));
|
|
||||||
}
|
|
||||||
|
|
||||||
m = m + 0X00000FFF + ((m >> 13) & 1);
|
|
||||||
|
|
||||||
if ((m & 0x00800000) != 0) {
|
|
||||||
m = 0;
|
|
||||||
e++;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (e > 30)
|
|
||||||
return (ushort)(s | 0X7C00);
|
|
||||||
|
|
||||||
return (ushort)(s | (e << 10) | (m >> 13));
|
|
||||||
}
|
|
||||||
|
|
||||||
public static float Dequantize(ushort value) {
|
|
||||||
uint result;
|
|
||||||
uint mantissa = (uint)(value & 1023);
|
|
||||||
uint exponent = 0XFFFFFFF2;
|
|
||||||
|
|
||||||
if ((value & -33792) == 0) {
|
|
||||||
if (mantissa != 0) {
|
|
||||||
while ((mantissa & 1024) == 0) {
|
|
||||||
exponent--;
|
|
||||||
mantissa = mantissa << 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
mantissa &= 0XFFFFFBFF;
|
|
||||||
result = ((uint)((((uint)value & 0x8000) << 16) | ((exponent + 127) << 23))) | (mantissa << 13);
|
|
||||||
} else {
|
|
||||||
result = (uint)((value & 0x8000) << 16);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
result = ((((uint)value & 0x8000) << 16) | ((((((uint)value >> 10) & 0X1F) - 15) + 127) << 23)) | (mantissa << 13);
|
|
||||||
}
|
|
||||||
|
|
||||||
var values = new Values {
|
|
||||||
u = result
|
|
||||||
};
|
|
||||||
|
|
||||||
return values.f;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,223 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) 2020 Stanislav Denisov, Maxim Munning, Davin Carten
|
|
||||||
*
|
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
* of this software and associated documentation files (the "Software"), to deal
|
|
||||||
* in the Software without restriction, including without limitation the rights
|
|
||||||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
* copies of the Software, and to permit persons to whom the Software is
|
|
||||||
* furnished to do so, subject to the following conditions:
|
|
||||||
*
|
|
||||||
* The above copyright notice and this permission notice shall be included in all
|
|
||||||
* copies or substantial portions of the Software.
|
|
||||||
*
|
|
||||||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
* SOFTWARE.
|
|
||||||
*/
|
|
||||||
|
|
||||||
using System;
|
|
||||||
|
|
||||||
#if !(ENABLE_MONO || ENABLE_IL2CPP)
|
|
||||||
using System.Numerics;
|
|
||||||
#else
|
|
||||||
using UnityEngine;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
namespace NetStack.Quantization {
|
|
||||||
public struct QuantizedQuaternion {
|
|
||||||
public uint m;
|
|
||||||
public uint a;
|
|
||||||
public uint b;
|
|
||||||
public uint c;
|
|
||||||
|
|
||||||
public QuantizedQuaternion(uint m, uint a, uint b, uint c) {
|
|
||||||
this.m = m;
|
|
||||||
this.a = a;
|
|
||||||
this.b = b;
|
|
||||||
this.c = c;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public static class SmallestThree {
|
|
||||||
private const float smallestThreeUnpack = 0.70710678118654752440084436210485f + 0.0000001f;
|
|
||||||
private const float smallestThreePack = 1.0f / smallestThreeUnpack;
|
|
||||||
|
|
||||||
public static QuantizedQuaternion Quantize(Quaternion quaternion, int bitsPerElement = 12) {
|
|
||||||
float halfRange = (1 << bitsPerElement - 1);
|
|
||||||
float packer = smallestThreePack * halfRange;
|
|
||||||
float maxValue = float.MinValue;
|
|
||||||
bool signMinus = false;
|
|
||||||
uint m = 0;
|
|
||||||
uint a = 0;
|
|
||||||
uint b = 0;
|
|
||||||
uint c = 0;
|
|
||||||
|
|
||||||
for (uint i = 0; i <= 3; i++) {
|
|
||||||
float element = 0.0f;
|
|
||||||
float abs = 0.0f;
|
|
||||||
|
|
||||||
switch (i) {
|
|
||||||
#if ENABLE_MONO || ENABLE_IL2CPP
|
|
||||||
case 0:
|
|
||||||
element = quaternion.x;
|
|
||||||
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 1:
|
|
||||||
element = quaternion.y;
|
|
||||||
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 2:
|
|
||||||
element = quaternion.z;
|
|
||||||
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 3:
|
|
||||||
element = quaternion.w;
|
|
||||||
|
|
||||||
break;
|
|
||||||
#else
|
|
||||||
case 0:
|
|
||||||
element = quaternion.X;
|
|
||||||
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 1:
|
|
||||||
element = quaternion.Y;
|
|
||||||
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 2:
|
|
||||||
element = quaternion.Z;
|
|
||||||
|
|
||||||
break;
|
|
||||||
|
|
||||||
case 3:
|
|
||||||
element = quaternion.W;
|
|
||||||
|
|
||||||
break;
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
abs = Math.Abs(element);
|
|
||||||
|
|
||||||
if (abs > maxValue) {
|
|
||||||
signMinus = (element < 0.0f);
|
|
||||||
m = i;
|
|
||||||
maxValue = abs;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
float af = 0.0f;
|
|
||||||
float bf = 0.0f;
|
|
||||||
float cf = 0.0f;
|
|
||||||
|
|
||||||
#if ENABLE_MONO || ENABLE_IL2CPP
|
|
||||||
switch (m) {
|
|
||||||
case 0:
|
|
||||||
af = quaternion.y;
|
|
||||||
bf = quaternion.z;
|
|
||||||
cf = quaternion.w;
|
|
||||||
|
|
||||||
break;
|
|
||||||
case 1:
|
|
||||||
af = quaternion.x;
|
|
||||||
bf = quaternion.z;
|
|
||||||
cf = quaternion.w;
|
|
||||||
|
|
||||||
break;
|
|
||||||
case 2:
|
|
||||||
af = quaternion.x;
|
|
||||||
bf = quaternion.y;
|
|
||||||
cf = quaternion.w;
|
|
||||||
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
af = quaternion.x;
|
|
||||||
bf = quaternion.y;
|
|
||||||
cf = quaternion.z;
|
|
||||||
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
#else
|
|
||||||
switch (m) {
|
|
||||||
case 0:
|
|
||||||
af = quaternion.Y;
|
|
||||||
bf = quaternion.Z;
|
|
||||||
cf = quaternion.W;
|
|
||||||
|
|
||||||
break;
|
|
||||||
case 1:
|
|
||||||
af = quaternion.X;
|
|
||||||
bf = quaternion.Z;
|
|
||||||
cf = quaternion.W;
|
|
||||||
|
|
||||||
break;
|
|
||||||
case 2:
|
|
||||||
af = quaternion.X;
|
|
||||||
bf = quaternion.Y;
|
|
||||||
cf = quaternion.W;
|
|
||||||
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
af = quaternion.X;
|
|
||||||
bf = quaternion.Y;
|
|
||||||
cf = quaternion.Z;
|
|
||||||
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
if (signMinus) {
|
|
||||||
a = (uint)((-af * packer) + halfRange);
|
|
||||||
b = (uint)((-bf * packer) + halfRange);
|
|
||||||
c = (uint)((-cf * packer) + halfRange);
|
|
||||||
} else {
|
|
||||||
a = (uint)((af * packer) + halfRange);
|
|
||||||
b = (uint)((bf * packer) + halfRange);
|
|
||||||
c = (uint)((cf * packer) + halfRange);
|
|
||||||
}
|
|
||||||
|
|
||||||
return new QuantizedQuaternion(m, a, b, c);
|
|
||||||
}
|
|
||||||
|
|
||||||
public static Quaternion Dequantize(QuantizedQuaternion data, int bitsPerElement = 12) {
|
|
||||||
int halfRange = (1 << bitsPerElement - 1);
|
|
||||||
float unpacker = smallestThreeUnpack * (1.0f / halfRange);
|
|
||||||
uint m = data.m;
|
|
||||||
int ai = (int)data.a;
|
|
||||||
int bi = (int)data.b;
|
|
||||||
int ci = (int)data.c;
|
|
||||||
|
|
||||||
ai -= halfRange;
|
|
||||||
bi -= halfRange;
|
|
||||||
ci -= halfRange;
|
|
||||||
|
|
||||||
float a = ai * unpacker;
|
|
||||||
float b = bi * unpacker;
|
|
||||||
float c = ci * unpacker;
|
|
||||||
|
|
||||||
float d = (float)Math.Sqrt(1.0f - ((a * a) + (b * b) + (c * c)));
|
|
||||||
|
|
||||||
switch (m) {
|
|
||||||
case 0:
|
|
||||||
return new Quaternion(d, a, b, c);
|
|
||||||
|
|
||||||
case 1:
|
|
||||||
return new Quaternion(a, d, b, c);
|
|
||||||
|
|
||||||
case 2:
|
|
||||||
return new Quaternion(a, b, d, c);
|
|
||||||
|
|
||||||
default:
|
|
||||||
return new Quaternion(a, b, c, d);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,546 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) 2018 Stanislav Denisov, Maxim Munnig
|
|
||||||
*
|
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
* of this software and associated documentation files (the "Software"), to deal
|
|
||||||
* in the Software without restriction, including without limitation the rights
|
|
||||||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
* copies of the Software, and to permit persons to whom the Software is
|
|
||||||
* furnished to do so, subject to the following conditions:
|
|
||||||
*
|
|
||||||
* The above copyright notice and this permission notice shall be included in all
|
|
||||||
* copies or substantial portions of the Software.
|
|
||||||
*
|
|
||||||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
* SOFTWARE.
|
|
||||||
*/
|
|
||||||
|
|
||||||
/*
|
|
||||||
* Copyright (c) 2018 Alexander Shoulson
|
|
||||||
*
|
|
||||||
* This software is provided 'as-is', without any express or implied
|
|
||||||
* warranty. In no event will the authors be held liable for any damages
|
|
||||||
* arising from the use of this software.
|
|
||||||
* Permission is granted to anyone to use this software for any purpose,
|
|
||||||
* including commercial applications, and to alter it and redistribute it
|
|
||||||
* freely, subject to the following restrictions:
|
|
||||||
*
|
|
||||||
* 1. The origin of this software must not be misrepresented; you must not
|
|
||||||
* claim that you wrote the original software. If you use this software
|
|
||||||
* in a product, an acknowledgment in the product documentation would be
|
|
||||||
* appreciated but is not required.
|
|
||||||
* 2. Altered source versions must be plainly marked as such, and must not be
|
|
||||||
* misrepresented as being the original software.
|
|
||||||
* 3. This notice may not be removed or altered from any source distribution.
|
|
||||||
*/
|
|
||||||
|
|
||||||
using System;
|
|
||||||
using System.Diagnostics;
|
|
||||||
using System.Runtime.CompilerServices;
|
|
||||||
using System.Text;
|
|
||||||
|
|
||||||
#if ENABLE_MONO || ENABLE_IL2CPP
|
|
||||||
using UnityEngine.Assertions;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
namespace NetStack.Serialization {
|
|
||||||
public class BitBuffer {
|
|
||||||
private const int defaultCapacity = 375; // 375 * 4 = 1500 bytes
|
|
||||||
private const int stringLengthBits = 8;
|
|
||||||
private const int stringLengthMax = (1 << stringLengthBits) - 1; // 255
|
|
||||||
private const int bitsASCII = 7;
|
|
||||||
private const int growFactor = 2;
|
|
||||||
private const int minGrow = 1;
|
|
||||||
private int readPosition;
|
|
||||||
private int nextPosition;
|
|
||||||
private uint[] chunks;
|
|
||||||
|
|
||||||
public BitBuffer(int capacity = defaultCapacity) {
|
|
||||||
readPosition = 0;
|
|
||||||
nextPosition = 0;
|
|
||||||
chunks = new uint[capacity];
|
|
||||||
}
|
|
||||||
|
|
||||||
public int Length {
|
|
||||||
get {
|
|
||||||
return ((nextPosition - 1) >> 3) + 1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public bool IsFinished {
|
|
||||||
get {
|
|
||||||
return nextPosition == readPosition;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
[MethodImpl(256)]
|
|
||||||
public void Clear() {
|
|
||||||
readPosition = 0;
|
|
||||||
nextPosition = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
[MethodImpl(256)]
|
|
||||||
public BitBuffer Add(int numBits, uint value) {
|
|
||||||
#if ENABLE_MONO || ENABLE_IL2CPP
|
|
||||||
Assert.IsFalse(numBits < 0); // Pushing negative bits
|
|
||||||
Assert.IsFalse(numBits > 32); // Pushing too many bits
|
|
||||||
#else
|
|
||||||
Debug.Assert(!(numBits < 0));
|
|
||||||
Debug.Assert(!(numBits > 32));
|
|
||||||
#endif
|
|
||||||
|
|
||||||
int index = nextPosition >> 5;
|
|
||||||
int used = nextPosition & 0x0000001F;
|
|
||||||
|
|
||||||
if ((index + 1) >= chunks.Length)
|
|
||||||
ExpandArray();
|
|
||||||
|
|
||||||
ulong chunkMask = ((1UL << used) - 1);
|
|
||||||
ulong scratch = chunks[index] & chunkMask;
|
|
||||||
ulong result = scratch | ((ulong)value << used);
|
|
||||||
|
|
||||||
chunks[index] = (uint)result;
|
|
||||||
chunks[index + 1] = (uint)(result >> 32);
|
|
||||||
nextPosition += numBits;
|
|
||||||
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
[MethodImpl(256)]
|
|
||||||
public uint Read(int numBits) {
|
|
||||||
uint result = Peek(numBits);
|
|
||||||
|
|
||||||
readPosition += numBits;
|
|
||||||
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
[MethodImpl(256)]
|
|
||||||
public uint Peek(int numBits) {
|
|
||||||
#if ENABLE_MONO || ENABLE_IL2CPP
|
|
||||||
Assert.IsFalse(numBits < 0); // Pushing negative bits
|
|
||||||
Assert.IsFalse(numBits > 32); // Pushing too many bits
|
|
||||||
#else
|
|
||||||
Debug.Assert(!(numBits < 0));
|
|
||||||
Debug.Assert(!(numBits > 32));
|
|
||||||
#endif
|
|
||||||
|
|
||||||
int index = readPosition >> 5;
|
|
||||||
int used = readPosition & 0x0000001F;
|
|
||||||
|
|
||||||
ulong chunkMask = ((1UL << numBits) - 1) << used;
|
|
||||||
ulong scratch = (ulong)chunks[index];
|
|
||||||
|
|
||||||
if ((index + 1) < chunks.Length)
|
|
||||||
scratch |= (ulong)chunks[index + 1] << 32;
|
|
||||||
|
|
||||||
ulong result = (scratch & chunkMask) >> used;
|
|
||||||
|
|
||||||
return (uint)result;
|
|
||||||
}
|
|
||||||
|
|
||||||
public int ToArray(byte[] data) {
|
|
||||||
Add(1, 1);
|
|
||||||
|
|
||||||
int numChunks = (nextPosition >> 5) + 1;
|
|
||||||
int length = data.Length;
|
|
||||||
|
|
||||||
for (int i = 0; i < numChunks; i++) {
|
|
||||||
int dataIdx = i * 4;
|
|
||||||
uint chunk = chunks[i];
|
|
||||||
|
|
||||||
if (dataIdx < length)
|
|
||||||
data[dataIdx] = (byte)(chunk);
|
|
||||||
|
|
||||||
if (dataIdx + 1 < length)
|
|
||||||
data[dataIdx + 1] = (byte)(chunk >> 8);
|
|
||||||
|
|
||||||
if (dataIdx + 2 < length)
|
|
||||||
data[dataIdx + 2] = (byte)(chunk >> 16);
|
|
||||||
|
|
||||||
if (dataIdx + 3 < length)
|
|
||||||
data[dataIdx + 3] = (byte)(chunk >> 24);
|
|
||||||
}
|
|
||||||
|
|
||||||
return Length;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void FromArray(byte[] data, int length) {
|
|
||||||
int numChunks = (length / 4) + 1;
|
|
||||||
|
|
||||||
if (chunks.Length < numChunks)
|
|
||||||
chunks = new uint[numChunks];
|
|
||||||
|
|
||||||
for (int i = 0; i < numChunks; i++) {
|
|
||||||
int dataIdx = i * 4;
|
|
||||||
uint chunk = 0;
|
|
||||||
|
|
||||||
if (dataIdx < length)
|
|
||||||
chunk = (uint)data[dataIdx];
|
|
||||||
|
|
||||||
if (dataIdx + 1 < length)
|
|
||||||
chunk = chunk | (uint)data[dataIdx + 1] << 8;
|
|
||||||
|
|
||||||
if (dataIdx + 2 < length)
|
|
||||||
chunk = chunk | (uint)data[dataIdx + 2] << 16;
|
|
||||||
|
|
||||||
if (dataIdx + 3 < length)
|
|
||||||
chunk = chunk | (uint)data[dataIdx + 3] << 24;
|
|
||||||
|
|
||||||
chunks[i] = chunk;
|
|
||||||
}
|
|
||||||
|
|
||||||
int positionInByte = FindHighestBitPosition(data[length - 1]);
|
|
||||||
|
|
||||||
nextPosition = ((length - 1) * 8) + (positionInByte - 1);
|
|
||||||
readPosition = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
#if NETSTACK_SPAN
|
|
||||||
public int ToSpan(ref Span<byte> data) {
|
|
||||||
Add(1, 1);
|
|
||||||
|
|
||||||
int numChunks = (nextPosition >> 5) + 1;
|
|
||||||
int length = data.Length;
|
|
||||||
|
|
||||||
for (int i = 0; i < numChunks; i++) {
|
|
||||||
int dataIdx = i * 4;
|
|
||||||
uint chunk = chunks[i];
|
|
||||||
|
|
||||||
if (dataIdx < length)
|
|
||||||
data[dataIdx] = (byte)(chunk);
|
|
||||||
|
|
||||||
if (dataIdx + 1 < length)
|
|
||||||
data[dataIdx + 1] = (byte)(chunk >> 8);
|
|
||||||
|
|
||||||
if (dataIdx + 2 < length)
|
|
||||||
data[dataIdx + 2] = (byte)(chunk >> 16);
|
|
||||||
|
|
||||||
if (dataIdx + 3 < length)
|
|
||||||
data[dataIdx + 3] = (byte)(chunk >> 24);
|
|
||||||
}
|
|
||||||
|
|
||||||
return Length;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void FromSpan(ref ReadOnlySpan<byte> data, int length) {
|
|
||||||
int numChunks = (length / 4) + 1;
|
|
||||||
|
|
||||||
if (chunks.Length < numChunks)
|
|
||||||
chunks = new uint[numChunks];
|
|
||||||
|
|
||||||
for (int i = 0; i < numChunks; i++) {
|
|
||||||
int dataIdx = i * 4;
|
|
||||||
uint chunk = 0;
|
|
||||||
|
|
||||||
if (dataIdx < length)
|
|
||||||
chunk = (uint)data[dataIdx];
|
|
||||||
|
|
||||||
if (dataIdx + 1 < length)
|
|
||||||
chunk = chunk | (uint)data[dataIdx + 1] << 8;
|
|
||||||
|
|
||||||
if (dataIdx + 2 < length)
|
|
||||||
chunk = chunk | (uint)data[dataIdx + 2] << 16;
|
|
||||||
|
|
||||||
if (dataIdx + 3 < length)
|
|
||||||
chunk = chunk | (uint)data[dataIdx + 3] << 24;
|
|
||||||
|
|
||||||
chunks[i] = chunk;
|
|
||||||
}
|
|
||||||
|
|
||||||
int positionInByte = FindHighestBitPosition(data[length - 1]);
|
|
||||||
|
|
||||||
nextPosition = ((length - 1) * 8) + (positionInByte - 1);
|
|
||||||
readPosition = 0;
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
|
|
||||||
[MethodImpl(256)]
|
|
||||||
public BitBuffer AddBool(bool value) {
|
|
||||||
Add(1, value ? 1U : 0U);
|
|
||||||
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
[MethodImpl(256)]
|
|
||||||
public bool ReadBool() {
|
|
||||||
return Read(1) > 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
[MethodImpl(256)]
|
|
||||||
public bool PeekBool() {
|
|
||||||
return Peek(1) > 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
[MethodImpl(256)]
|
|
||||||
public BitBuffer AddByte(byte value) {
|
|
||||||
Add(8, value);
|
|
||||||
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
[MethodImpl(256)]
|
|
||||||
public byte ReadByte() {
|
|
||||||
return (byte)Read(8);
|
|
||||||
}
|
|
||||||
|
|
||||||
[MethodImpl(256)]
|
|
||||||
public byte PeekByte() {
|
|
||||||
return (byte)Peek(8);
|
|
||||||
}
|
|
||||||
|
|
||||||
[MethodImpl(256)]
|
|
||||||
public BitBuffer AddShort(short value) {
|
|
||||||
AddInt(value);
|
|
||||||
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
[MethodImpl(256)]
|
|
||||||
public short ReadShort() {
|
|
||||||
return (short)ReadInt();
|
|
||||||
}
|
|
||||||
|
|
||||||
[MethodImpl(256)]
|
|
||||||
public short PeekShort() {
|
|
||||||
return (short)PeekInt();
|
|
||||||
}
|
|
||||||
|
|
||||||
[MethodImpl(256)]
|
|
||||||
public BitBuffer AddUShort(ushort value) {
|
|
||||||
AddUInt(value);
|
|
||||||
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
[MethodImpl(256)]
|
|
||||||
public ushort ReadUShort() {
|
|
||||||
return (ushort)ReadUInt();
|
|
||||||
}
|
|
||||||
|
|
||||||
[MethodImpl(256)]
|
|
||||||
public ushort PeekUShort() {
|
|
||||||
return (ushort)PeekUInt();
|
|
||||||
}
|
|
||||||
|
|
||||||
[MethodImpl(256)]
|
|
||||||
public BitBuffer AddInt(int value) {
|
|
||||||
uint zigzag = (uint)((value << 1) ^ (value >> 31));
|
|
||||||
|
|
||||||
AddUInt(zigzag);
|
|
||||||
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
[MethodImpl(256)]
|
|
||||||
public int ReadInt() {
|
|
||||||
uint value = ReadUInt();
|
|
||||||
int zagzig = (int)((value >> 1) ^ (-(int)(value & 1)));
|
|
||||||
|
|
||||||
return zagzig;
|
|
||||||
}
|
|
||||||
|
|
||||||
[MethodImpl(256)]
|
|
||||||
public int PeekInt() {
|
|
||||||
uint value = PeekUInt();
|
|
||||||
int zagzig = (int)((value >> 1) ^ (-(int)(value & 1)));
|
|
||||||
|
|
||||||
return zagzig;
|
|
||||||
}
|
|
||||||
|
|
||||||
[MethodImpl(256)]
|
|
||||||
public BitBuffer AddUInt(uint value) {
|
|
||||||
uint buffer = 0x0u;
|
|
||||||
|
|
||||||
do {
|
|
||||||
buffer = value & 0x7Fu;
|
|
||||||
value >>= 7;
|
|
||||||
|
|
||||||
if (value > 0)
|
|
||||||
buffer |= 0x80u;
|
|
||||||
|
|
||||||
Add(8, buffer);
|
|
||||||
}
|
|
||||||
|
|
||||||
while (value > 0);
|
|
||||||
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
[MethodImpl(256)]
|
|
||||||
public uint ReadUInt() {
|
|
||||||
uint buffer = 0x0u;
|
|
||||||
uint value = 0x0u;
|
|
||||||
int shift = 0;
|
|
||||||
|
|
||||||
do {
|
|
||||||
buffer = Read(8);
|
|
||||||
|
|
||||||
value |= (buffer & 0x7Fu) << shift;
|
|
||||||
shift += 7;
|
|
||||||
}
|
|
||||||
|
|
||||||
while ((buffer & 0x80u) > 0);
|
|
||||||
|
|
||||||
return value;
|
|
||||||
}
|
|
||||||
|
|
||||||
[MethodImpl(256)]
|
|
||||||
public uint PeekUInt() {
|
|
||||||
int tempPosition = readPosition;
|
|
||||||
uint value = ReadUInt();
|
|
||||||
|
|
||||||
readPosition = tempPosition;
|
|
||||||
|
|
||||||
return value;
|
|
||||||
}
|
|
||||||
|
|
||||||
[MethodImpl(256)]
|
|
||||||
public BitBuffer AddLong(long value) {
|
|
||||||
AddInt((int)(value & uint.MaxValue));
|
|
||||||
AddInt((int)(value >> 32));
|
|
||||||
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
[MethodImpl(256)]
|
|
||||||
public long ReadLong() {
|
|
||||||
int low = ReadInt();
|
|
||||||
int high = ReadInt();
|
|
||||||
long value = high;
|
|
||||||
|
|
||||||
return value << 32 | (uint)low;
|
|
||||||
}
|
|
||||||
|
|
||||||
[MethodImpl(256)]
|
|
||||||
public long PeekLong() {
|
|
||||||
int tempPosition = readPosition;
|
|
||||||
long value = ReadLong();
|
|
||||||
|
|
||||||
readPosition = tempPosition;
|
|
||||||
|
|
||||||
return value;
|
|
||||||
}
|
|
||||||
|
|
||||||
[MethodImpl(256)]
|
|
||||||
public BitBuffer AddULong(ulong value) {
|
|
||||||
AddUInt((uint)(value & uint.MaxValue));
|
|
||||||
AddUInt((uint)(value >> 32));
|
|
||||||
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
[MethodImpl(256)]
|
|
||||||
public ulong ReadULong() {
|
|
||||||
uint low = ReadUInt();
|
|
||||||
uint high = ReadUInt();
|
|
||||||
|
|
||||||
return (ulong)high << 32 | low;
|
|
||||||
}
|
|
||||||
|
|
||||||
[MethodImpl(256)]
|
|
||||||
public ulong PeekULong() {
|
|
||||||
int tempPosition = readPosition;
|
|
||||||
ulong value = ReadULong();
|
|
||||||
|
|
||||||
readPosition = tempPosition;
|
|
||||||
|
|
||||||
return value;
|
|
||||||
}
|
|
||||||
|
|
||||||
[MethodImpl(256)]
|
|
||||||
public BitBuffer AddString(string value) {
|
|
||||||
if (value == null)
|
|
||||||
throw new ArgumentNullException("value");
|
|
||||||
|
|
||||||
uint length = (uint)value.Length;
|
|
||||||
|
|
||||||
if (length > stringLengthMax) {
|
|
||||||
length = (uint)stringLengthMax;
|
|
||||||
|
|
||||||
throw new ArgumentOutOfRangeException("value length exceeded");
|
|
||||||
}
|
|
||||||
|
|
||||||
Add(stringLengthBits, length);
|
|
||||||
|
|
||||||
for (int i = 0; i < length; i++) {
|
|
||||||
Add(bitsASCII, ToASCII(value[i]));
|
|
||||||
}
|
|
||||||
|
|
||||||
return this;
|
|
||||||
}
|
|
||||||
|
|
||||||
[MethodImpl(256)]
|
|
||||||
public string ReadString() {
|
|
||||||
StringBuilder builder = new StringBuilder();
|
|
||||||
uint length = Read(stringLengthBits);
|
|
||||||
|
|
||||||
for (int i = 0; i < length; i++) {
|
|
||||||
builder.Append((char)Read(bitsASCII));
|
|
||||||
}
|
|
||||||
|
|
||||||
return builder.ToString();
|
|
||||||
}
|
|
||||||
|
|
||||||
public override string ToString() {
|
|
||||||
StringBuilder builder = new StringBuilder();
|
|
||||||
|
|
||||||
for (int i = chunks.Length - 1; i >= 0; i--) {
|
|
||||||
builder.Append(Convert.ToString(chunks[i], 2).PadLeft(32, '0'));
|
|
||||||
}
|
|
||||||
|
|
||||||
StringBuilder spaced = new StringBuilder();
|
|
||||||
|
|
||||||
for (int i = 0; i < builder.Length; i++) {
|
|
||||||
spaced.Append(builder[i]);
|
|
||||||
|
|
||||||
if (((i + 1) % 8) == 0)
|
|
||||||
spaced.Append(" ");
|
|
||||||
}
|
|
||||||
|
|
||||||
return spaced.ToString();
|
|
||||||
}
|
|
||||||
|
|
||||||
private void ExpandArray() {
|
|
||||||
int newCapacity = (chunks.Length * growFactor) + minGrow;
|
|
||||||
uint[] newChunks = new uint[newCapacity];
|
|
||||||
|
|
||||||
Array.Copy(chunks, newChunks, chunks.Length);
|
|
||||||
chunks = newChunks;
|
|
||||||
}
|
|
||||||
|
|
||||||
[MethodImpl(256)]
|
|
||||||
private static int FindHighestBitPosition(byte data) {
|
|
||||||
int shiftCount = 0;
|
|
||||||
|
|
||||||
while (data > 0) {
|
|
||||||
data >>= 1;
|
|
||||||
shiftCount++;
|
|
||||||
}
|
|
||||||
|
|
||||||
return shiftCount;
|
|
||||||
}
|
|
||||||
|
|
||||||
private static byte ToASCII(char character) {
|
|
||||||
byte value = 0;
|
|
||||||
|
|
||||||
try {
|
|
||||||
value = Convert.ToByte(character);
|
|
||||||
}
|
|
||||||
|
|
||||||
catch (OverflowException) {
|
|
||||||
throw new Exception("Cannot convert to ASCII: " + character);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (value > 127)
|
|
||||||
throw new Exception("Cannot convert to ASCII: " + character);
|
|
||||||
|
|
||||||
return value;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,150 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) 2019 Stanislav Denisov
|
|
||||||
*
|
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
* of this software and associated documentation files (the "Software"), to deal
|
|
||||||
* in the Software without restriction, including without limitation the rights
|
|
||||||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
* copies of the Software, and to permit persons to whom the Software is
|
|
||||||
* furnished to do so, subject to the following conditions:
|
|
||||||
*
|
|
||||||
* The above copyright notice and this permission notice shall be included in all
|
|
||||||
* copies or substantial portions of the Software.
|
|
||||||
*
|
|
||||||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
* SOFTWARE.
|
|
||||||
*/
|
|
||||||
|
|
||||||
using System;
|
|
||||||
using System.Runtime.InteropServices;
|
|
||||||
using System.Threading;
|
|
||||||
|
|
||||||
namespace NetStack.Threading {
|
|
||||||
[StructLayout(LayoutKind.Explicit, Size = 192)]
|
|
||||||
public sealed class ArrayQueue {
|
|
||||||
[FieldOffset(0)]
|
|
||||||
private readonly Entry[] _array;
|
|
||||||
[FieldOffset(8)]
|
|
||||||
private readonly int _arrayMask;
|
|
||||||
[FieldOffset(64)]
|
|
||||||
private int _enqueuePosition;
|
|
||||||
[FieldOffset(128)]
|
|
||||||
private int _dequeuePosition;
|
|
||||||
|
|
||||||
public int Count {
|
|
||||||
get {
|
|
||||||
return _enqueuePosition - _dequeuePosition;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public ArrayQueue(int capacity) {
|
|
||||||
if (capacity < 2)
|
|
||||||
throw new ArgumentException("Queue size should be greater than or equal to two");
|
|
||||||
|
|
||||||
if ((capacity & (capacity - 1)) != 0)
|
|
||||||
throw new ArgumentException("Queue size should be a power of two");
|
|
||||||
|
|
||||||
_arrayMask = capacity - 1;
|
|
||||||
_array = new Entry[capacity];
|
|
||||||
_enqueuePosition = 0;
|
|
||||||
_dequeuePosition = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void Enqueue(object item) {
|
|
||||||
while (true) {
|
|
||||||
if (TryEnqueue(item))
|
|
||||||
break;
|
|
||||||
|
|
||||||
Thread.SpinWait(1);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public bool TryEnqueue(object item) {
|
|
||||||
var array = _array;
|
|
||||||
var position = _enqueuePosition;
|
|
||||||
var index = position & _arrayMask;
|
|
||||||
|
|
||||||
if (array[index].IsSet != 0)
|
|
||||||
return false;
|
|
||||||
|
|
||||||
array[index].element = item;
|
|
||||||
array[index].IsSet = 1;
|
|
||||||
|
|
||||||
#if NET_4_6 || NET_STANDARD_2_0
|
|
||||||
Volatile.Write(ref _enqueuePosition, position + 1);
|
|
||||||
#else
|
|
||||||
Thread.MemoryBarrier();
|
|
||||||
_enqueuePosition = position + 1;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
public object Dequeue() {
|
|
||||||
while (true) {
|
|
||||||
object element;
|
|
||||||
|
|
||||||
if (TryDequeue(out element))
|
|
||||||
return element;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public bool TryDequeue(out object result) {
|
|
||||||
var array = _array;
|
|
||||||
var position = _dequeuePosition;
|
|
||||||
var index = position & _arrayMask;
|
|
||||||
|
|
||||||
if (array[index].IsSet == 0) {
|
|
||||||
result = default(object);
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
result = array[index].element;
|
|
||||||
array[index].element = default(object);
|
|
||||||
array[index].IsSet = 0;
|
|
||||||
|
|
||||||
#if NET_4_6 || NET_STANDARD_2_0
|
|
||||||
Volatile.Write(ref _dequeuePosition, position + 1);
|
|
||||||
#else
|
|
||||||
Thread.MemoryBarrier();
|
|
||||||
_dequeuePosition = position + 1;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
[StructLayout(LayoutKind.Explicit, Size = 16)]
|
|
||||||
private struct Entry {
|
|
||||||
[FieldOffset(0)]
|
|
||||||
private int isSet;
|
|
||||||
[FieldOffset(8)]
|
|
||||||
internal object element;
|
|
||||||
|
|
||||||
internal int IsSet {
|
|
||||||
get {
|
|
||||||
#if NET_4_6 || NET_STANDARD_2_0
|
|
||||||
return Volatile.Read(ref isSet);
|
|
||||||
#else
|
|
||||||
Thread.MemoryBarrier();
|
|
||||||
return isSet;
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
set {
|
|
||||||
#if NET_4_6 || NET_STANDARD_2_0
|
|
||||||
Volatile.Write(ref isSet, value);
|
|
||||||
#else
|
|
||||||
Thread.MemoryBarrier();
|
|
||||||
isSet = value;
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,153 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) 2018 Alexander Nikitin, Stanislav Denisov
|
|
||||||
*
|
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
* of this software and associated documentation files (the "Software"), to deal
|
|
||||||
* in the Software without restriction, including without limitation the rights
|
|
||||||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
* copies of the Software, and to permit persons to whom the Software is
|
|
||||||
* furnished to do so, subject to the following conditions:
|
|
||||||
*
|
|
||||||
* The above copyright notice and this permission notice shall be included in all
|
|
||||||
* copies or substantial portions of the Software.
|
|
||||||
*
|
|
||||||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
* SOFTWARE.
|
|
||||||
*/
|
|
||||||
|
|
||||||
using System;
|
|
||||||
using System.Runtime.InteropServices;
|
|
||||||
using System.Threading;
|
|
||||||
|
|
||||||
namespace NetStack.Threading {
|
|
||||||
[StructLayout(LayoutKind.Explicit, Size = 192)]
|
|
||||||
public sealed class ConcurrentBuffer {
|
|
||||||
[FieldOffset(0)]
|
|
||||||
private readonly Cell[] _buffer;
|
|
||||||
[FieldOffset(8)]
|
|
||||||
private readonly int _bufferMask;
|
|
||||||
[FieldOffset(64)]
|
|
||||||
private int _enqueuePosition;
|
|
||||||
[FieldOffset(128)]
|
|
||||||
private int _dequeuePosition;
|
|
||||||
|
|
||||||
public int Count {
|
|
||||||
get {
|
|
||||||
return _enqueuePosition - _dequeuePosition;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public ConcurrentBuffer(int bufferSize) {
|
|
||||||
if (bufferSize < 2)
|
|
||||||
throw new ArgumentException("Buffer size should be greater than or equal to two");
|
|
||||||
|
|
||||||
if ((bufferSize & (bufferSize - 1)) != 0)
|
|
||||||
throw new ArgumentException("Buffer size should be a power of two");
|
|
||||||
|
|
||||||
_bufferMask = bufferSize - 1;
|
|
||||||
_buffer = new Cell[bufferSize];
|
|
||||||
|
|
||||||
for (var i = 0; i < bufferSize; i++) {
|
|
||||||
_buffer[i] = new Cell(i, null);
|
|
||||||
}
|
|
||||||
|
|
||||||
_enqueuePosition = 0;
|
|
||||||
_dequeuePosition = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void Enqueue(object item) {
|
|
||||||
while (true) {
|
|
||||||
if (TryEnqueue(item))
|
|
||||||
break;
|
|
||||||
|
|
||||||
Thread.SpinWait(1);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public bool TryEnqueue(object item) {
|
|
||||||
do {
|
|
||||||
var buffer = _buffer;
|
|
||||||
var position = _enqueuePosition;
|
|
||||||
var index = position & _bufferMask;
|
|
||||||
var cell = buffer[index];
|
|
||||||
|
|
||||||
if (cell.Sequence == position && Interlocked.CompareExchange(ref _enqueuePosition, position + 1, position) == position) {
|
|
||||||
buffer[index].Element = item;
|
|
||||||
|
|
||||||
#if NET_4_6 || NET_STANDARD_2_0
|
|
||||||
Volatile.Write(ref buffer[index].Sequence, position + 1);
|
|
||||||
#else
|
|
||||||
Thread.MemoryBarrier();
|
|
||||||
buffer[index].Sequence = position + 1;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (cell.Sequence < position)
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
while (true);
|
|
||||||
}
|
|
||||||
|
|
||||||
public object Dequeue() {
|
|
||||||
while (true) {
|
|
||||||
object element;
|
|
||||||
|
|
||||||
if (TryDequeue(out element))
|
|
||||||
return element;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public bool TryDequeue(out object result) {
|
|
||||||
do {
|
|
||||||
var buffer = _buffer;
|
|
||||||
var bufferMask = _bufferMask;
|
|
||||||
var position = _dequeuePosition;
|
|
||||||
var index = position & bufferMask;
|
|
||||||
var cell = buffer[index];
|
|
||||||
|
|
||||||
if (cell.Sequence == position + 1 && Interlocked.CompareExchange(ref _dequeuePosition, position + 1, position) == position) {
|
|
||||||
result = cell.Element;
|
|
||||||
buffer[index].Element = null;
|
|
||||||
|
|
||||||
#if NET_4_6 || NET_STANDARD_2_0
|
|
||||||
Volatile.Write(ref buffer[index].Sequence, position + bufferMask + 1);
|
|
||||||
#else
|
|
||||||
Thread.MemoryBarrier();
|
|
||||||
buffer[index].Sequence = position + bufferMask + 1;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (cell.Sequence < position + 1) {
|
|
||||||
result = default(object);
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
while (true);
|
|
||||||
}
|
|
||||||
|
|
||||||
[StructLayout(LayoutKind.Explicit, Size = 16)]
|
|
||||||
private struct Cell {
|
|
||||||
[FieldOffset(0)]
|
|
||||||
public int Sequence;
|
|
||||||
[FieldOffset(8)]
|
|
||||||
public object Element;
|
|
||||||
|
|
||||||
public Cell(int sequence, object element) {
|
|
||||||
Sequence = sequence;
|
|
||||||
Element = element;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,173 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) 2018 Virgile Bello, Stanislav Denisov
|
|
||||||
*
|
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
* of this software and associated documentation files (the "Software"), to deal
|
|
||||||
* in the Software without restriction, including without limitation the rights
|
|
||||||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
* copies of the Software, and to permit persons to whom the Software is
|
|
||||||
* furnished to do so, subject to the following conditions:
|
|
||||||
*
|
|
||||||
* The above copyright notice and this permission notice shall be included in all
|
|
||||||
* copies or substantial portions of the Software.
|
|
||||||
*
|
|
||||||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
* SOFTWARE.
|
|
||||||
*/
|
|
||||||
|
|
||||||
using System;
|
|
||||||
using System.Threading;
|
|
||||||
|
|
||||||
namespace NetStack.Threading {
|
|
||||||
public sealed class ConcurrentPool<T> where T : class {
|
|
||||||
#if NET_4_6 || NET_STANDARD_2_0
|
|
||||||
private SpinLock _lock;
|
|
||||||
#else
|
|
||||||
private object _lock;
|
|
||||||
#endif
|
|
||||||
private readonly Func<T> _factory;
|
|
||||||
private Segment _head;
|
|
||||||
private Segment _tail;
|
|
||||||
|
|
||||||
public ConcurrentPool(int capacity, Func<T> factory) {
|
|
||||||
#if NET_4_6 || NET_STANDARD_2_0
|
|
||||||
_lock = new SpinLock();
|
|
||||||
#else
|
|
||||||
_lock = new Object();
|
|
||||||
#endif
|
|
||||||
_head = _tail = new Segment(capacity);
|
|
||||||
_factory = factory;
|
|
||||||
}
|
|
||||||
|
|
||||||
public T Acquire() {
|
|
||||||
while (true) {
|
|
||||||
var localHead = _head;
|
|
||||||
var count = localHead.Count;
|
|
||||||
|
|
||||||
if (count == 0) {
|
|
||||||
if (localHead.Next != null) {
|
|
||||||
#if NET_4_6 || NET_STANDARD_2_0
|
|
||||||
bool lockTaken = false;
|
|
||||||
|
|
||||||
try {
|
|
||||||
_lock.Enter(ref lockTaken);
|
|
||||||
|
|
||||||
if (_head.Next != null && _head.Count == 0)
|
|
||||||
_head = _head.Next;
|
|
||||||
}
|
|
||||||
|
|
||||||
finally {
|
|
||||||
if (lockTaken)
|
|
||||||
_lock.Exit(false);
|
|
||||||
}
|
|
||||||
#else
|
|
||||||
try {
|
|
||||||
Monitor.Enter(_lock);
|
|
||||||
|
|
||||||
if (_head.Next != null && _head.Count == 0)
|
|
||||||
_head = _head.Next;
|
|
||||||
}
|
|
||||||
|
|
||||||
finally {
|
|
||||||
Monitor.Exit(_lock);
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
} else {
|
|
||||||
return _factory();
|
|
||||||
}
|
|
||||||
} else if (Interlocked.CompareExchange(ref localHead.Count, count - 1, count) == count) {
|
|
||||||
var localLow = Interlocked.Increment(ref localHead.Low) - 1;
|
|
||||||
var index = localLow & localHead.Mask;
|
|
||||||
T item;
|
|
||||||
#if NET_4_6 || NET_STANDARD_2_0
|
|
||||||
var spinWait = new SpinWait();
|
|
||||||
#endif
|
|
||||||
|
|
||||||
while ((item = Interlocked.Exchange(ref localHead.Items[index], null)) == null) {
|
|
||||||
#if NET_4_6 || NET_STANDARD_2_0
|
|
||||||
spinWait.SpinOnce();
|
|
||||||
#else
|
|
||||||
Thread.SpinWait(1);
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
return item;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public void Release(T item) {
|
|
||||||
while (true) {
|
|
||||||
var localTail = _tail;
|
|
||||||
var count = localTail.Count;
|
|
||||||
|
|
||||||
if (count == localTail.Items.Length) {
|
|
||||||
#if NET_4_6 || NET_STANDARD_2_0
|
|
||||||
bool lockTaken = false;
|
|
||||||
|
|
||||||
try {
|
|
||||||
_lock.Enter(ref lockTaken);
|
|
||||||
|
|
||||||
if (_tail.Next == null && count == _tail.Items.Length)
|
|
||||||
_tail = _tail.Next = new Segment(_tail.Items.Length << 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
finally {
|
|
||||||
if (lockTaken)
|
|
||||||
_lock.Exit(false);
|
|
||||||
}
|
|
||||||
#else
|
|
||||||
try {
|
|
||||||
Monitor.Enter(_lock);
|
|
||||||
|
|
||||||
if (_tail.Next == null && count == _tail.Items.Length)
|
|
||||||
_tail = _tail.Next = new Segment(_tail.Items.Length << 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
finally {
|
|
||||||
Monitor.Exit(_lock);
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
} else if (Interlocked.CompareExchange(ref localTail.Count, count + 1, count) == count) {
|
|
||||||
var localHigh = Interlocked.Increment(ref localTail.High) - 1;
|
|
||||||
var index = localHigh & localTail.Mask;
|
|
||||||
#if NET_4_6 || NET_STANDARD_2_0
|
|
||||||
var spinWait = new SpinWait();
|
|
||||||
#endif
|
|
||||||
|
|
||||||
while (Interlocked.CompareExchange(ref localTail.Items[index], item, null) != null) {
|
|
||||||
#if NET_4_6 || NET_STANDARD_2_0
|
|
||||||
spinWait.SpinOnce();
|
|
||||||
#else
|
|
||||||
Thread.SpinWait(1);
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
private class Segment {
|
|
||||||
public readonly T[] Items;
|
|
||||||
public readonly int Mask;
|
|
||||||
public int Low;
|
|
||||||
public int High;
|
|
||||||
public int Count;
|
|
||||||
public Segment Next;
|
|
||||||
|
|
||||||
public Segment(int size) {
|
|
||||||
if (size <= 0 || ((size & (size - 1)) != 0))
|
|
||||||
throw new ArgumentOutOfRangeException("Segment size must be power of two");
|
|
||||||
|
|
||||||
Items = new T[size];
|
|
||||||
Mask = size - 1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
-63
@@ -1,63 +0,0 @@
|
|||||||
/*
|
|
||||||
* Copyright (c) 2018 Stanislav Denisov
|
|
||||||
*
|
|
||||||
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
* of this software and associated documentation files (the "Software"), to deal
|
|
||||||
* in the Software without restriction, including without limitation the rights
|
|
||||||
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
* copies of the Software, and to permit persons to whom the Software is
|
|
||||||
* furnished to do so, subject to the following conditions:
|
|
||||||
*
|
|
||||||
* The above copyright notice and this permission notice shall be included in all
|
|
||||||
* copies or substantial portions of the Software.
|
|
||||||
*
|
|
||||||
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
* SOFTWARE.
|
|
||||||
*/
|
|
||||||
|
|
||||||
using System;
|
|
||||||
using System.Runtime.CompilerServices;
|
|
||||||
|
|
||||||
namespace NetStack.Unsafe {
|
|
||||||
#if NET_4_6 || NET_STANDARD_2_0
|
|
||||||
public static class Memory {
|
|
||||||
[MethodImpl(256)]
|
|
||||||
public static unsafe void Copy(IntPtr source, int sourceOffset, byte[] destination, int destinationOffset, int length) {
|
|
||||||
if (length > 0) {
|
|
||||||
fixed (byte* destinationPointer = &destination[destinationOffset]) {
|
|
||||||
byte* sourcePointer = (byte*)source + sourceOffset;
|
|
||||||
|
|
||||||
Buffer.MemoryCopy(sourcePointer, destinationPointer, length, length);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
[MethodImpl(256)]
|
|
||||||
public static unsafe void Copy(byte[] source, int sourceOffset, IntPtr destination, int destinationOffset, int length) {
|
|
||||||
if (length > 0) {
|
|
||||||
fixed (byte* sourcePointer = &source[sourceOffset]) {
|
|
||||||
byte* destinationPointer = (byte*)destination + destinationOffset;
|
|
||||||
|
|
||||||
Buffer.MemoryCopy(sourcePointer, destinationPointer, length, length);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
[MethodImpl(256)]
|
|
||||||
public static unsafe void Copy(byte[] source, int sourceOffset, byte[] destination, int destinationOffset, int length) {
|
|
||||||
if (length > 0) {
|
|
||||||
fixed (byte* sourcePointer = &source[sourceOffset]) {
|
|
||||||
fixed (byte* destinationPointer = &destination[destinationOffset]) {
|
|
||||||
Buffer.MemoryCopy(sourcePointer, destinationPointer, length, length);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
@@ -1,35 +0,0 @@
|
|||||||
using System;
|
|
||||||
using System.Runtime.CompilerServices;
|
|
||||||
using NetStack.Buffers;
|
|
||||||
|
|
||||||
namespace Ragon.Common
|
|
||||||
{
|
|
||||||
public static class RagonHeader
|
|
||||||
{
|
|
||||||
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
||||||
public static void WriteUShort(ushort id, ref Span<byte> data) {
|
|
||||||
data[0] = (byte)(id & 0x00FF);
|
|
||||||
data[1] = (byte)((id & 0xFF00) >> 8);
|
|
||||||
}
|
|
||||||
|
|
||||||
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
||||||
public static ushort ReadUShort(ref ReadOnlySpan<byte> data)
|
|
||||||
{
|
|
||||||
return (ushort)(data[0] + (data[1] << 8));
|
|
||||||
}
|
|
||||||
|
|
||||||
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
||||||
public static void WriteInt(int id, ref Span<byte> data) {
|
|
||||||
data[0] = (byte)(id & 0x00FF);
|
|
||||||
data[1] = (byte)((id & 0xFF00) >> 8);
|
|
||||||
data[2] = (byte)((id & 0xFF00) >> 16);
|
|
||||||
data[3] = (byte)((id & 0xFF00) >> 24);
|
|
||||||
}
|
|
||||||
|
|
||||||
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
|
||||||
public static int ReadInt(ref ReadOnlySpan<byte> data)
|
|
||||||
{
|
|
||||||
return (ushort)(data[0] + (data[1] << 8) + (data[2] << 16) + (data[3] << 24));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,10 +0,0 @@
|
|||||||
using NetStack.Serialization;
|
|
||||||
|
|
||||||
namespace Ragon.Common
|
|
||||||
{
|
|
||||||
public interface IRagonSerializable
|
|
||||||
{
|
|
||||||
public void Serialize(BitBuffer buffer);
|
|
||||||
public void Deserialize(BitBuffer buffer);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -0,0 +1,6 @@
|
|||||||
|
namespace Ragon.Core.Time;
|
||||||
|
|
||||||
|
public interface IAction
|
||||||
|
{
|
||||||
|
public void Tick();
|
||||||
|
}
|
||||||
@@ -0,0 +1,28 @@
|
|||||||
|
namespace Ragon.Core.Time;
|
||||||
|
|
||||||
|
public class Loop
|
||||||
|
{
|
||||||
|
private List<IAction> _tasks;
|
||||||
|
|
||||||
|
public Loop()
|
||||||
|
{
|
||||||
|
|
||||||
|
_tasks = new List<IAction>(35);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Run(IAction task)
|
||||||
|
{
|
||||||
|
_tasks.Add(task);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Stop(IAction task)
|
||||||
|
{
|
||||||
|
_tasks.Remove(task);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Tick()
|
||||||
|
{
|
||||||
|
foreach (var task in _tasks)
|
||||||
|
task.Tick();
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,115 @@
|
|||||||
|
using System.Diagnostics;
|
||||||
|
using NLog;
|
||||||
|
using Ragon.Common;
|
||||||
|
using Ragon.Core.Lobby;
|
||||||
|
using Ragon.Core.Server;
|
||||||
|
using Ragon.Core.Time;
|
||||||
|
using Ragon.Server;
|
||||||
|
using Ragon.Server.ENet;
|
||||||
|
|
||||||
|
namespace Ragon.Core;
|
||||||
|
|
||||||
|
public class Application : INetworkListener
|
||||||
|
{
|
||||||
|
private readonly Logger _logger = LogManager.GetCurrentClassLogger();
|
||||||
|
private readonly INetworkServer _server;
|
||||||
|
private readonly Thread _dedicatedThread;
|
||||||
|
private readonly Executor _executor;
|
||||||
|
private readonly Configuration _configuration;
|
||||||
|
private readonly HandlerRegistry _handlerRegistry;
|
||||||
|
private readonly ILobby _lobby;
|
||||||
|
private readonly Loop _loop;
|
||||||
|
private readonly Dictionary<ushort, PlayerContext> _contexts;
|
||||||
|
|
||||||
|
public Application(Configuration configuration)
|
||||||
|
{
|
||||||
|
_configuration = configuration;
|
||||||
|
_executor = new Executor();
|
||||||
|
_dedicatedThread = new Thread(Execute);
|
||||||
|
_dedicatedThread.IsBackground = true;
|
||||||
|
_contexts = new Dictionary<ushort, PlayerContext>();
|
||||||
|
_handlerRegistry = new HandlerRegistry();
|
||||||
|
_lobby = new LobbyInMemory();
|
||||||
|
_loop = new Loop();
|
||||||
|
|
||||||
|
if (configuration.ServerType == "enet")
|
||||||
|
_server = new ENetServer();
|
||||||
|
|
||||||
|
if (configuration.ServerType == "websocket")
|
||||||
|
_server = new NativeWebSocketServer(_executor);
|
||||||
|
|
||||||
|
Debug.Assert(_server != null, $"Socket type not supported: {configuration.ServerType}. Supported: [enet, websocket]");
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Execute()
|
||||||
|
{
|
||||||
|
while (true)
|
||||||
|
{
|
||||||
|
_executor.Execute();
|
||||||
|
_loop.Tick();
|
||||||
|
_server.Poll();
|
||||||
|
|
||||||
|
Thread.Sleep((int)1000.0f / _configuration.ServerTickRate);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Start()
|
||||||
|
{
|
||||||
|
var networkConfiguration = new NetworkConfiguration()
|
||||||
|
{
|
||||||
|
LimitConnections = _configuration.LimitConnections,
|
||||||
|
Protocol = RagonVersion.Parse(_configuration.GameProtocol),
|
||||||
|
Address = "0.0.0.0",
|
||||||
|
Port = _configuration.Port,
|
||||||
|
};
|
||||||
|
|
||||||
|
_server.Start(this, networkConfiguration);
|
||||||
|
_dedicatedThread.Start();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Stop()
|
||||||
|
{
|
||||||
|
_server.Stop();
|
||||||
|
_dedicatedThread.Interrupt();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void OnConnected(INetworkConnection connection)
|
||||||
|
{
|
||||||
|
var context = new PlayerContext(connection, new LobbyPlayer(connection));
|
||||||
|
context.Lobby = _lobby;
|
||||||
|
context.Loop = _loop;
|
||||||
|
|
||||||
|
_logger.Trace($"Connected {connection.Id}");
|
||||||
|
_contexts.Add(connection.Id, context);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void OnDisconnected(INetworkConnection connection)
|
||||||
|
{
|
||||||
|
_logger.Trace($"Disconnected {connection.Id}");
|
||||||
|
|
||||||
|
if (_contexts.Remove(connection.Id, out var context))
|
||||||
|
{
|
||||||
|
var room = context.Room;
|
||||||
|
if (room != null)
|
||||||
|
{
|
||||||
|
room.RemovePlayer(context.RoomPlayer);
|
||||||
|
|
||||||
|
_lobby.RemoveIfEmpty(room);
|
||||||
|
}
|
||||||
|
|
||||||
|
context.Dispose();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void OnTimeout(INetworkConnection connection)
|
||||||
|
{
|
||||||
|
if (_contexts.Remove(connection.Id, out var context))
|
||||||
|
context.Dispose();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void OnData(INetworkConnection connection, byte[] data)
|
||||||
|
{
|
||||||
|
if (_contexts.TryGetValue(connection.Id, out var context))
|
||||||
|
_handlerRegistry.Handle(context, data);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,43 @@
|
|||||||
|
using Newtonsoft.Json;
|
||||||
|
using NLog;
|
||||||
|
|
||||||
|
namespace Ragon.Core;
|
||||||
|
|
||||||
|
[Serializable]
|
||||||
|
public struct Configuration
|
||||||
|
{
|
||||||
|
public string ServerKey;
|
||||||
|
public string ServerType;
|
||||||
|
public ushort ServerTickRate;
|
||||||
|
public string GameProtocol;
|
||||||
|
public ushort Port;
|
||||||
|
public int LimitConnections;
|
||||||
|
public int LimitPlayersPerRoom;
|
||||||
|
public int LimitRooms;
|
||||||
|
|
||||||
|
private static readonly Logger Logger = LogManager.GetCurrentClassLogger();
|
||||||
|
private static readonly string ServerVersion = "1.1.0-rc";
|
||||||
|
|
||||||
|
private static void CopyrightInfo()
|
||||||
|
{
|
||||||
|
Logger.Info($"Server Version: {ServerVersion}");
|
||||||
|
Logger.Info($"Machine Name: {Environment.MachineName}");
|
||||||
|
Logger.Info($"OS: {Environment.OSVersion}");
|
||||||
|
Logger.Info($"Processors: {Environment.ProcessorCount}");
|
||||||
|
Logger.Info($"Runtime Version: {Environment.Version}");
|
||||||
|
Logger.Info("==================================");
|
||||||
|
Logger.Info("| |");
|
||||||
|
Logger.Info("| Ragon |");
|
||||||
|
Logger.Info("| |");
|
||||||
|
Logger.Info("==================================");
|
||||||
|
}
|
||||||
|
|
||||||
|
public static Configuration Load(string filePath)
|
||||||
|
{
|
||||||
|
CopyrightInfo();
|
||||||
|
|
||||||
|
var data = File.ReadAllText(filePath);
|
||||||
|
var configuration = JsonConvert.DeserializeObject<Configuration>(data);
|
||||||
|
return configuration;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,189 @@
|
|||||||
|
using Ragon.Common;
|
||||||
|
|
||||||
|
namespace Ragon.Core.Game;
|
||||||
|
|
||||||
|
public class Entity
|
||||||
|
{
|
||||||
|
private static ushort _idGenerator = 0;
|
||||||
|
|
||||||
|
public ushort Id { get; private set; }
|
||||||
|
public RoomPlayer Owner { get; private set; }
|
||||||
|
public RagonAuthority Authority { get; private set; }
|
||||||
|
public EntityState State { get; private set; }
|
||||||
|
public byte[] Payload { get; private set; }
|
||||||
|
public ushort StaticId { get; private set; }
|
||||||
|
public ushort Type { get; private set; }
|
||||||
|
|
||||||
|
private List<EntityEvent> _bufferedEvents;
|
||||||
|
|
||||||
|
public Entity(RoomPlayer owner, ushort type, ushort staticId, RagonAuthority eventAuthority)
|
||||||
|
{
|
||||||
|
Owner = owner;
|
||||||
|
StaticId = staticId;
|
||||||
|
Type = type;
|
||||||
|
Id = _idGenerator++;
|
||||||
|
Payload = Array.Empty<byte>();
|
||||||
|
Authority = eventAuthority;
|
||||||
|
State = new EntityState(this);
|
||||||
|
|
||||||
|
_bufferedEvents = new List<EntityEvent>();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void SetPayload(byte[] payload)
|
||||||
|
{
|
||||||
|
Payload = payload;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void SetOwner(RoomPlayer owner)
|
||||||
|
{
|
||||||
|
Owner = owner;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void RestoreBufferedEvents(RoomPlayer roomPlayer, RagonSerializer writer)
|
||||||
|
{
|
||||||
|
foreach (var bufferedEvent in _bufferedEvents)
|
||||||
|
{
|
||||||
|
writer.Clear();
|
||||||
|
writer.WriteOperation(RagonOperation.REPLICATE_ENTITY_EVENT);
|
||||||
|
writer.WriteUShort(bufferedEvent.EventId);
|
||||||
|
writer.WriteUShort(bufferedEvent.Invoker.Connection.Id);
|
||||||
|
writer.WriteByte((byte)RagonReplicationMode.Server);
|
||||||
|
writer.WriteUShort(Id);
|
||||||
|
|
||||||
|
ReadOnlySpan<byte> data = bufferedEvent.EventData.AsSpan();
|
||||||
|
writer.WriteData(ref data);
|
||||||
|
|
||||||
|
var sendData = writer.ToArray();
|
||||||
|
roomPlayer.Connection.Reliable.Send(sendData);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Create()
|
||||||
|
{
|
||||||
|
var room = Owner.Room;
|
||||||
|
var serializer = room.Writer;
|
||||||
|
|
||||||
|
serializer.Clear();
|
||||||
|
serializer.WriteOperation(RagonOperation.CREATE_ENTITY);
|
||||||
|
serializer.WriteUShort(Type);
|
||||||
|
serializer.WriteUShort(Id);
|
||||||
|
serializer.WriteUShort(Owner.Connection.Id);
|
||||||
|
|
||||||
|
ReadOnlySpan<byte> entityPayload = Payload.AsSpan();
|
||||||
|
serializer.WriteUShort((ushort)entityPayload.Length);
|
||||||
|
serializer.WriteData(ref entityPayload);
|
||||||
|
|
||||||
|
var sendData = serializer.ToArray();
|
||||||
|
foreach (var player in room.ReadyPlayersList)
|
||||||
|
player.Connection.Reliable.Send(sendData);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Destroy(byte[] payload)
|
||||||
|
{
|
||||||
|
var room = Owner.Room;
|
||||||
|
var serializer = room.Writer;
|
||||||
|
|
||||||
|
serializer.Clear();
|
||||||
|
serializer.WriteOperation(RagonOperation.DESTROY_ENTITY);
|
||||||
|
serializer.WriteInt(Id);
|
||||||
|
serializer.WriteUShort(0);
|
||||||
|
// serializer.WriteData(ref Payload);
|
||||||
|
|
||||||
|
var sendData = serializer.ToArray();
|
||||||
|
foreach (var player in room.ReadyPlayersList)
|
||||||
|
player.Connection.Reliable.Send(sendData);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void ReplicateEvent(
|
||||||
|
RoomPlayer caller,
|
||||||
|
ushort eventId,
|
||||||
|
ReadOnlySpan<byte> payload,
|
||||||
|
RagonReplicationMode eventMode,
|
||||||
|
RoomPlayer targetPlayer
|
||||||
|
)
|
||||||
|
{
|
||||||
|
var room = Owner.Room;
|
||||||
|
var serializer = room.Writer;
|
||||||
|
|
||||||
|
serializer.Clear();
|
||||||
|
serializer.WriteOperation(RagonOperation.REPLICATE_ENTITY_EVENT);
|
||||||
|
serializer.WriteUShort(eventId);
|
||||||
|
serializer.WriteUShort(caller.Connection.Id);
|
||||||
|
serializer.WriteByte((byte)eventMode);
|
||||||
|
serializer.WriteUShort(Id);
|
||||||
|
serializer.WriteData(ref payload);
|
||||||
|
|
||||||
|
var sendData = serializer.ToArray();
|
||||||
|
targetPlayer.Connection.Reliable.Send(sendData);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void ReplicateEvent(
|
||||||
|
RoomPlayer caller,
|
||||||
|
ushort eventId,
|
||||||
|
ReadOnlySpan<byte> payload,
|
||||||
|
RagonReplicationMode eventMode,
|
||||||
|
RagonTarget targetMode
|
||||||
|
)
|
||||||
|
{
|
||||||
|
if (Authority == RagonAuthority.OwnerOnly &&
|
||||||
|
Owner.Connection.Id != caller.Connection.Id)
|
||||||
|
{
|
||||||
|
Console.WriteLine($"Player have not enought authority for event with Id {eventId}");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (eventMode == RagonReplicationMode.Buffered && targetMode != RagonTarget.Owner)
|
||||||
|
{
|
||||||
|
var bufferedEvent = new EntityEvent(caller, eventId, payload.ToArray(), targetMode);
|
||||||
|
_bufferedEvents.Add(bufferedEvent);
|
||||||
|
}
|
||||||
|
|
||||||
|
var room = Owner.Room;
|
||||||
|
var serializer = room.Writer;
|
||||||
|
|
||||||
|
serializer.Clear();
|
||||||
|
serializer.WriteOperation(RagonOperation.REPLICATE_ENTITY_EVENT);
|
||||||
|
serializer.WriteUShort(eventId);
|
||||||
|
serializer.WriteUShort(caller.Connection.Id);
|
||||||
|
serializer.WriteByte((byte)eventMode);
|
||||||
|
serializer.WriteUShort(Id);
|
||||||
|
serializer.WriteData(ref payload);
|
||||||
|
|
||||||
|
var sendData = serializer.ToArray();
|
||||||
|
|
||||||
|
switch (targetMode)
|
||||||
|
{
|
||||||
|
case RagonTarget.Owner:
|
||||||
|
{
|
||||||
|
Owner.Connection.Reliable.Send(sendData);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case RagonTarget.ExceptOwner:
|
||||||
|
{
|
||||||
|
foreach (var roomPlayer in room.ReadyPlayersList)
|
||||||
|
{
|
||||||
|
if (roomPlayer.Connection.Id != Owner.Connection.Id)
|
||||||
|
roomPlayer.Connection.Reliable.Send(sendData);
|
||||||
|
}
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case RagonTarget.ExceptInvoker:
|
||||||
|
{
|
||||||
|
foreach (var roomPlayer in room.ReadyPlayersList)
|
||||||
|
{
|
||||||
|
if (roomPlayer.Connection.Id != caller.Connection.Id)
|
||||||
|
roomPlayer.Connection.Reliable.Send(sendData);
|
||||||
|
}
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case RagonTarget.All:
|
||||||
|
{
|
||||||
|
foreach (var roomPlayer in room.ReadyPlayersList)
|
||||||
|
roomPlayer.Connection.Reliable.Send(sendData);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,24 @@
|
|||||||
|
using Ragon.Common;
|
||||||
|
|
||||||
|
namespace Ragon.Core.Game;
|
||||||
|
|
||||||
|
public class EntityEvent
|
||||||
|
{
|
||||||
|
public RoomPlayer Invoker { get; private set; }
|
||||||
|
public ushort EventId { get; private set; }
|
||||||
|
public byte[] EventData { get; private set; }
|
||||||
|
public RagonTarget Target { set; private get; }
|
||||||
|
|
||||||
|
public EntityEvent(
|
||||||
|
RoomPlayer invoker,
|
||||||
|
ushort eventId,
|
||||||
|
byte[] payload,
|
||||||
|
RagonTarget target
|
||||||
|
)
|
||||||
|
{
|
||||||
|
Invoker = invoker;
|
||||||
|
EventId = eventId;
|
||||||
|
EventData = payload;
|
||||||
|
Target = target;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,34 @@
|
|||||||
|
namespace Ragon.Core.Game;
|
||||||
|
|
||||||
|
public class EntityList
|
||||||
|
{
|
||||||
|
private readonly List<Entity> _dynamicEntitiesList = new List<Entity>();
|
||||||
|
private readonly List<Entity> _staticEntitiesList = new List<Entity>();
|
||||||
|
private readonly Dictionary<ushort, Entity> _entitiesMap = new Dictionary<ushort, Entity>();
|
||||||
|
|
||||||
|
public IReadOnlyList<Entity> StaticList => _staticEntitiesList;
|
||||||
|
public IReadOnlyList<Entity> DynamicList => _dynamicEntitiesList;
|
||||||
|
public IReadOnlyDictionary<ushort, Entity> Map => _entitiesMap;
|
||||||
|
|
||||||
|
public void Add(Entity entity)
|
||||||
|
{
|
||||||
|
if (entity.StaticId != 0)
|
||||||
|
_staticEntitiesList.Add(entity);
|
||||||
|
else
|
||||||
|
_dynamicEntitiesList.Add(entity);
|
||||||
|
|
||||||
|
_entitiesMap.Add(entity.Id, entity);
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool Remove(Entity entity)
|
||||||
|
{
|
||||||
|
if (_entitiesMap.Remove(entity.Id, out var existEntity))
|
||||||
|
{
|
||||||
|
_staticEntitiesList.Remove(entity);
|
||||||
|
_dynamicEntitiesList.Remove(entity);
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,100 @@
|
|||||||
|
using NLog;
|
||||||
|
using Ragon.Common;
|
||||||
|
|
||||||
|
namespace Ragon.Core.Game;
|
||||||
|
|
||||||
|
public class EntityState
|
||||||
|
{
|
||||||
|
private Logger _logger = LogManager.GetCurrentClassLogger();
|
||||||
|
private List<EntityStateProperty> _properties;
|
||||||
|
private Entity _entity;
|
||||||
|
|
||||||
|
public EntityState(Entity entity, int capacity = 10)
|
||||||
|
{
|
||||||
|
_entity = entity;
|
||||||
|
_properties = new List<EntityStateProperty>(10);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void AddProperty(EntityStateProperty property)
|
||||||
|
{
|
||||||
|
_properties.Add(property);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Write(RagonSerializer serializer)
|
||||||
|
{
|
||||||
|
serializer.WriteUShort(_entity.Id);
|
||||||
|
|
||||||
|
for (int propertyIndex = 0; propertyIndex < _properties.Count; propertyIndex++)
|
||||||
|
{
|
||||||
|
var property = _properties[propertyIndex];
|
||||||
|
if (property.IsDirty)
|
||||||
|
{
|
||||||
|
serializer.WriteBool(true);
|
||||||
|
var span = serializer.GetWritableData(property.Size);
|
||||||
|
var data = property.Read();
|
||||||
|
data.CopyTo(span);
|
||||||
|
property.Clear();
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
serializer.WriteBool(false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Read(RagonSerializer serializer)
|
||||||
|
{
|
||||||
|
for (var i = 0; i < _properties.Count; i++)
|
||||||
|
{
|
||||||
|
if (serializer.ReadBool())
|
||||||
|
{
|
||||||
|
var property = _properties[i];
|
||||||
|
var size = property.Size;
|
||||||
|
if (!property.IsFixed)
|
||||||
|
size = serializer.ReadUShort();
|
||||||
|
|
||||||
|
if (size > property.Capacity)
|
||||||
|
{
|
||||||
|
Console.WriteLine($"Property {i} payload too large, size: {size}");
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
var propertyPayload = serializer.ReadData(size);
|
||||||
|
property.Write(ref propertyPayload);
|
||||||
|
property.Size = size;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Snapshot(RagonSerializer serializer)
|
||||||
|
{
|
||||||
|
ReadOnlySpan<byte> payload = _entity.Payload.AsSpan();
|
||||||
|
|
||||||
|
serializer.WriteUShort(_entity.Type);
|
||||||
|
serializer.WriteUShort(_entity.Id);
|
||||||
|
|
||||||
|
if (_entity.StaticId != 0)
|
||||||
|
serializer.WriteUShort(_entity.StaticId);
|
||||||
|
|
||||||
|
serializer.WriteUShort(_entity.Owner.Connection.Id);
|
||||||
|
serializer.WriteUShort((ushort) payload.Length);
|
||||||
|
serializer.WriteData(ref payload);
|
||||||
|
|
||||||
|
for (int propertyIndex = 0; propertyIndex < _properties.Count; propertyIndex++)
|
||||||
|
{
|
||||||
|
var property = _properties[propertyIndex];
|
||||||
|
var hasPayload = property.IsFixed || property.Size > 0 && !property.IsFixed;
|
||||||
|
if (hasPayload)
|
||||||
|
{
|
||||||
|
serializer.WriteBool(true);
|
||||||
|
var span = serializer.GetWritableData(property.Size);
|
||||||
|
var data = property.Read();
|
||||||
|
data.CopyTo(span);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
serializer.WriteBool(false);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,41 @@
|
|||||||
|
using System;
|
||||||
|
using Ragon.Common;
|
||||||
|
|
||||||
|
namespace Ragon.Core.Game;
|
||||||
|
|
||||||
|
public class EntityStateProperty
|
||||||
|
{
|
||||||
|
public int Size { get; set; }
|
||||||
|
public int Capacity { get; set; }
|
||||||
|
public bool IsDirty { get; private set; }
|
||||||
|
public bool IsFixed { get; private set; }
|
||||||
|
private byte[] _data;
|
||||||
|
|
||||||
|
public EntityStateProperty(int size, bool isFixed)
|
||||||
|
{
|
||||||
|
Capacity = 512;
|
||||||
|
Size = size;
|
||||||
|
IsFixed = isFixed;
|
||||||
|
IsDirty = true;
|
||||||
|
|
||||||
|
_data = new byte[Capacity];
|
||||||
|
}
|
||||||
|
|
||||||
|
public ReadOnlySpan<byte> Read()
|
||||||
|
{
|
||||||
|
var dataSpan = _data.AsSpan();
|
||||||
|
|
||||||
|
return dataSpan.Slice(0, Size);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Write(ref ReadOnlySpan<byte> src)
|
||||||
|
{
|
||||||
|
src.CopyTo(_data);
|
||||||
|
IsDirty = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Clear()
|
||||||
|
{
|
||||||
|
IsDirty = false;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,166 @@
|
|||||||
|
using Ragon.Common;
|
||||||
|
using Ragon.Core.Time;
|
||||||
|
|
||||||
|
namespace Ragon.Core.Game;
|
||||||
|
|
||||||
|
public class Room: IAction
|
||||||
|
{
|
||||||
|
public string Id { get; private set; }
|
||||||
|
public RoomInformation Info { get; private set; }
|
||||||
|
public RoomPlayer Owner { get; private set; }
|
||||||
|
public RagonSerializer Writer { get; }
|
||||||
|
public Dictionary<ushort, RoomPlayer> Players { get; private set; }
|
||||||
|
public List<RoomPlayer> WaitPlayersList { get; private set; }
|
||||||
|
public List<RoomPlayer> ReadyPlayersList { get; private set; }
|
||||||
|
public List<RoomPlayer> PlayerList { get; private set; }
|
||||||
|
|
||||||
|
public Dictionary<ushort, Entity> Entities { get; private set; }
|
||||||
|
public List<Entity> DynamicEntitiesList { get; private set; }
|
||||||
|
public List<Entity> StaticEntitiesList { get; private set; }
|
||||||
|
public List<Entity> EntityList { get; private set; }
|
||||||
|
|
||||||
|
private readonly HashSet<Entity> _entitiesDirtySet;
|
||||||
|
|
||||||
|
public Room(string roomId, RoomInformation info)
|
||||||
|
{
|
||||||
|
Id = roomId;
|
||||||
|
Info = info;
|
||||||
|
|
||||||
|
Players = new Dictionary<ushort, RoomPlayer>(info.Max);
|
||||||
|
WaitPlayersList = new List<RoomPlayer>(info.Max);
|
||||||
|
ReadyPlayersList = new List<RoomPlayer>(info.Max);
|
||||||
|
PlayerList = new List<RoomPlayer>(info.Max);
|
||||||
|
|
||||||
|
Entities = new Dictionary<ushort, Entity>();
|
||||||
|
DynamicEntitiesList = new List<Entity>();
|
||||||
|
StaticEntitiesList = new List<Entity>();
|
||||||
|
EntityList = new List<Entity>();
|
||||||
|
|
||||||
|
_entitiesDirtySet = new HashSet<Entity>();
|
||||||
|
Writer = new RagonSerializer(512);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void AttachEntity(RoomPlayer newOwner, Entity entity)
|
||||||
|
{
|
||||||
|
Entities.Add(entity.Id, entity);
|
||||||
|
EntityList.Add(entity);
|
||||||
|
|
||||||
|
if (entity.StaticId == 0)
|
||||||
|
DynamicEntitiesList.Add(entity);
|
||||||
|
else
|
||||||
|
StaticEntitiesList.Add(entity);
|
||||||
|
|
||||||
|
entity.Create();
|
||||||
|
|
||||||
|
newOwner.Entities.Add(entity);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void DetachEntity(RoomPlayer currentOwner, Entity entity, byte[] payload)
|
||||||
|
{
|
||||||
|
Entities.Remove(entity.Id);
|
||||||
|
EntityList.Remove(entity);
|
||||||
|
StaticEntitiesList.Remove(entity);
|
||||||
|
DynamicEntitiesList.Remove(entity);
|
||||||
|
_entitiesDirtySet.Remove(entity);
|
||||||
|
|
||||||
|
entity.Destroy(payload);
|
||||||
|
currentOwner.Entities.Remove(entity);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Tick()
|
||||||
|
{
|
||||||
|
var entities = (ushort) _entitiesDirtySet.Count;
|
||||||
|
if (entities > 0)
|
||||||
|
{
|
||||||
|
Writer.Clear();
|
||||||
|
Writer.WriteOperation(RagonOperation.REPLICATE_ENTITY_STATE);
|
||||||
|
Writer.WriteUShort(entities);
|
||||||
|
|
||||||
|
foreach (var entity in _entitiesDirtySet)
|
||||||
|
entity.State.Write(Writer);
|
||||||
|
|
||||||
|
_entitiesDirtySet.Clear();
|
||||||
|
|
||||||
|
var sendData = Writer.ToArray();
|
||||||
|
foreach (var roomPlayer in ReadyPlayersList)
|
||||||
|
roomPlayer.Connection.Unreliable.Send(sendData);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void AddPlayer(RoomPlayer player)
|
||||||
|
{
|
||||||
|
if (Players.Count == 0)
|
||||||
|
Owner = player;
|
||||||
|
|
||||||
|
player.Attach(this);
|
||||||
|
|
||||||
|
PlayerList.Add(player);
|
||||||
|
Players.Add(player.Connection.Id, player);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void RemovePlayer(RoomPlayer roomPlayer)
|
||||||
|
{
|
||||||
|
if (Players.Remove(roomPlayer.Connection.Id, out var player))
|
||||||
|
{
|
||||||
|
PlayerList.Remove(player);
|
||||||
|
|
||||||
|
{
|
||||||
|
Writer.Clear();
|
||||||
|
Writer.WriteOperation(RagonOperation.PLAYER_LEAVED);
|
||||||
|
Writer.WriteString(player.Id);
|
||||||
|
|
||||||
|
var entitiesToDelete = player.Entities.DynamicList;
|
||||||
|
Writer.WriteUShort((ushort) entitiesToDelete.Count);
|
||||||
|
foreach (var entity in entitiesToDelete)
|
||||||
|
{
|
||||||
|
Writer.WriteUShort(entity.Id);
|
||||||
|
EntityList.Remove(entity);
|
||||||
|
}
|
||||||
|
|
||||||
|
var sendData = Writer.ToArray();
|
||||||
|
Broadcast(sendData);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (roomPlayer.Connection.Id == Owner.Connection.Id && PlayerList.Count > 0)
|
||||||
|
{
|
||||||
|
var nextOwner = PlayerList[0];
|
||||||
|
|
||||||
|
Owner = nextOwner;
|
||||||
|
|
||||||
|
var entitiesToUpdate = roomPlayer.Entities.StaticList;
|
||||||
|
|
||||||
|
Writer.Clear();
|
||||||
|
Writer.WriteOperation(RagonOperation.OWNERSHIP_CHANGED);
|
||||||
|
Writer.WriteString(Owner.Id);
|
||||||
|
Writer.WriteUShort((ushort) entitiesToUpdate.Count);
|
||||||
|
|
||||||
|
foreach (var entity in entitiesToUpdate)
|
||||||
|
{
|
||||||
|
Writer.WriteUShort(entity.Id);
|
||||||
|
|
||||||
|
entity.SetOwner(nextOwner);
|
||||||
|
nextOwner.Entities.Add(entity);
|
||||||
|
}
|
||||||
|
|
||||||
|
var sendData = Writer.ToArray();
|
||||||
|
Broadcast(sendData);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void UpdateReadyPlayerList()
|
||||||
|
{
|
||||||
|
ReadyPlayersList = PlayerList.Where(p => p.IsLoaded).ToList();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Track(Entity entity)
|
||||||
|
{
|
||||||
|
_entitiesDirtySet.Add(entity);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Broadcast(byte[] data)
|
||||||
|
{
|
||||||
|
foreach (var readyPlayer in ReadyPlayersList)
|
||||||
|
readyPlayer.Connection.Reliable.Send(data);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,13 @@
|
|||||||
|
namespace Ragon.Core.Game;
|
||||||
|
|
||||||
|
public class RoomInformation
|
||||||
|
{
|
||||||
|
public string Map { get; init; } = "none";
|
||||||
|
public int Min { get; init; }
|
||||||
|
public int Max { get; init; }
|
||||||
|
|
||||||
|
public override string ToString()
|
||||||
|
{
|
||||||
|
return $"Map: {Map} Count: {Min}/{Max}";
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,36 @@
|
|||||||
|
using Ragon.Server;
|
||||||
|
|
||||||
|
namespace Ragon.Core.Game;
|
||||||
|
|
||||||
|
public class RoomPlayer
|
||||||
|
{
|
||||||
|
public INetworkConnection Connection { get; }
|
||||||
|
public string Id { get; }
|
||||||
|
public string Name { get; }
|
||||||
|
public bool IsLoaded { get; private set; }
|
||||||
|
public Room Room { get; private set; }
|
||||||
|
public EntityList Entities { get; private set; }
|
||||||
|
|
||||||
|
public RoomPlayer(INetworkConnection connection, string id, string name)
|
||||||
|
{
|
||||||
|
Id = id;
|
||||||
|
Name = name;
|
||||||
|
Connection = connection;
|
||||||
|
Entities = new EntityList();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Attach(Room room)
|
||||||
|
{
|
||||||
|
Room = room;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Detach()
|
||||||
|
{
|
||||||
|
Room = null!;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void SetReady()
|
||||||
|
{
|
||||||
|
IsLoaded = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,109 @@
|
|||||||
|
using NLog;
|
||||||
|
using Ragon.Common;
|
||||||
|
using Ragon.Core.Handlers;
|
||||||
|
|
||||||
|
namespace Ragon.Core;
|
||||||
|
|
||||||
|
public sealed class HandlerRegistry
|
||||||
|
{
|
||||||
|
private IHandler _entityEventHandler;
|
||||||
|
private IHandler _entityCreateHandler;
|
||||||
|
private IHandler _entityDestroyHandler;
|
||||||
|
private IHandler _entityStateHandler;
|
||||||
|
private IHandler _sceneLoadedHandler;
|
||||||
|
|
||||||
|
private IHandler _authorizationHandler;
|
||||||
|
private IHandler _joinOrCreateHandler;
|
||||||
|
private IHandler _createHandler;
|
||||||
|
private IHandler _joinHandler;
|
||||||
|
private IHandler _leaveHandler;
|
||||||
|
|
||||||
|
private Logger _logger = LogManager.GetCurrentClassLogger();
|
||||||
|
private RagonSerializer _reader;
|
||||||
|
private RagonSerializer _writer;
|
||||||
|
|
||||||
|
public HandlerRegistry()
|
||||||
|
{
|
||||||
|
_reader = new RagonSerializer(2048);
|
||||||
|
_writer = new RagonSerializer(2048);
|
||||||
|
|
||||||
|
_authorizationHandler = new AuthorizationHandler();
|
||||||
|
_joinOrCreateHandler = new JoinOrCreateHandler();
|
||||||
|
_sceneLoadedHandler = new SceneLoadedHandler();
|
||||||
|
_createHandler = new CreateHandler();
|
||||||
|
_joinHandler = new JoinHandler();
|
||||||
|
_leaveHandler = new LeaveHandler();
|
||||||
|
|
||||||
|
_entityEventHandler = new EntityEventHandler();
|
||||||
|
_entityCreateHandler = new EntityCreateHandler();
|
||||||
|
_entityDestroyHandler = new EntityDestroyHandler();
|
||||||
|
_entityStateHandler = new EntityStateHandler();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Handle(PlayerContext context, byte[] data)
|
||||||
|
{
|
||||||
|
_writer.Clear();
|
||||||
|
_reader.Clear();
|
||||||
|
_reader.FromArray(data);
|
||||||
|
|
||||||
|
var operation = _reader.ReadOperation();
|
||||||
|
switch (operation)
|
||||||
|
{
|
||||||
|
case RagonOperation.REPLICATE_ENTITY_EVENT:
|
||||||
|
{
|
||||||
|
if (context.RoomPlayer != null)
|
||||||
|
_entityEventHandler.Handle(context, _reader, _writer);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case RagonOperation.REPLICATE_ENTITY_STATE:
|
||||||
|
{
|
||||||
|
if (context.RoomPlayer != null)
|
||||||
|
_entityStateHandler.Handle(context, _reader, _writer);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case RagonOperation.CREATE_ENTITY:
|
||||||
|
{
|
||||||
|
if (context.RoomPlayer != null)
|
||||||
|
_entityCreateHandler.Handle(context, _reader, _writer);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case RagonOperation.DESTROY_ENTITY:
|
||||||
|
{
|
||||||
|
if (context.RoomPlayer != null)
|
||||||
|
_entityDestroyHandler.Handle(context, _reader, _writer);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case RagonOperation.SCENE_LOADED:
|
||||||
|
{
|
||||||
|
if (context.RoomPlayer != null)
|
||||||
|
_sceneLoadedHandler.Handle(context, _reader, _writer);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case RagonOperation.JOIN_OR_CREATE_ROOM:
|
||||||
|
{
|
||||||
|
_joinOrCreateHandler.Handle(context, _reader, _writer);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case RagonOperation.CREATE_ROOM:
|
||||||
|
{
|
||||||
|
_createHandler.Handle(context, _reader, _writer);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case RagonOperation.JOIN_ROOM:
|
||||||
|
{
|
||||||
|
_joinHandler.Handle(context, _reader, _writer);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case RagonOperation.LEAVE_ROOM:
|
||||||
|
{
|
||||||
|
_leaveHandler.Handle(context, _reader, _writer);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case RagonOperation.AUTHORIZE:
|
||||||
|
{
|
||||||
|
_authorizationHandler.Handle(context, _reader, _writer);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,8 @@
|
|||||||
|
using Ragon.Common;
|
||||||
|
|
||||||
|
namespace Ragon.Core;
|
||||||
|
|
||||||
|
public interface IHandler
|
||||||
|
{
|
||||||
|
public void Handle(PlayerContext context, RagonSerializer reader, RagonSerializer writer);
|
||||||
|
}
|
||||||
@@ -0,0 +1,39 @@
|
|||||||
|
using NLog;
|
||||||
|
using Ragon.Common;
|
||||||
|
using Ragon.Core.Lobby;
|
||||||
|
|
||||||
|
namespace Ragon.Core.Handlers;
|
||||||
|
|
||||||
|
public sealed class AuthorizationHandler: IHandler
|
||||||
|
{
|
||||||
|
private Logger _logger = LogManager.GetCurrentClassLogger();
|
||||||
|
|
||||||
|
public void Handle(PlayerContext context, RagonSerializer reader, RagonSerializer writer)
|
||||||
|
{
|
||||||
|
if (context.LobbyPlayer.Status == LobbyPlayerStatus.Authorized)
|
||||||
|
{
|
||||||
|
_logger.Warn("Player already authorized");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var key = reader.ReadString();
|
||||||
|
var playerName = reader.ReadString();
|
||||||
|
var additionalData = reader.ReadData(reader.Size);
|
||||||
|
|
||||||
|
context.LobbyPlayer.Name = playerName;
|
||||||
|
context.LobbyPlayer.AdditionalData = additionalData.ToArray();
|
||||||
|
context.LobbyPlayer.Status = LobbyPlayerStatus.Authorized;
|
||||||
|
|
||||||
|
var playerId = context.LobbyPlayer.Id;
|
||||||
|
|
||||||
|
writer.Clear();
|
||||||
|
writer.WriteOperation(RagonOperation.AUTHORIZED_SUCCESS);
|
||||||
|
writer.WriteString(playerId);
|
||||||
|
writer.WriteString(playerName);
|
||||||
|
|
||||||
|
var sendData = writer.ToArray();
|
||||||
|
context.Connection.Reliable.Send(sendData);
|
||||||
|
|
||||||
|
_logger.Trace($"Player {context.Connection.Id}|{context.LobbyPlayer.Name} authorized");
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,32 @@
|
|||||||
|
using NLog;
|
||||||
|
using Ragon.Common;
|
||||||
|
using Ragon.Core.Game;
|
||||||
|
|
||||||
|
|
||||||
|
namespace Ragon.Core.Handlers;
|
||||||
|
|
||||||
|
public sealed class EntityCreateHandler: IHandler
|
||||||
|
{
|
||||||
|
private Logger _logger = LogManager.GetCurrentClassLogger();
|
||||||
|
public void Handle(PlayerContext context, RagonSerializer reader, RagonSerializer writer)
|
||||||
|
{
|
||||||
|
var entityType = reader.ReadUShort();
|
||||||
|
var eventAuthority = (RagonAuthority) reader.ReadByte();
|
||||||
|
var propertiesCount = reader.ReadUShort();
|
||||||
|
|
||||||
|
var entity = new Entity(context.RoomPlayer, entityType, 0, eventAuthority);
|
||||||
|
for (var i = 0; i < propertiesCount; i++)
|
||||||
|
{
|
||||||
|
var propertyType = reader.ReadBool();
|
||||||
|
var propertySize = reader.ReadUShort();
|
||||||
|
entity.State.AddProperty(new EntityStateProperty(propertySize, propertyType));
|
||||||
|
}
|
||||||
|
|
||||||
|
var entityPayload = reader.ReadData(reader.Size);
|
||||||
|
entity.SetPayload(entityPayload.ToArray());
|
||||||
|
|
||||||
|
context.Room.AttachEntity(context.RoomPlayer, entity);
|
||||||
|
|
||||||
|
_logger.Trace($"Player {context.Connection.Id}|{context.LobbyPlayer.Name} created entity {entity.Id}:{entity.Type}");
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,21 @@
|
|||||||
|
using NLog;
|
||||||
|
using Ragon.Common;
|
||||||
|
|
||||||
|
namespace Ragon.Core.Handlers;
|
||||||
|
|
||||||
|
public sealed class EntityDestroyHandler: IHandler
|
||||||
|
{
|
||||||
|
private Logger _logger = LogManager.GetCurrentClassLogger();
|
||||||
|
public void Handle(PlayerContext context, RagonSerializer reader, RagonSerializer writer)
|
||||||
|
{
|
||||||
|
var entityId = reader.ReadUShort();
|
||||||
|
if (context.Room.Entities.TryGetValue(entityId, out var entity))
|
||||||
|
{
|
||||||
|
var player = context.RoomPlayer;
|
||||||
|
var payload = reader.ReadData(reader.Size);
|
||||||
|
|
||||||
|
context.Room.DetachEntity(player, entity, Array.Empty<byte>());
|
||||||
|
_logger.Trace($"Player {context.Connection.Id}|{context.LobbyPlayer.Name} destoyed entity {entity.Id}");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,47 @@
|
|||||||
|
using NLog;
|
||||||
|
using Ragon.Common;
|
||||||
|
|
||||||
|
namespace Ragon.Core.Handlers;
|
||||||
|
|
||||||
|
public sealed class EntityEventHandler: IHandler
|
||||||
|
{
|
||||||
|
private Logger _logger = LogManager.GetCurrentClassLogger();
|
||||||
|
|
||||||
|
public void Handle(PlayerContext context, RagonSerializer reader, RagonSerializer writer)
|
||||||
|
{
|
||||||
|
var player = context.RoomPlayer;
|
||||||
|
var room = context.Room;
|
||||||
|
var entityId = reader.ReadUShort();
|
||||||
|
|
||||||
|
if (!room.Entities.TryGetValue(entityId, out var ent))
|
||||||
|
{
|
||||||
|
_logger.Warn($"Entity not found for event with Id {entityId}");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var eventId = reader.ReadUShort();
|
||||||
|
var eventMode = (RagonReplicationMode) reader.ReadByte();
|
||||||
|
var targetMode = (RagonTarget) reader.ReadByte();
|
||||||
|
var payloadData = reader.ReadData(reader.Size);
|
||||||
|
var targetPlayerPeerId = reader.ReadUShort();
|
||||||
|
|
||||||
|
if (targetMode == RagonTarget.Player && context.Room.Players.TryGetValue(targetPlayerPeerId, out var targetPlayer))
|
||||||
|
{
|
||||||
|
Span<byte> payloadRaw = stackalloc byte[payloadData.Length];
|
||||||
|
ReadOnlySpan<byte> payload = payloadRaw;
|
||||||
|
payloadData.CopyTo(payloadRaw);
|
||||||
|
|
||||||
|
_logger.Trace($"Event {eventId} Payload: {payloadData.Length} to {targetMode}");
|
||||||
|
ent.ReplicateEvent(player, eventId, payload, eventMode, targetPlayer);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
Span<byte> payloadRaw = stackalloc byte[payloadData.Length];
|
||||||
|
ReadOnlySpan<byte> payload = payloadRaw;
|
||||||
|
payloadData.CopyTo(payloadRaw);
|
||||||
|
|
||||||
|
_logger.Trace($"Event {eventId} Payload: {payloadData.Length} to {targetMode}");
|
||||||
|
ent.ReplicateEvent(player, eventId, payload, eventMode, targetMode);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,28 @@
|
|||||||
|
using NLog;
|
||||||
|
using Ragon.Common;
|
||||||
|
|
||||||
|
namespace Ragon.Core.Handlers;
|
||||||
|
|
||||||
|
public sealed class EntityStateHandler: IHandler
|
||||||
|
{
|
||||||
|
private ILogger _logger = LogManager.GetCurrentClassLogger();
|
||||||
|
|
||||||
|
public void Handle(PlayerContext context, RagonSerializer reader, RagonSerializer writer)
|
||||||
|
{
|
||||||
|
var room = context.Room;
|
||||||
|
var entitiesCount = reader.ReadUShort();
|
||||||
|
for (var entityIndex = 0; entityIndex < entitiesCount; entityIndex++)
|
||||||
|
{
|
||||||
|
var entityId = reader.ReadUShort();
|
||||||
|
if (room.Entities.TryGetValue(entityId, out var entity))
|
||||||
|
{
|
||||||
|
entity.State.Read(reader);
|
||||||
|
room.Track(entity);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
_logger.Error($"Entity with Id {entityId} not found, replication interrupted");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,84 @@
|
|||||||
|
using NLog;
|
||||||
|
using Ragon.Common;
|
||||||
|
using Ragon.Core.Game;
|
||||||
|
using Ragon.Core.Lobby;
|
||||||
|
|
||||||
|
namespace Ragon.Core.Handlers;
|
||||||
|
|
||||||
|
public sealed class CreateHandler: IHandler
|
||||||
|
{
|
||||||
|
private RagonRoomParameters _roomParameters = new();
|
||||||
|
private Logger _logger = LogManager.GetCurrentClassLogger();
|
||||||
|
|
||||||
|
public void Handle(PlayerContext context, RagonSerializer reader, RagonSerializer writer)
|
||||||
|
{
|
||||||
|
if (context.LobbyPlayer.Status == LobbyPlayerStatus.Unauthorized)
|
||||||
|
{
|
||||||
|
_logger.Warn($"Player {context.Connection.Id} not authorized for this request");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var custom = reader.ReadBool();
|
||||||
|
var roomId = Guid.NewGuid().ToString();
|
||||||
|
|
||||||
|
if (custom)
|
||||||
|
{
|
||||||
|
roomId = reader.ReadString();
|
||||||
|
if (context.Lobby.FindRoomById(roomId, out _))
|
||||||
|
{
|
||||||
|
writer.Clear();
|
||||||
|
writer.WriteOperation(RagonOperation.JOIN_FAILED);
|
||||||
|
writer.WriteString($"Room with id {roomId} already exists");
|
||||||
|
|
||||||
|
var sendData = writer.ToArray();
|
||||||
|
context.Connection.Reliable.Send(sendData);
|
||||||
|
|
||||||
|
_logger.Trace($"Player {context.Connection.Id}|{context.LobbyPlayer.Name} join failed to room {roomId}, room already exist");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
_roomParameters.Deserialize(reader);
|
||||||
|
|
||||||
|
var information = new RoomInformation()
|
||||||
|
{
|
||||||
|
Map = _roomParameters.Map,
|
||||||
|
Max = _roomParameters.Max,
|
||||||
|
Min = _roomParameters.Min,
|
||||||
|
};
|
||||||
|
|
||||||
|
var lobbyPlayer = context.LobbyPlayer;
|
||||||
|
var roomPlayer = new RoomPlayer(lobbyPlayer.Connection, lobbyPlayer.Id, lobbyPlayer.Name);
|
||||||
|
|
||||||
|
var room = new Room(roomId, information);
|
||||||
|
room.AddPlayer(roomPlayer);
|
||||||
|
|
||||||
|
context.Room?.RemovePlayer(context.RoomPlayer);
|
||||||
|
context.Room = room;
|
||||||
|
context.RoomPlayer = roomPlayer;
|
||||||
|
context.Lobby.Persist(room);
|
||||||
|
|
||||||
|
_logger.Trace($"Player {context.Connection.Id}|{context.LobbyPlayer.Name} create room {room.Id} {information}");
|
||||||
|
|
||||||
|
JoinSuccess(roomPlayer, room, writer);
|
||||||
|
|
||||||
|
context.Loop.Run(room);
|
||||||
|
|
||||||
|
_logger.Trace($"Player {context.Connection.Id}|{context.LobbyPlayer.Name} joined to room {room.Id}");
|
||||||
|
}
|
||||||
|
|
||||||
|
private void JoinSuccess(RoomPlayer player, Room room, RagonSerializer writer)
|
||||||
|
{
|
||||||
|
writer.Clear();
|
||||||
|
writer.WriteOperation(RagonOperation.JOIN_SUCCESS);
|
||||||
|
writer.WriteString(room.Id);
|
||||||
|
writer.WriteString(player.Id);
|
||||||
|
writer.WriteString(room.Owner.Id);
|
||||||
|
writer.WriteUShort((ushort) room.Info.Min);
|
||||||
|
writer.WriteUShort((ushort) room.Info.Max);
|
||||||
|
writer.WriteString(room.Info.Map);
|
||||||
|
|
||||||
|
var sendData = writer.ToArray();
|
||||||
|
player.Connection.Reliable.Send(sendData);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,62 @@
|
|||||||
|
using NLog;
|
||||||
|
using Ragon.Common;
|
||||||
|
using Ragon.Core.Game;
|
||||||
|
using Ragon.Core.Lobby;
|
||||||
|
|
||||||
|
namespace Ragon.Core.Handlers;
|
||||||
|
|
||||||
|
public sealed class JoinHandler : IHandler
|
||||||
|
{
|
||||||
|
private Logger _logger = LogManager.GetCurrentClassLogger();
|
||||||
|
|
||||||
|
public void Handle(PlayerContext context, RagonSerializer reader, RagonSerializer writer)
|
||||||
|
{
|
||||||
|
var roomId = reader.ReadString();
|
||||||
|
var lobbyPlayer = context.LobbyPlayer;
|
||||||
|
|
||||||
|
if (!context.Lobby.FindRoomById(roomId, out var existsRoom))
|
||||||
|
{
|
||||||
|
JoinFailed(lobbyPlayer, writer);
|
||||||
|
|
||||||
|
_logger.Trace($"Player {context.Connection.Id}|{context.LobbyPlayer.Name} failed to join room {roomId}");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var roomPlayer = new RoomPlayer(lobbyPlayer.Connection, lobbyPlayer.Id, lobbyPlayer.Name);
|
||||||
|
|
||||||
|
context.Room?.RemovePlayer(context.RoomPlayer);
|
||||||
|
context.Room = existsRoom;
|
||||||
|
context.RoomPlayer = roomPlayer;
|
||||||
|
|
||||||
|
existsRoom.AddPlayer(roomPlayer);
|
||||||
|
|
||||||
|
JoinSuccess(roomPlayer, existsRoom, writer);
|
||||||
|
|
||||||
|
_logger.Trace($"Player {context.Connection.Id}|{context.LobbyPlayer.Name} joined to {existsRoom.Id}");
|
||||||
|
}
|
||||||
|
|
||||||
|
private void JoinSuccess(RoomPlayer player, Room room, RagonSerializer writer)
|
||||||
|
{
|
||||||
|
writer.Clear();
|
||||||
|
writer.WriteOperation(RagonOperation.JOIN_SUCCESS);
|
||||||
|
writer.WriteString(room.Id);
|
||||||
|
writer.WriteString(player.Id);
|
||||||
|
writer.WriteString(room.Owner.Id);
|
||||||
|
writer.WriteUShort((ushort) room.Info.Min);
|
||||||
|
writer.WriteUShort((ushort) room.Info.Max);
|
||||||
|
writer.WriteString(room.Info.Map);
|
||||||
|
|
||||||
|
var sendData = writer.ToArray();
|
||||||
|
player.Connection.Reliable.Send(sendData);
|
||||||
|
}
|
||||||
|
|
||||||
|
private void JoinFailed(LobbyPlayer player, RagonSerializer writer)
|
||||||
|
{
|
||||||
|
writer.Clear();
|
||||||
|
writer.WriteOperation(RagonOperation.JOIN_FAILED);
|
||||||
|
writer.WriteString($"Room not exists");
|
||||||
|
|
||||||
|
var sendData = writer.ToArray();
|
||||||
|
player.Connection.Reliable.Send(sendData);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,81 @@
|
|||||||
|
using NLog;
|
||||||
|
using Ragon.Common;
|
||||||
|
using Ragon.Core.Game;
|
||||||
|
using Ragon.Core.Lobby;
|
||||||
|
|
||||||
|
namespace Ragon.Core.Handlers;
|
||||||
|
|
||||||
|
public sealed class JoinOrCreateHandler : IHandler
|
||||||
|
{
|
||||||
|
private RagonRoomParameters _roomParameters = new();
|
||||||
|
private Logger _logger = LogManager.GetCurrentClassLogger();
|
||||||
|
|
||||||
|
public void Handle(PlayerContext context, RagonSerializer reader, RagonSerializer writer)
|
||||||
|
{
|
||||||
|
if (context.LobbyPlayer.Status == LobbyPlayerStatus.Unauthorized)
|
||||||
|
{
|
||||||
|
_logger.Warn("Player not authorized for this request");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var roomId = Guid.NewGuid().ToString();
|
||||||
|
var lobbyPlayer = context.LobbyPlayer;
|
||||||
|
|
||||||
|
_roomParameters.Deserialize(reader);
|
||||||
|
|
||||||
|
if (context.Lobby.FindRoomByMap(_roomParameters.Map, out var existsRoom))
|
||||||
|
{
|
||||||
|
var roomPlayer = new RoomPlayer(lobbyPlayer.Connection, lobbyPlayer.Id, lobbyPlayer.Name);
|
||||||
|
|
||||||
|
context.Room?.RemovePlayer(context.RoomPlayer);
|
||||||
|
context.Room = existsRoom;
|
||||||
|
context.RoomPlayer = roomPlayer;
|
||||||
|
|
||||||
|
existsRoom.AddPlayer(roomPlayer);
|
||||||
|
|
||||||
|
JoinSuccess(roomPlayer, existsRoom, writer);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
var information = new RoomInformation()
|
||||||
|
{
|
||||||
|
Map = _roomParameters.Map,
|
||||||
|
Max = _roomParameters.Max,
|
||||||
|
Min = _roomParameters.Min,
|
||||||
|
};
|
||||||
|
|
||||||
|
var room = new Room(roomId, information);
|
||||||
|
context.Lobby.Persist(room);
|
||||||
|
|
||||||
|
var roomPlayer = new RoomPlayer(lobbyPlayer.Connection, lobbyPlayer.Id, lobbyPlayer.Name);
|
||||||
|
room.AddPlayer(roomPlayer);
|
||||||
|
|
||||||
|
context.Room?.RemovePlayer(context.RoomPlayer);
|
||||||
|
context.Room = room;
|
||||||
|
context.RoomPlayer = roomPlayer;
|
||||||
|
|
||||||
|
_logger.Trace($"Player {context.Connection.Id}|{context.LobbyPlayer.Name} create room {room.Id} {information}");
|
||||||
|
|
||||||
|
JoinSuccess(roomPlayer, room, writer);
|
||||||
|
|
||||||
|
context.Loop.Run(room);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void JoinSuccess(RoomPlayer player, Room room, RagonSerializer writer)
|
||||||
|
{
|
||||||
|
writer.Clear();
|
||||||
|
writer.WriteOperation(RagonOperation.JOIN_SUCCESS);
|
||||||
|
writer.WriteString(room.Id);
|
||||||
|
writer.WriteString(player.Id);
|
||||||
|
writer.WriteString(room.Owner.Id);
|
||||||
|
writer.WriteUShort((ushort) room.Info.Min);
|
||||||
|
writer.WriteUShort((ushort) room.Info.Max);
|
||||||
|
writer.WriteString(room.Info.Map);
|
||||||
|
|
||||||
|
var sendData = writer.ToArray();
|
||||||
|
player.Connection.Reliable.Send(sendData);
|
||||||
|
|
||||||
|
_logger.Trace($"Joined to room {room.Id}");
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,19 @@
|
|||||||
|
using NLog;
|
||||||
|
using Ragon.Common;
|
||||||
|
|
||||||
|
namespace Ragon.Core.Handlers;
|
||||||
|
|
||||||
|
public sealed class LeaveHandler: IHandler
|
||||||
|
{
|
||||||
|
private Logger _logger = LogManager.GetCurrentClassLogger();
|
||||||
|
public void Handle(PlayerContext context, RagonSerializer reader, RagonSerializer writer)
|
||||||
|
{
|
||||||
|
var room = context.Room;
|
||||||
|
var roomPlayer = context.RoomPlayer;
|
||||||
|
if (room != null)
|
||||||
|
{
|
||||||
|
context.Room?.RemovePlayer(roomPlayer);
|
||||||
|
_logger.Trace($"Player {context.Connection.Id}|{context.LobbyPlayer.Name} leaved from {room.Id}");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,123 @@
|
|||||||
|
using NLog;
|
||||||
|
using Ragon.Common;
|
||||||
|
using Ragon.Core.Game;
|
||||||
|
using Ragon.Core.Lobby;
|
||||||
|
|
||||||
|
namespace Ragon.Core.Handlers;
|
||||||
|
|
||||||
|
public sealed class SceneLoadedHandler : IHandler
|
||||||
|
{
|
||||||
|
private Logger _logger = LogManager.GetCurrentClassLogger();
|
||||||
|
|
||||||
|
public void Handle(PlayerContext context, RagonSerializer reader, RagonSerializer writer)
|
||||||
|
{
|
||||||
|
if (context.LobbyPlayer.Status == LobbyPlayerStatus.Unauthorized)
|
||||||
|
return;
|
||||||
|
|
||||||
|
var owner = context.Room.Owner;
|
||||||
|
var player = context.RoomPlayer;
|
||||||
|
var room = context.Room;
|
||||||
|
|
||||||
|
if (player == owner)
|
||||||
|
{
|
||||||
|
var statics = reader.ReadUShort();
|
||||||
|
for (var staticIndex = 0; staticIndex < statics; staticIndex++)
|
||||||
|
{
|
||||||
|
var entityType = reader.ReadUShort();
|
||||||
|
var eventAuthority = (RagonAuthority) reader.ReadByte();
|
||||||
|
var staticId = reader.ReadUShort();
|
||||||
|
var propertiesCount = reader.ReadUShort();
|
||||||
|
|
||||||
|
var entity = new Entity(player, entityType, staticId, eventAuthority);
|
||||||
|
for (var propertyIndex = 0; propertyIndex < propertiesCount; propertyIndex++)
|
||||||
|
{
|
||||||
|
var propertyType = reader.ReadBool();
|
||||||
|
var propertySize = reader.ReadUShort();
|
||||||
|
entity.State.AddProperty(new EntityStateProperty(propertySize, propertyType));
|
||||||
|
}
|
||||||
|
|
||||||
|
room.AttachEntity(player, entity);
|
||||||
|
}
|
||||||
|
|
||||||
|
_logger.Trace($"Player {context.Connection.Id}|{context.LobbyPlayer.Name} loaded with {statics} scene entities");
|
||||||
|
|
||||||
|
room.WaitPlayersList.Add(player);
|
||||||
|
|
||||||
|
foreach (var roomPlayer in room.WaitPlayersList)
|
||||||
|
{
|
||||||
|
DispatchPlayerJoinExcludePlayer(room, roomPlayer, writer);
|
||||||
|
|
||||||
|
roomPlayer.SetReady();
|
||||||
|
}
|
||||||
|
|
||||||
|
room.UpdateReadyPlayerList();
|
||||||
|
|
||||||
|
DispatchSnapshot(room, room.WaitPlayersList, writer);
|
||||||
|
|
||||||
|
room.WaitPlayersList.Clear();
|
||||||
|
}
|
||||||
|
else if (owner.IsLoaded)
|
||||||
|
{
|
||||||
|
player.SetReady();
|
||||||
|
|
||||||
|
DispatchPlayerJoinExcludePlayer(room, player, writer);
|
||||||
|
|
||||||
|
room.UpdateReadyPlayerList();
|
||||||
|
|
||||||
|
DispatchSnapshot(room, new List<RoomPlayer>() { player }, writer);
|
||||||
|
|
||||||
|
foreach (var entity in room.EntityList)
|
||||||
|
entity.RestoreBufferedEvents(player, writer);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
_logger.Trace($"Player {player.Connection.Id}|{context.LobbyPlayer.Name} waiting owner of room");
|
||||||
|
room.WaitPlayersList.Add(player);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void DispatchPlayerJoinExcludePlayer(Room room, RoomPlayer roomPlayer, RagonSerializer writer)
|
||||||
|
{
|
||||||
|
writer.Clear();
|
||||||
|
writer.WriteOperation(RagonOperation.PLAYER_JOINED);
|
||||||
|
writer.WriteUShort(roomPlayer.Connection.Id);
|
||||||
|
writer.WriteString(roomPlayer.Id);
|
||||||
|
writer.WriteString(roomPlayer.Name);
|
||||||
|
|
||||||
|
var sendData = writer.ToArray();
|
||||||
|
foreach (var awaiter in room.ReadyPlayersList)
|
||||||
|
{
|
||||||
|
if (awaiter != roomPlayer)
|
||||||
|
awaiter.Connection.Reliable.Send(sendData);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private void DispatchSnapshot(Room room, List<RoomPlayer> receviersList, RagonSerializer writer)
|
||||||
|
{
|
||||||
|
writer.Clear();
|
||||||
|
writer.WriteOperation(RagonOperation.SNAPSHOT);
|
||||||
|
writer.WriteUShort((ushort) room.ReadyPlayersList.Count);
|
||||||
|
foreach (var roomPlayer in room.ReadyPlayersList)
|
||||||
|
{
|
||||||
|
writer.WriteUShort(roomPlayer.Connection.Id);
|
||||||
|
writer.WriteString(roomPlayer.Id);
|
||||||
|
writer.WriteString(roomPlayer.Name);
|
||||||
|
}
|
||||||
|
|
||||||
|
var dynamicEntities = room.DynamicEntitiesList;
|
||||||
|
var dynamicEntitiesCount = (ushort) dynamicEntities.Count;
|
||||||
|
writer.WriteUShort(dynamicEntitiesCount);
|
||||||
|
foreach (var entity in dynamicEntities)
|
||||||
|
entity.State.Snapshot(writer);
|
||||||
|
|
||||||
|
var staticEntities = room.StaticEntitiesList;
|
||||||
|
var staticEntitiesCount = (ushort) staticEntities.Count;
|
||||||
|
writer.WriteUShort(staticEntitiesCount);
|
||||||
|
foreach (var entity in staticEntities)
|
||||||
|
entity.State.Snapshot(writer);
|
||||||
|
|
||||||
|
var sendData = writer.ToArray();
|
||||||
|
foreach (var player in receviersList)
|
||||||
|
player.Connection.Reliable.Send(sendData);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,12 @@
|
|||||||
|
using System.Diagnostics.CodeAnalysis;
|
||||||
|
using Ragon.Core.Game;
|
||||||
|
|
||||||
|
namespace Ragon.Core.Lobby;
|
||||||
|
|
||||||
|
public interface ILobby
|
||||||
|
{
|
||||||
|
public bool FindRoomById(string roomId, [MaybeNullWhen(false)] out Room room);
|
||||||
|
public bool FindRoomByMap(string map, [MaybeNullWhen(false)] out Room room);
|
||||||
|
public void Persist(Room room);
|
||||||
|
public void RemoveIfEmpty(Room room);
|
||||||
|
}
|
||||||
@@ -0,0 +1,61 @@
|
|||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Diagnostics.CodeAnalysis;
|
||||||
|
using NLog;
|
||||||
|
using Ragon.Core.Game;
|
||||||
|
|
||||||
|
namespace Ragon.Core.Lobby;
|
||||||
|
|
||||||
|
public class LobbyInMemory : ILobby
|
||||||
|
{
|
||||||
|
private readonly List<Room> _rooms = new();
|
||||||
|
private readonly Logger _logger = LogManager.GetCurrentClassLogger();
|
||||||
|
|
||||||
|
public bool FindRoomById(string roomId, [MaybeNullWhen(false)] out Room room)
|
||||||
|
{
|
||||||
|
foreach (var existRoom in _rooms)
|
||||||
|
{
|
||||||
|
var info = existRoom.Info;
|
||||||
|
if (existRoom.Id == roomId && info.Min < info.Max)
|
||||||
|
{
|
||||||
|
room = existRoom;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
room = default;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
public bool FindRoomByMap(string map, [MaybeNullWhen(false)] out Room room)
|
||||||
|
{
|
||||||
|
foreach (var existRoom in _rooms)
|
||||||
|
{
|
||||||
|
var info = existRoom.Info;
|
||||||
|
if (info.Map == map && existRoom.Players.Count < info.Max)
|
||||||
|
{
|
||||||
|
room = existRoom;
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
room = default;
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Persist(Room room)
|
||||||
|
{
|
||||||
|
_rooms.Add(room);
|
||||||
|
|
||||||
|
foreach (var r in _rooms)
|
||||||
|
_logger.Trace($"Room: {r.Id} {r.Info} Players: {r.Players.Count}");
|
||||||
|
}
|
||||||
|
|
||||||
|
public void RemoveIfEmpty(Room room)
|
||||||
|
{
|
||||||
|
if (room.Players.Count == 0)
|
||||||
|
_rooms.Remove(room);
|
||||||
|
|
||||||
|
foreach (var r in _rooms)
|
||||||
|
_logger.Trace($"Room: {r.Id} {r.Info} Players: {r.Players.Count}");
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,27 @@
|
|||||||
|
using Ragon.Server;
|
||||||
|
|
||||||
|
namespace Ragon.Core.Lobby;
|
||||||
|
|
||||||
|
public enum LobbyPlayerStatus
|
||||||
|
{
|
||||||
|
Unauthorized,
|
||||||
|
Authorized,
|
||||||
|
}
|
||||||
|
|
||||||
|
public class LobbyPlayer
|
||||||
|
{
|
||||||
|
public string Id { get; private set; }
|
||||||
|
public string Name { get; set; }
|
||||||
|
public byte[] AdditionalData { get; set; }
|
||||||
|
public LobbyPlayerStatus Status { get; set; }
|
||||||
|
public INetworkConnection Connection { get; private set; }
|
||||||
|
|
||||||
|
public LobbyPlayer(INetworkConnection connection)
|
||||||
|
{
|
||||||
|
Id = Guid.NewGuid().ToString();
|
||||||
|
Connection = connection;
|
||||||
|
Status = LobbyPlayerStatus.Unauthorized;
|
||||||
|
Name = "None";
|
||||||
|
AdditionalData = Array.Empty<byte>();
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,28 @@
|
|||||||
|
using NLog;
|
||||||
|
using Ragon.Core.Game;
|
||||||
|
using Ragon.Core.Lobby;
|
||||||
|
using Ragon.Core.Time;
|
||||||
|
using Ragon.Server;
|
||||||
|
|
||||||
|
namespace Ragon.Core;
|
||||||
|
|
||||||
|
public class PlayerContext: IDisposable
|
||||||
|
{
|
||||||
|
public INetworkConnection Connection { get; }
|
||||||
|
public Loop Loop;
|
||||||
|
public ILobby Lobby { get; set; }
|
||||||
|
public LobbyPlayer LobbyPlayer { private set; get; }
|
||||||
|
public Room? Room { get; set; }
|
||||||
|
public RoomPlayer? RoomPlayer { get; set; }
|
||||||
|
|
||||||
|
public PlayerContext(INetworkConnection conn, LobbyPlayer player)
|
||||||
|
{
|
||||||
|
Connection = conn;
|
||||||
|
LobbyPlayer = player;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Dispose()
|
||||||
|
{
|
||||||
|
RoomPlayer?.Room.RemovePlayer(RoomPlayer);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,21 @@
|
|||||||
|
<Project Sdk="Microsoft.NET.Sdk">
|
||||||
|
|
||||||
|
<PropertyGroup>
|
||||||
|
<TargetFramework>net6.0</TargetFramework>
|
||||||
|
<ImplicitUsings>enable</ImplicitUsings>
|
||||||
|
<Nullable>enable</Nullable>
|
||||||
|
</PropertyGroup>
|
||||||
|
|
||||||
|
<ItemGroup>
|
||||||
|
<PackageReference Include="Newtonsoft.Json" Version="13.0.2-beta2" />
|
||||||
|
<PackageReference Include="NLog" Version="5.0.5" />
|
||||||
|
</ItemGroup>
|
||||||
|
|
||||||
|
<ItemGroup>
|
||||||
|
<ProjectReference Include="..\Ragon.Protocol\Ragon.Protocol.csproj" />
|
||||||
|
<ProjectReference Include="..\Ragon.Server.ENet\Ragon.Server.ENet.csproj" />
|
||||||
|
<ProjectReference Include="..\Ragon.Server.NativeWebSockets\Ragon.Server.NativeWebSockets.csproj" />
|
||||||
|
<ProjectReference Include="..\Ragon.Server\Ragon.Server.csproj" />
|
||||||
|
</ItemGroup>
|
||||||
|
|
||||||
|
</Project>
|
||||||
@@ -5,6 +5,7 @@
|
|||||||
<Nullable>disable</Nullable>
|
<Nullable>disable</Nullable>
|
||||||
<LangVersion>8</LangVersion>
|
<LangVersion>8</LangVersion>
|
||||||
<TargetFramework>netstandard2.1</TargetFramework>
|
<TargetFramework>netstandard2.1</TargetFramework>
|
||||||
|
<RootNamespace>Ragon.Common</RootNamespace>
|
||||||
</PropertyGroup>
|
</PropertyGroup>
|
||||||
|
|
||||||
<PropertyGroup Condition=" '$(Configuration)' == 'Debug' ">
|
<PropertyGroup Condition=" '$(Configuration)' == 'Debug' ">
|
||||||
@@ -18,8 +19,4 @@
|
|||||||
<OutputPath></OutputPath>
|
<OutputPath></OutputPath>
|
||||||
<DefineConstants>TRACE;NETSTACK_SPAN</DefineConstants>
|
<DefineConstants>TRACE;NETSTACK_SPAN</DefineConstants>
|
||||||
</PropertyGroup>
|
</PropertyGroup>
|
||||||
|
|
||||||
<ItemGroup>
|
|
||||||
<Folder Include="External" />
|
|
||||||
</ItemGroup>
|
|
||||||
</Project>
|
</Project>
|
||||||
@@ -2,7 +2,8 @@ namespace Ragon.Common
|
|||||||
{
|
{
|
||||||
public enum RagonAuthority: byte
|
public enum RagonAuthority: byte
|
||||||
{
|
{
|
||||||
OWNER_ONLY,
|
None,
|
||||||
ALL,
|
OwnerOnly,
|
||||||
|
All,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -1,29 +1,25 @@
|
|||||||
namespace Ragon.Common
|
namespace Ragon.Common
|
||||||
{
|
{
|
||||||
public enum RagonOperation: ushort
|
public enum RagonOperation: byte
|
||||||
{
|
{
|
||||||
AUTHORIZE,
|
AUTHORIZE,
|
||||||
AUTHORIZED_SUCCESS,
|
AUTHORIZED_SUCCESS,
|
||||||
AUTHORIZED_FAILED,
|
AUTHORIZED_FAILED,
|
||||||
|
JOIN_OR_CREATE_ROOM,
|
||||||
|
CREATE_ROOM,
|
||||||
JOIN_ROOM,
|
JOIN_ROOM,
|
||||||
LEAVE_ROOM,
|
LEAVE_ROOM,
|
||||||
|
OWNERSHIP_CHANGED,
|
||||||
|
JOIN_SUCCESS,
|
||||||
|
JOIN_FAILED,
|
||||||
LOAD_SCENE,
|
LOAD_SCENE,
|
||||||
SCENE_IS_LOADED,
|
SCENE_LOADED,
|
||||||
|
|
||||||
PLAYER_JOINED,
|
PLAYER_JOINED,
|
||||||
PLAYER_LEAVED,
|
PLAYER_LEAVED,
|
||||||
|
|
||||||
CREATE_ENTITY,
|
CREATE_ENTITY,
|
||||||
DESTROY_ENTITY,
|
DESTROY_ENTITY,
|
||||||
|
SNAPSHOT,
|
||||||
RESTORE_BEGIN,
|
|
||||||
RESTORE_END,
|
|
||||||
RESTORED,
|
|
||||||
|
|
||||||
REPLICATE_ENTITY_STATE,
|
REPLICATE_ENTITY_STATE,
|
||||||
REPLICATE_ENTITY_EVENT,
|
REPLICATE_ENTITY_EVENT,
|
||||||
REPLICATE_EVENT,
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -0,0 +1,10 @@
|
|||||||
|
namespace Ragon.Common
|
||||||
|
{
|
||||||
|
public enum RagonReplicationMode: byte
|
||||||
|
{
|
||||||
|
Local,
|
||||||
|
Server,
|
||||||
|
LocalAndServer,
|
||||||
|
Buffered,
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,24 @@
|
|||||||
|
|
||||||
|
namespace Ragon.Common
|
||||||
|
{
|
||||||
|
public class RagonRoomParameters: IRagonSerializable
|
||||||
|
{
|
||||||
|
public string Map { get; set; }
|
||||||
|
public int Min { get; set; }
|
||||||
|
public int Max { get; set; }
|
||||||
|
|
||||||
|
public void Serialize(RagonSerializer buffer)
|
||||||
|
{
|
||||||
|
buffer.WriteString(Map);
|
||||||
|
buffer.WriteInt(Min);
|
||||||
|
buffer.WriteInt(Max);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Deserialize(RagonSerializer buffer)
|
||||||
|
{
|
||||||
|
Map = buffer.ReadString();
|
||||||
|
Min = buffer.ReadInt();
|
||||||
|
Max = buffer.ReadInt();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,8 @@
|
|||||||
|
namespace Ragon.Common
|
||||||
|
{
|
||||||
|
public interface IRagonSerializable
|
||||||
|
{
|
||||||
|
public void Serialize(RagonSerializer serializer);
|
||||||
|
public void Deserialize(RagonSerializer serializer);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,335 @@
|
|||||||
|
using System;
|
||||||
|
using System.Runtime.CompilerServices;
|
||||||
|
using System.Runtime.InteropServices;
|
||||||
|
using System.Text;
|
||||||
|
|
||||||
|
|
||||||
|
namespace Ragon.Common
|
||||||
|
{
|
||||||
|
[StructLayout(LayoutKind.Explicit)]
|
||||||
|
internal struct ValueConverter
|
||||||
|
{
|
||||||
|
[FieldOffset(0)] public int Int;
|
||||||
|
[FieldOffset(0)] public float Float;
|
||||||
|
[FieldOffset(0)] public long Long;
|
||||||
|
[FieldOffset(0)] public byte Byte0;
|
||||||
|
[FieldOffset(1)] public byte Byte1;
|
||||||
|
[FieldOffset(2)] public byte Byte2;
|
||||||
|
[FieldOffset(3)] public byte Byte3;
|
||||||
|
[FieldOffset(4)] public byte Byte4;
|
||||||
|
[FieldOffset(5)] public byte Byte5;
|
||||||
|
[FieldOffset(6)] public byte Byte6;
|
||||||
|
[FieldOffset(7)] public byte Byte7;
|
||||||
|
}
|
||||||
|
|
||||||
|
public class RagonSerializer
|
||||||
|
{
|
||||||
|
private byte[] _data;
|
||||||
|
private int _offset;
|
||||||
|
private int _size;
|
||||||
|
|
||||||
|
public int Lenght => _offset;
|
||||||
|
public int Size => _size - _offset;
|
||||||
|
|
||||||
|
public RagonSerializer(int capacity = 256)
|
||||||
|
{
|
||||||
|
_data = new byte[capacity];
|
||||||
|
_offset = 0;
|
||||||
|
_size = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
|
public void Reset()
|
||||||
|
{
|
||||||
|
_size = _offset;
|
||||||
|
_offset = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
|
public void AddOffset(int offset)
|
||||||
|
{
|
||||||
|
_offset += offset;
|
||||||
|
}
|
||||||
|
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
|
public int WriteByte(byte value)
|
||||||
|
{
|
||||||
|
ResizeIfNeed(1);
|
||||||
|
_data[_offset] = value;
|
||||||
|
_offset += 1;
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
|
public byte ReadByte()
|
||||||
|
{
|
||||||
|
var value = _data[_offset];
|
||||||
|
_offset += 1;
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
|
public int WriteBool(bool value)
|
||||||
|
{
|
||||||
|
ResizeIfNeed(1);
|
||||||
|
_data[_offset] = value ? (byte) 1 : (byte) 0;
|
||||||
|
_offset += 1;
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
|
public bool ReadBool()
|
||||||
|
{
|
||||||
|
var value = _data[_offset];
|
||||||
|
_offset += 1;
|
||||||
|
return value == 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
|
public int WriteInt(int value)
|
||||||
|
{
|
||||||
|
ResizeIfNeed(4);
|
||||||
|
var converter = new ValueConverter() {Int = value};
|
||||||
|
_data[_offset] = converter.Byte0;
|
||||||
|
_data[_offset + 1] = converter.Byte1;
|
||||||
|
_data[_offset + 2] = converter.Byte2;
|
||||||
|
_data[_offset + 3] = converter.Byte3;
|
||||||
|
_offset += 4;
|
||||||
|
return 4;
|
||||||
|
}
|
||||||
|
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
|
public int WriteInt(int value, int offset)
|
||||||
|
{
|
||||||
|
ResizeIfNeed(4);
|
||||||
|
var converter = new ValueConverter() {Int = value};
|
||||||
|
_data[offset] = converter.Byte0;
|
||||||
|
_data[offset + 1] = converter.Byte1;
|
||||||
|
_data[offset + 2] = converter.Byte2;
|
||||||
|
_data[offset + 3] = converter.Byte3;
|
||||||
|
return 4;
|
||||||
|
}
|
||||||
|
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
|
public int ReadInt()
|
||||||
|
{
|
||||||
|
var converter = new ValueConverter {Byte0 = _data[_offset], Byte1 = _data[_offset + 1], Byte2 = _data[_offset + 2], Byte3 = _data[_offset + 3]};
|
||||||
|
_offset += 4;
|
||||||
|
return converter.Int;
|
||||||
|
}
|
||||||
|
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
|
public void WriteLong(long value)
|
||||||
|
{
|
||||||
|
ResizeIfNeed(8);
|
||||||
|
WriteLong(value, _offset);
|
||||||
|
_offset += 8;
|
||||||
|
}
|
||||||
|
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
|
public int WriteLong(long value, int offset)
|
||||||
|
{
|
||||||
|
var converter = new ValueConverter() {Long = value};
|
||||||
|
_data[offset] = converter.Byte0;
|
||||||
|
_data[offset + 1] = converter.Byte1;
|
||||||
|
_data[offset + 2] = converter.Byte2;
|
||||||
|
_data[offset + 3] = converter.Byte3;
|
||||||
|
_data[offset + 4] = converter.Byte4;
|
||||||
|
_data[offset + 5] = converter.Byte5;
|
||||||
|
_data[offset + 6] = converter.Byte6;
|
||||||
|
_data[offset + 7] = converter.Byte7;
|
||||||
|
return 8;
|
||||||
|
}
|
||||||
|
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
|
public long ReadLong()
|
||||||
|
{
|
||||||
|
var converter = new ValueConverter
|
||||||
|
{
|
||||||
|
Byte0 = _data[_offset],
|
||||||
|
Byte1 = _data[_offset + 1],
|
||||||
|
Byte2 = _data[_offset + 2],
|
||||||
|
Byte3 = _data[_offset + 3],
|
||||||
|
Byte4 = _data[_offset + 4],
|
||||||
|
Byte5 = _data[_offset + 5],
|
||||||
|
Byte6 = _data[_offset + 6],
|
||||||
|
Byte7 = _data[_offset + 7],
|
||||||
|
};
|
||||||
|
_offset += 8;
|
||||||
|
return converter.Long;
|
||||||
|
}
|
||||||
|
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
|
public int WriteFloat(float value)
|
||||||
|
{
|
||||||
|
var converter = new ValueConverter() {Float = value};
|
||||||
|
WriteInt(converter.Int);
|
||||||
|
return 4;
|
||||||
|
}
|
||||||
|
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
|
public float ReadFloat()
|
||||||
|
{
|
||||||
|
var rawValue = ReadInt();
|
||||||
|
var converter = new ValueConverter() {Int = rawValue};
|
||||||
|
var value = converter.Float;
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
|
public int WriteString(string value)
|
||||||
|
{
|
||||||
|
var rawData = Encoding.UTF8.GetBytes(value).AsSpan();
|
||||||
|
ResizeIfNeed(2 + rawData.Length);
|
||||||
|
WriteUShort((ushort) rawData.Length);
|
||||||
|
var data = _data.AsSpan().Slice(_offset, rawData.Length);
|
||||||
|
rawData.CopyTo(data);
|
||||||
|
_offset += rawData.Length;
|
||||||
|
|
||||||
|
return rawData.Length + 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
|
public string ReadString()
|
||||||
|
{
|
||||||
|
var lenght = ReadUShort();
|
||||||
|
var stringRaw = _data.AsSpan().Slice(_offset, lenght);
|
||||||
|
var str = Encoding.UTF8.GetString(stringRaw);
|
||||||
|
_offset += lenght;
|
||||||
|
return str;
|
||||||
|
}
|
||||||
|
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
|
public ReadOnlySpan<byte> ReadData(int lenght)
|
||||||
|
{
|
||||||
|
var data = _data.AsSpan();
|
||||||
|
var payloadData = data.Slice(_offset, lenght);
|
||||||
|
|
||||||
|
_offset += payloadData.Length;
|
||||||
|
return payloadData;
|
||||||
|
}
|
||||||
|
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
|
public int WriteData(ref ReadOnlySpan<byte> payload)
|
||||||
|
{
|
||||||
|
ResizeIfNeed(payload.Length);
|
||||||
|
|
||||||
|
var data = _data.AsSpan();
|
||||||
|
var payloadData = data.Slice(_offset, payload.Length);
|
||||||
|
|
||||||
|
payload.CopyTo(payloadData);
|
||||||
|
_offset += payload.Length;
|
||||||
|
return payload.Length;
|
||||||
|
}
|
||||||
|
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
|
public Span<byte> GetWritableData(int lenght)
|
||||||
|
{
|
||||||
|
ResizeIfNeed(lenght);
|
||||||
|
|
||||||
|
var data = _data.AsSpan();
|
||||||
|
var payloadData = data.Slice(_offset, lenght);
|
||||||
|
|
||||||
|
_offset += lenght;
|
||||||
|
return payloadData;
|
||||||
|
}
|
||||||
|
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
|
public int WriteOperation(RagonOperation ragonOperation)
|
||||||
|
{
|
||||||
|
ResizeIfNeed(1);
|
||||||
|
|
||||||
|
_data[_offset] = (byte) ragonOperation;
|
||||||
|
_offset += 1;
|
||||||
|
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
|
public RagonOperation ReadOperation()
|
||||||
|
{
|
||||||
|
var op = (RagonOperation) _data[_offset];
|
||||||
|
_offset += 1;
|
||||||
|
return op;
|
||||||
|
}
|
||||||
|
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
|
public int WriteUShort(ushort value)
|
||||||
|
{
|
||||||
|
ResizeIfNeed(2);
|
||||||
|
|
||||||
|
_data[_offset] = (byte) (value & 0x00FF);
|
||||||
|
_data[_offset + 1] = (byte) ((value & 0xFF00) >> 8);
|
||||||
|
_offset += 2;
|
||||||
|
|
||||||
|
return 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
|
public int WriteUShort(ushort value, int offset)
|
||||||
|
{
|
||||||
|
ResizeIfNeed(2);
|
||||||
|
_data[offset] = (byte) (value & 0x00FF);
|
||||||
|
_data[offset + 1] = (byte) ((value & 0xFF00) >> 8);
|
||||||
|
return 2;
|
||||||
|
}
|
||||||
|
|
||||||
|
[MethodImpl(MethodImplOptions.AggressiveInlining)]
|
||||||
|
public ushort ReadUShort()
|
||||||
|
{
|
||||||
|
var value = (ushort) (_data[_offset] + (_data[_offset + 1] << 8));
|
||||||
|
_offset += 2;
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Clear()
|
||||||
|
{
|
||||||
|
_offset = 0;
|
||||||
|
_size = 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void ToSpan(ref Span<byte> data)
|
||||||
|
{
|
||||||
|
var span = _data.AsSpan();
|
||||||
|
var dataSpan = span.Slice(0, _offset);
|
||||||
|
dataSpan.CopyTo(data);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void FromSpan(ref ReadOnlySpan<byte> data)
|
||||||
|
{
|
||||||
|
Clear();
|
||||||
|
ResizeIfNeed(data.Length);
|
||||||
|
var dataSpan = _data.AsSpan();
|
||||||
|
data.CopyTo(dataSpan);
|
||||||
|
_size = data.Length;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void FromArray(byte[] data)
|
||||||
|
{
|
||||||
|
Clear();
|
||||||
|
ResizeIfNeed(data.Length);
|
||||||
|
Buffer.BlockCopy(data, 0, _data, 0, data.Length);
|
||||||
|
_size = data.Length;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
public byte[] ToArray()
|
||||||
|
{
|
||||||
|
var bytes = new byte[_offset];
|
||||||
|
Buffer.BlockCopy(_data, 0, bytes, 0, _offset);
|
||||||
|
return bytes;
|
||||||
|
}
|
||||||
|
|
||||||
|
private void ResizeIfNeed(int lenght)
|
||||||
|
{
|
||||||
|
if (_offset + lenght < _data.Length)
|
||||||
|
return;
|
||||||
|
|
||||||
|
var newData = new byte[_data.Length * 4 + lenght];
|
||||||
|
Buffer.BlockCopy(_data, 0, newData, 0, _data.Length);
|
||||||
|
_data = newData;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,11 @@
|
|||||||
|
namespace Ragon.Common
|
||||||
|
{
|
||||||
|
public enum RagonTarget: byte
|
||||||
|
{
|
||||||
|
Owner,
|
||||||
|
ExceptOwner,
|
||||||
|
ExceptInvoker,
|
||||||
|
All,
|
||||||
|
Player,
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,28 @@
|
|||||||
|
namespace Ragon.Common
|
||||||
|
{
|
||||||
|
public static class RagonVersion
|
||||||
|
{
|
||||||
|
public static uint Parse(string version)
|
||||||
|
{
|
||||||
|
var strings = version.Split(".");
|
||||||
|
if (strings.Length < 3)
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
var parts = new uint[] {0, 0, 0};
|
||||||
|
for (int i = 0; i < parts.Length; i++)
|
||||||
|
{
|
||||||
|
if (!uint.TryParse(strings[i], out var v))
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
parts[i] = v;
|
||||||
|
}
|
||||||
|
|
||||||
|
return (parts[0] << 16) | (parts[1] << 8) | parts[2];
|
||||||
|
}
|
||||||
|
|
||||||
|
public static string Parse(uint version)
|
||||||
|
{
|
||||||
|
return (version >> 16 & 0xFF) + "." + (version >> 8 & 0xFF) + "." + (version & 0xFF);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
Executable
+28
@@ -0,0 +1,28 @@
|
|||||||
|
using System;
|
||||||
|
using System.Runtime.InteropServices;
|
||||||
|
using NLog;
|
||||||
|
using Ragon.Core;
|
||||||
|
|
||||||
|
namespace Ragon.Relay
|
||||||
|
{
|
||||||
|
class Program
|
||||||
|
{
|
||||||
|
static void Main(string[] args)
|
||||||
|
{
|
||||||
|
var logger = LogManager.GetLogger("Ragon.Relay");
|
||||||
|
|
||||||
|
logger.Info("Relay Application");
|
||||||
|
|
||||||
|
var configuration = Configuration.Load("relay.config.json");
|
||||||
|
var relay = new Application(configuration);
|
||||||
|
|
||||||
|
logger.Info("Started");
|
||||||
|
relay.Start();
|
||||||
|
|
||||||
|
Console.ReadKey();
|
||||||
|
|
||||||
|
relay.Stop();
|
||||||
|
logger.Info("Stopped");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -14,17 +14,13 @@
|
|||||||
<None Update="NLog.config">
|
<None Update="NLog.config">
|
||||||
<CopyToOutputDirectory>Always</CopyToOutputDirectory>
|
<CopyToOutputDirectory>Always</CopyToOutputDirectory>
|
||||||
</None>
|
</None>
|
||||||
<None Update="config.json">
|
<None Update="relay.config.json">
|
||||||
<CopyToOutputDirectory>Always</CopyToOutputDirectory>
|
<CopyToOutputDirectory>Always</CopyToOutputDirectory>
|
||||||
</None>
|
</None>
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
|
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<ProjectReference Include="..\Ragon\Ragon.csproj" />
|
<ProjectReference Include="..\Ragon.Core\Ragon.Core.csproj" />
|
||||||
</ItemGroup>
|
|
||||||
|
|
||||||
<ItemGroup>
|
|
||||||
<Folder Include="Source\Managers" />
|
|
||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
|
|
||||||
</Project>
|
</Project>
|
||||||
@@ -0,0 +1,10 @@
|
|||||||
|
{
|
||||||
|
"serverKey": "defaultkey",
|
||||||
|
"serverType": "websocket",
|
||||||
|
"serverTickRate": 20,
|
||||||
|
"gameProtocol": "1.0.0",
|
||||||
|
"port": 5000,
|
||||||
|
"limitConnections": 4095,
|
||||||
|
"limitPlayersPerRoom": 20,
|
||||||
|
"limitRooms": 200
|
||||||
|
}
|
||||||
@@ -0,0 +1,17 @@
|
|||||||
|
using ENet;
|
||||||
|
|
||||||
|
namespace Ragon.Server.ENet;
|
||||||
|
|
||||||
|
public sealed class ENetConnection: INetworkConnection
|
||||||
|
{
|
||||||
|
public ushort Id { get; }
|
||||||
|
public INetworkChannel Reliable { get; private set; }
|
||||||
|
public INetworkChannel Unreliable { get; private set; }
|
||||||
|
|
||||||
|
public ENetConnection(Peer peer)
|
||||||
|
{
|
||||||
|
Id = (ushort) peer.ID;
|
||||||
|
Reliable = new ENetReliableChannel(peer, 0);
|
||||||
|
Unreliable = new ENetUnreliableChannel(peer, 1);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,23 @@
|
|||||||
|
using ENet;
|
||||||
|
|
||||||
|
namespace Ragon.Server.ENet;
|
||||||
|
|
||||||
|
public sealed class ENetReliableChannel: INetworkChannel
|
||||||
|
{
|
||||||
|
private Peer _peer;
|
||||||
|
private byte _channelId;
|
||||||
|
|
||||||
|
public ENetReliableChannel(Peer peer, int channelId)
|
||||||
|
{
|
||||||
|
_peer = peer;
|
||||||
|
_channelId = (byte) channelId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Send(byte[] data)
|
||||||
|
{
|
||||||
|
var newPacket = new Packet();
|
||||||
|
newPacket.Create(data, data.Length, PacketFlags.Reliable);
|
||||||
|
|
||||||
|
_peer.Send(_channelId, ref newPacket);
|
||||||
|
}
|
||||||
|
}
|
||||||
Executable
+116
@@ -0,0 +1,116 @@
|
|||||||
|
using System.Diagnostics;
|
||||||
|
using ENet;
|
||||||
|
using NLog;
|
||||||
|
using Ragon.Common;
|
||||||
|
|
||||||
|
|
||||||
|
namespace Ragon.Server.ENet
|
||||||
|
{
|
||||||
|
public sealed class ENetServer: INetworkServer
|
||||||
|
{
|
||||||
|
private readonly Host _host;
|
||||||
|
private readonly ILogger _logger = LogManager.GetCurrentClassLogger();
|
||||||
|
|
||||||
|
private ENetConnection[] _connections;
|
||||||
|
private uint _protocol;
|
||||||
|
private INetworkListener _listener;
|
||||||
|
private Event _event;
|
||||||
|
|
||||||
|
public ENetServer()
|
||||||
|
{
|
||||||
|
_host = new Host();
|
||||||
|
_connections = Array.Empty<ENetConnection>();
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Start(INetworkListener listener, NetworkConfiguration configuration)
|
||||||
|
{
|
||||||
|
Library.Initialize();
|
||||||
|
|
||||||
|
_connections = new ENetConnection[configuration.LimitConnections];
|
||||||
|
|
||||||
|
_listener = listener;
|
||||||
|
_protocol = configuration.Protocol;
|
||||||
|
|
||||||
|
var address = new Address { Port = (ushort) configuration.Port };
|
||||||
|
_host.Create(address, _connections.Length, 2, 0, 0, 1024 * 1024);
|
||||||
|
|
||||||
|
var protocolDecoded = RagonVersion.Parse(_protocol);
|
||||||
|
_logger.Info($"Network listening on {configuration.Port}");
|
||||||
|
_logger.Info($"Protocol: {protocolDecoded}");
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Poll()
|
||||||
|
{
|
||||||
|
bool polled = false;
|
||||||
|
while (!polled)
|
||||||
|
{
|
||||||
|
if (_host.CheckEvents(out _event) <= 0)
|
||||||
|
{
|
||||||
|
if (_host.Service(0, out _event) <= 0)
|
||||||
|
break;
|
||||||
|
|
||||||
|
polled = true;
|
||||||
|
}
|
||||||
|
|
||||||
|
switch (_event.Type)
|
||||||
|
{
|
||||||
|
case EventType.None:
|
||||||
|
{
|
||||||
|
_logger.Trace("None event");
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case EventType.Connect:
|
||||||
|
{
|
||||||
|
if (!IsValidProtocol(_event.Data))
|
||||||
|
{
|
||||||
|
_logger.Warn($"Mismatched protocol Server: {RagonVersion.Parse(_protocol)} Client: {RagonVersion.Parse(_event.Data)}, close connection");
|
||||||
|
_event.Peer.DisconnectNow(0);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
var connection = new ENetConnection(_event.Peer);
|
||||||
|
_connections[_event.Peer.ID] = connection;
|
||||||
|
_listener.OnConnected(connection);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case EventType.Disconnect:
|
||||||
|
{
|
||||||
|
var connection = _connections[_event.Peer.ID];
|
||||||
|
_listener.OnDisconnected(connection);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case EventType.Timeout:
|
||||||
|
{
|
||||||
|
var connection = _connections[_event.Peer.ID];
|
||||||
|
_listener.OnTimeout(connection);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case EventType.Receive:
|
||||||
|
{
|
||||||
|
var peerId = (ushort) _event.Peer.ID;
|
||||||
|
var connection = _connections[peerId];
|
||||||
|
var dataRaw = new byte[_event.Packet.Length];
|
||||||
|
|
||||||
|
_event.Packet.CopyTo(dataRaw);
|
||||||
|
_event.Packet.Dispose();
|
||||||
|
|
||||||
|
_listener.OnData(connection, dataRaw);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Stop()
|
||||||
|
{
|
||||||
|
_host?.Dispose();
|
||||||
|
|
||||||
|
Library.Deinitialize();
|
||||||
|
}
|
||||||
|
|
||||||
|
private bool IsValidProtocol(uint protocol)
|
||||||
|
{
|
||||||
|
return protocol == _protocol;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,23 @@
|
|||||||
|
using ENet;
|
||||||
|
|
||||||
|
namespace Ragon.Server.ENet;
|
||||||
|
|
||||||
|
public sealed class ENetUnreliableChannel: INetworkChannel
|
||||||
|
{
|
||||||
|
private Peer _peer;
|
||||||
|
private byte _channelId;
|
||||||
|
|
||||||
|
public ENetUnreliableChannel(Peer peer, int channelId)
|
||||||
|
{
|
||||||
|
_peer = peer;
|
||||||
|
_channelId = (byte) channelId;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Send(byte[] data)
|
||||||
|
{
|
||||||
|
var newPacket = new Packet();
|
||||||
|
newPacket.Create(data, data.Length, PacketFlags.None);
|
||||||
|
|
||||||
|
_peer.Send(_channelId, ref newPacket);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,19 @@
|
|||||||
|
<Project Sdk="Microsoft.NET.Sdk">
|
||||||
|
|
||||||
|
<PropertyGroup>
|
||||||
|
<TargetFramework>net6.0</TargetFramework>
|
||||||
|
<ImplicitUsings>enable</ImplicitUsings>
|
||||||
|
<Nullable>enable</Nullable>
|
||||||
|
<RootNamespace>Ragon.ENet</RootNamespace>
|
||||||
|
</PropertyGroup>
|
||||||
|
|
||||||
|
<ItemGroup>
|
||||||
|
<PackageReference Include="ENet-CSharp" Version="2.4.8" />
|
||||||
|
<PackageReference Include="NLog" Version="5.0.5" />
|
||||||
|
</ItemGroup>
|
||||||
|
|
||||||
|
<ItemGroup>
|
||||||
|
<ProjectReference Include="..\Ragon.Server\Ragon.Server.csproj" />
|
||||||
|
</ItemGroup>
|
||||||
|
|
||||||
|
</Project>
|
||||||
@@ -0,0 +1,18 @@
|
|||||||
|
<Project Sdk="Microsoft.NET.Sdk">
|
||||||
|
|
||||||
|
<PropertyGroup>
|
||||||
|
<TargetFramework>net6.0</TargetFramework>
|
||||||
|
<ImplicitUsings>enable</ImplicitUsings>
|
||||||
|
<Nullable>enable</Nullable>
|
||||||
|
<RootNamespace>Ragon.WebSockets</RootNamespace>
|
||||||
|
</PropertyGroup>
|
||||||
|
|
||||||
|
<ItemGroup>
|
||||||
|
<ProjectReference Include="..\Ragon.Server\Ragon.Server.csproj" />
|
||||||
|
</ItemGroup>
|
||||||
|
|
||||||
|
<ItemGroup>
|
||||||
|
<PackageReference Include="NLog" Version="5.0.5" />
|
||||||
|
</ItemGroup>
|
||||||
|
|
||||||
|
</Project>
|
||||||
@@ -0,0 +1,44 @@
|
|||||||
|
using System.Net.WebSockets;
|
||||||
|
using NLog;
|
||||||
|
|
||||||
|
namespace Ragon.Server.NativeWebSockets;
|
||||||
|
|
||||||
|
public sealed class WebSocketConnection : INetworkConnection
|
||||||
|
{
|
||||||
|
private Logger _logger = LogManager.GetCurrentClassLogger();
|
||||||
|
public ushort Id { get; }
|
||||||
|
public INetworkChannel Reliable { get; private set; }
|
||||||
|
public INetworkChannel Unreliable { get; private set; }
|
||||||
|
|
||||||
|
public WebSocket Socket { get; private set; }
|
||||||
|
private WebSocketReliableChannel[] _channels;
|
||||||
|
|
||||||
|
public WebSocketConnection(WebSocket webSocket, ushort peerId)
|
||||||
|
{
|
||||||
|
Id = peerId;
|
||||||
|
Socket = webSocket;
|
||||||
|
|
||||||
|
var reliableChannel = new WebSocketReliableChannel(webSocket);
|
||||||
|
var unreliableChannel = new WebSocketReliableChannel(webSocket);
|
||||||
|
|
||||||
|
_channels = new[] { reliableChannel, unreliableChannel };
|
||||||
|
|
||||||
|
Reliable = reliableChannel;
|
||||||
|
Unreliable = unreliableChannel;
|
||||||
|
}
|
||||||
|
|
||||||
|
public async Task Flush()
|
||||||
|
{
|
||||||
|
foreach (var channel in _channels)
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
await channel.Flush();
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
_logger.Error(ex);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,27 @@
|
|||||||
|
using System.Net.WebSockets;
|
||||||
|
using Ragon.Server;
|
||||||
|
|
||||||
|
namespace Ragon.Server.NativeWebSockets;
|
||||||
|
|
||||||
|
public class WebSocketReliableChannel : INetworkChannel
|
||||||
|
{
|
||||||
|
private Queue<byte[]> _queue;
|
||||||
|
private WebSocket _socket;
|
||||||
|
|
||||||
|
public WebSocketReliableChannel(WebSocket webSocket)
|
||||||
|
{
|
||||||
|
_socket = webSocket;
|
||||||
|
_queue = new Queue<byte[]>(512);
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Send(byte[] data)
|
||||||
|
{
|
||||||
|
_queue.Enqueue(data);
|
||||||
|
}
|
||||||
|
|
||||||
|
public async Task Flush()
|
||||||
|
{
|
||||||
|
while (_queue.TryDequeue(out var sendData) && _socket.State == WebSocketState.Open)
|
||||||
|
await _socket.SendAsync(sendData, WebSocketMessageType.Binary, WebSocketMessageFlags.EndOfMessage, CancellationToken.None);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,121 @@
|
|||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Diagnostics;
|
||||||
|
using System.Net;
|
||||||
|
using System.Net.WebSockets;
|
||||||
|
using NLog;
|
||||||
|
using Ragon.Common;
|
||||||
|
using Ragon.Core.Server;
|
||||||
|
using Ragon.Server;
|
||||||
|
using Ragon.Server.NativeWebSockets;
|
||||||
|
|
||||||
|
namespace Ragon.Core;
|
||||||
|
|
||||||
|
public class NativeWebSocketServer : INetworkServer
|
||||||
|
{
|
||||||
|
private ILogger _logger = LogManager.GetCurrentClassLogger();
|
||||||
|
private INetworkListener _networkListener;
|
||||||
|
private Stack<ushort> _sequencer;
|
||||||
|
private Executor _executor;
|
||||||
|
private HttpListener _httpListener;
|
||||||
|
private WebSocketConnection[] _connections;
|
||||||
|
private ushort _lastPeerId;
|
||||||
|
private CancellationTokenSource _cancellationTokenSource;
|
||||||
|
|
||||||
|
public NativeWebSocketServer(Executor executor)
|
||||||
|
{
|
||||||
|
_sequencer = new Stack<ushort>();
|
||||||
|
_connections = Array.Empty<WebSocketConnection>();
|
||||||
|
_executor = executor;
|
||||||
|
}
|
||||||
|
|
||||||
|
public async Task StartAccept(CancellationToken cancellationToken)
|
||||||
|
{
|
||||||
|
while (!cancellationToken.IsCancellationRequested)
|
||||||
|
{
|
||||||
|
var context = await _httpListener.GetContextAsync();
|
||||||
|
if (!context.Request.IsWebSocketRequest) continue;
|
||||||
|
|
||||||
|
var webSocketContext = await context.AcceptWebSocketAsync(null);
|
||||||
|
var webSocket = webSocketContext.WebSocket;
|
||||||
|
|
||||||
|
var peerId = _sequencer.Pop();
|
||||||
|
var connection = new WebSocketConnection(webSocket, peerId);
|
||||||
|
|
||||||
|
_lastPeerId = peerId;
|
||||||
|
_connections[peerId] = connection;
|
||||||
|
_networkListener.OnConnected(connection);
|
||||||
|
_executor.Run(StartListen(connection, cancellationToken));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async Task StartListen(WebSocketConnection connection, CancellationToken cancellationToken)
|
||||||
|
{
|
||||||
|
var webSocket = connection.Socket;
|
||||||
|
var bytes = new byte[2048];
|
||||||
|
var buffer = new Memory<byte>(bytes);
|
||||||
|
while (
|
||||||
|
webSocket.State == WebSocketState.Open ||
|
||||||
|
!cancellationToken.IsCancellationRequested)
|
||||||
|
{
|
||||||
|
try
|
||||||
|
{
|
||||||
|
var result = await webSocket.ReceiveAsync(buffer, cancellationToken);
|
||||||
|
var dataRaw = buffer.Slice(0, result.Count);
|
||||||
|
|
||||||
|
_networkListener.OnData(connection, dataRaw.ToArray());
|
||||||
|
}
|
||||||
|
catch (Exception ex)
|
||||||
|
{
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
_sequencer.Push(connection.Id);
|
||||||
|
_networkListener.OnDisconnected(connection);
|
||||||
|
}
|
||||||
|
|
||||||
|
public async void Poll()
|
||||||
|
{
|
||||||
|
foreach (var conn in _connections)
|
||||||
|
{
|
||||||
|
if (conn != null)
|
||||||
|
{
|
||||||
|
await conn.Flush();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Start(
|
||||||
|
INetworkListener listener,
|
||||||
|
NetworkConfiguration configuration
|
||||||
|
)
|
||||||
|
{
|
||||||
|
_networkListener = listener;
|
||||||
|
_cancellationTokenSource = new CancellationTokenSource();
|
||||||
|
|
||||||
|
var limit = (ushort) configuration.LimitConnections;
|
||||||
|
for (ushort i = limit; i != 0; i--)
|
||||||
|
_sequencer.Push(i);
|
||||||
|
|
||||||
|
_sequencer.Push(0);
|
||||||
|
|
||||||
|
_connections = new WebSocketConnection[configuration.LimitConnections];
|
||||||
|
|
||||||
|
_httpListener = new HttpListener();
|
||||||
|
_httpListener.Prefixes.Add($"http://127.0.0.1:{configuration.Port}/");
|
||||||
|
_httpListener.Start();
|
||||||
|
|
||||||
|
_executor.Run(StartAccept(_cancellationTokenSource.Token));
|
||||||
|
|
||||||
|
var protocolDecoded = RagonVersion.Parse(configuration.Protocol);
|
||||||
|
_logger.Info($"Network listening on http://*:{configuration.Port}/");
|
||||||
|
_logger.Info($"Protocol: {protocolDecoded}");
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Stop()
|
||||||
|
{
|
||||||
|
_cancellationTokenSource.Cancel();
|
||||||
|
_httpListener.Stop();
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,6 @@
|
|||||||
|
namespace Ragon.Server;
|
||||||
|
|
||||||
|
public interface INetworkChannel
|
||||||
|
{
|
||||||
|
void Send(byte[] data);
|
||||||
|
}
|
||||||
@@ -0,0 +1,8 @@
|
|||||||
|
namespace Ragon.Server;
|
||||||
|
|
||||||
|
public interface INetworkConnection
|
||||||
|
{
|
||||||
|
public ushort Id { get; }
|
||||||
|
public INetworkChannel Reliable { get; }
|
||||||
|
public INetworkChannel Unreliable { get; }
|
||||||
|
}
|
||||||
@@ -0,0 +1,9 @@
|
|||||||
|
namespace Ragon.Server;
|
||||||
|
|
||||||
|
public interface INetworkListener
|
||||||
|
{
|
||||||
|
void OnConnected(INetworkConnection connection);
|
||||||
|
void OnDisconnected(INetworkConnection connection);
|
||||||
|
void OnTimeout(INetworkConnection connection);
|
||||||
|
void OnData(INetworkConnection connection, byte[] data);
|
||||||
|
}
|
||||||
@@ -0,0 +1,8 @@
|
|||||||
|
namespace Ragon.Server;
|
||||||
|
|
||||||
|
public interface INetworkServer
|
||||||
|
{
|
||||||
|
public void Stop();
|
||||||
|
public void Poll();
|
||||||
|
public void Start(INetworkListener listener, NetworkConfiguration configuration);
|
||||||
|
}
|
||||||
@@ -0,0 +1,55 @@
|
|||||||
|
using System.Threading.Channels;
|
||||||
|
|
||||||
|
namespace Ragon.Core.Server;
|
||||||
|
|
||||||
|
public class Executor: TaskScheduler
|
||||||
|
{
|
||||||
|
private ChannelReader<Task> _reader;
|
||||||
|
private ChannelWriter<Task> _writer;
|
||||||
|
private Queue<Task> _pendingTasks;
|
||||||
|
private TaskFactory _taskFactory;
|
||||||
|
|
||||||
|
public void Run(Task task)
|
||||||
|
{
|
||||||
|
_taskFactory.StartNew(() => task);
|
||||||
|
}
|
||||||
|
|
||||||
|
public Executor()
|
||||||
|
{
|
||||||
|
var channel = Channel.CreateUnbounded<Task>();
|
||||||
|
_reader = channel.Reader;
|
||||||
|
_writer = channel.Writer;
|
||||||
|
|
||||||
|
_taskFactory = new TaskFactory(this);
|
||||||
|
_pendingTasks = new Queue<Task>();
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override IEnumerable<Task>? GetScheduledTasks()
|
||||||
|
{
|
||||||
|
throw new NotSupportedException();
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override void QueueTask(Task task)
|
||||||
|
{
|
||||||
|
_writer.TryWrite(task);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected override bool TryExecuteTaskInline(Task task, bool taskWasPreviouslyQueued)
|
||||||
|
{
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
public void Execute()
|
||||||
|
{
|
||||||
|
while (_reader.TryRead(out var task))
|
||||||
|
{
|
||||||
|
TryExecuteTask(task);
|
||||||
|
|
||||||
|
if (task.Status == TaskStatus.Running)
|
||||||
|
_pendingTasks.Enqueue(task);
|
||||||
|
}
|
||||||
|
|
||||||
|
while (_pendingTasks.TryDequeue(out var task))
|
||||||
|
_writer.TryWrite(task);
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,9 @@
|
|||||||
|
namespace Ragon.Server;
|
||||||
|
|
||||||
|
public struct NetworkConfiguration
|
||||||
|
{
|
||||||
|
public int LimitConnections { get; set; }
|
||||||
|
public int Port { get; set; }
|
||||||
|
public uint Protocol { get; set; }
|
||||||
|
public string Address { get; set; }
|
||||||
|
}
|
||||||
@@ -0,0 +1,13 @@
|
|||||||
|
<Project Sdk="Microsoft.NET.Sdk">
|
||||||
|
|
||||||
|
<PropertyGroup>
|
||||||
|
<TargetFramework>net6.0</TargetFramework>
|
||||||
|
<ImplicitUsings>enable</ImplicitUsings>
|
||||||
|
<Nullable>enable</Nullable>
|
||||||
|
</PropertyGroup>
|
||||||
|
|
||||||
|
<ItemGroup>
|
||||||
|
<ProjectReference Include="..\Ragon.Protocol\Ragon.Protocol.csproj" />
|
||||||
|
</ItemGroup>
|
||||||
|
|
||||||
|
</Project>
|
||||||
@@ -1,10 +1,16 @@
|
|||||||
|
|
||||||
Microsoft Visual Studio Solution File, Format Version 12.00
|
Microsoft Visual Studio Solution File, Format Version 12.00
|
||||||
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Ragon", "Ragon\Ragon.csproj", "{BABA1AF0-CF91-43F2-9577-53800068ACCF}"
|
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Ragon.Relay", "Ragon.Relay\Ragon.Relay.csproj", "{C2B87ADE-A122-4366-8EB7-24DAE11EBAF0}"
|
||||||
EndProject
|
EndProject
|
||||||
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "SimpleServer", "SimpleServer\SimpleServer.csproj", "{C2B87ADE-A122-4366-8EB7-24DAE11EBAF0}"
|
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Ragon.Protocol", "Ragon.Protocol\Ragon.Protocol.csproj", "{F478B2A2-36F4-43B9-9BB7-382A57C449B2}"
|
||||||
EndProject
|
EndProject
|
||||||
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Ragon.Common", "Ragon.Common\Ragon.Common.csproj", "{F478B2A2-36F4-43B9-9BB7-382A57C449B2}"
|
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Ragon.Server", "Ragon.Server\Ragon.Server.csproj", "{45D3B686-8960-4656-91B2-6F8BFCCAA225}"
|
||||||
|
EndProject
|
||||||
|
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Ragon.Core", "Ragon.Core\Ragon.Core.csproj", "{F4AA86B9-2486-4B53-BA77-43D958A2FDC3}"
|
||||||
|
EndProject
|
||||||
|
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Ragon.Server.NativeWebSockets", "Ragon.Server.NativeWebSockets\Ragon.Server.NativeWebSockets.csproj", "{81050343-A9B8-487B-86C8-7A5B7DD9C39B}"
|
||||||
|
EndProject
|
||||||
|
Project("{FAE04EC0-301F-11D3-BF4B-00C04F79EFBC}") = "Ragon.Server.ENet", "Ragon.Server.ENet\Ragon.Server.ENet.csproj", "{DD79AC4F-9E5C-4938-850E-805D537E68D0}"
|
||||||
EndProject
|
EndProject
|
||||||
Global
|
Global
|
||||||
GlobalSection(SolutionConfigurationPlatforms) = preSolution
|
GlobalSection(SolutionConfigurationPlatforms) = preSolution
|
||||||
@@ -12,10 +18,6 @@ Global
|
|||||||
Release|Any CPU = Release|Any CPU
|
Release|Any CPU = Release|Any CPU
|
||||||
EndGlobalSection
|
EndGlobalSection
|
||||||
GlobalSection(ProjectConfigurationPlatforms) = postSolution
|
GlobalSection(ProjectConfigurationPlatforms) = postSolution
|
||||||
{BABA1AF0-CF91-43F2-9577-53800068ACCF}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
|
||||||
{BABA1AF0-CF91-43F2-9577-53800068ACCF}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
|
||||||
{BABA1AF0-CF91-43F2-9577-53800068ACCF}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
|
||||||
{BABA1AF0-CF91-43F2-9577-53800068ACCF}.Release|Any CPU.Build.0 = Release|Any CPU
|
|
||||||
{C2B87ADE-A122-4366-8EB7-24DAE11EBAF0}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
{C2B87ADE-A122-4366-8EB7-24DAE11EBAF0}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||||
{C2B87ADE-A122-4366-8EB7-24DAE11EBAF0}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
{C2B87ADE-A122-4366-8EB7-24DAE11EBAF0}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||||
{C2B87ADE-A122-4366-8EB7-24DAE11EBAF0}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
{C2B87ADE-A122-4366-8EB7-24DAE11EBAF0}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||||
@@ -24,5 +26,21 @@ Global
|
|||||||
{F478B2A2-36F4-43B9-9BB7-382A57C449B2}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
{F478B2A2-36F4-43B9-9BB7-382A57C449B2}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||||
{F478B2A2-36F4-43B9-9BB7-382A57C449B2}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
{F478B2A2-36F4-43B9-9BB7-382A57C449B2}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||||
{F478B2A2-36F4-43B9-9BB7-382A57C449B2}.Release|Any CPU.Build.0 = Release|Any CPU
|
{F478B2A2-36F4-43B9-9BB7-382A57C449B2}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||||
|
{45D3B686-8960-4656-91B2-6F8BFCCAA225}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||||
|
{45D3B686-8960-4656-91B2-6F8BFCCAA225}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||||
|
{45D3B686-8960-4656-91B2-6F8BFCCAA225}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||||
|
{45D3B686-8960-4656-91B2-6F8BFCCAA225}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||||
|
{F4AA86B9-2486-4B53-BA77-43D958A2FDC3}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||||
|
{F4AA86B9-2486-4B53-BA77-43D958A2FDC3}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||||
|
{F4AA86B9-2486-4B53-BA77-43D958A2FDC3}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||||
|
{F4AA86B9-2486-4B53-BA77-43D958A2FDC3}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||||
|
{81050343-A9B8-487B-86C8-7A5B7DD9C39B}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||||
|
{81050343-A9B8-487B-86C8-7A5B7DD9C39B}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||||
|
{81050343-A9B8-487B-86C8-7A5B7DD9C39B}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||||
|
{81050343-A9B8-487B-86C8-7A5B7DD9C39B}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||||
|
{DD79AC4F-9E5C-4938-850E-805D537E68D0}.Debug|Any CPU.ActiveCfg = Debug|Any CPU
|
||||||
|
{DD79AC4F-9E5C-4938-850E-805D537E68D0}.Debug|Any CPU.Build.0 = Debug|Any CPU
|
||||||
|
{DD79AC4F-9E5C-4938-850E-805D537E68D0}.Release|Any CPU.ActiveCfg = Release|Any CPU
|
||||||
|
{DD79AC4F-9E5C-4938-850E-805D537E68D0}.Release|Any CPU.Build.0 = Release|Any CPU
|
||||||
EndGlobalSection
|
EndGlobalSection
|
||||||
EndGlobal
|
EndGlobal
|
||||||
|
|||||||
@@ -1,29 +0,0 @@
|
|||||||
<Project Sdk="Microsoft.NET.Sdk">
|
|
||||||
|
|
||||||
<PropertyGroup>
|
|
||||||
<LangVersion>10</LangVersion>
|
|
||||||
<Nullable>enable</Nullable>
|
|
||||||
<TargetFrameworks>net6.0;netstandard2.1</TargetFrameworks>
|
|
||||||
</PropertyGroup>
|
|
||||||
|
|
||||||
<PropertyGroup Condition=" '$(Configuration)' == 'Debug' ">
|
|
||||||
<AllowUnsafeBlocks>true</AllowUnsafeBlocks>
|
|
||||||
<DefineConstants>TRACE;NETSTACK_SPAN</DefineConstants>
|
|
||||||
</PropertyGroup>
|
|
||||||
|
|
||||||
<PropertyGroup Condition=" '$(Configuration)' == 'Release' ">
|
|
||||||
<AllowUnsafeBlocks>true</AllowUnsafeBlocks>
|
|
||||||
<DefineConstants>TRACE;NETSTACK_SPAN</DefineConstants>
|
|
||||||
</PropertyGroup>
|
|
||||||
|
|
||||||
<ItemGroup>
|
|
||||||
<PackageReference Include="ENet-CSharp" Version="2.4.8" />
|
|
||||||
<PackageReference Include="Newtonsoft.Json" Version="13.0.1" />
|
|
||||||
<PackageReference Include="NLog" Version="5.0.0-rc2" />
|
|
||||||
</ItemGroup>
|
|
||||||
|
|
||||||
<ItemGroup>
|
|
||||||
<ProjectReference Include="..\Ragon.Common\Ragon.Common.csproj" />
|
|
||||||
</ItemGroup>
|
|
||||||
|
|
||||||
</Project>
|
|
||||||
@@ -1,90 +0,0 @@
|
|||||||
using System;
|
|
||||||
using System.Collections.Generic;
|
|
||||||
using System.Linq;
|
|
||||||
using System.Threading;
|
|
||||||
using NLog;
|
|
||||||
|
|
||||||
namespace Ragon.Core
|
|
||||||
{
|
|
||||||
public class Application : IDisposable
|
|
||||||
{
|
|
||||||
private readonly Logger _logger = LogManager.GetCurrentClassLogger();
|
|
||||||
private readonly List<RoomThread> _roomThreads = new();
|
|
||||||
private readonly Dictionary<uint, RoomThread> _socketByRoomThreads = new();
|
|
||||||
private readonly Dictionary<RoomThread, int> _roomThreadCounter = new();
|
|
||||||
private readonly Configuration _configuration;
|
|
||||||
private readonly ENetServer _socketServer;
|
|
||||||
private int _roomThreadBalancer = 0;
|
|
||||||
|
|
||||||
public Application(PluginFactory factory, Configuration configuration, int threadsCount)
|
|
||||||
{
|
|
||||||
_socketServer = new ENetServer();
|
|
||||||
_configuration = configuration;
|
|
||||||
|
|
||||||
for (var i = 0; i < threadsCount; i++)
|
|
||||||
{
|
|
||||||
var roomThread = new RoomThread(factory, configuration);
|
|
||||||
_roomThreadCounter.Add(roomThread, 0);
|
|
||||||
_roomThreads.Add(roomThread);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public void Start()
|
|
||||||
{
|
|
||||||
_socketServer.Start(_configuration.Server.Port);
|
|
||||||
|
|
||||||
foreach (var roomThread in _roomThreads)
|
|
||||||
roomThread.Start();
|
|
||||||
|
|
||||||
while (true)
|
|
||||||
{
|
|
||||||
foreach (var roomThread in _roomThreads)
|
|
||||||
while (roomThread.ReadOutEvent(out var evnt))
|
|
||||||
_socketServer.WriteEvent(evnt);
|
|
||||||
|
|
||||||
while (_socketServer.ReadEvent(out var evnt))
|
|
||||||
{
|
|
||||||
if (evnt.Type == EventType.CONNECTED)
|
|
||||||
{
|
|
||||||
if (_roomThreadBalancer >= _roomThreads.Count)
|
|
||||||
_roomThreadBalancer = 0;
|
|
||||||
|
|
||||||
var roomThread = _roomThreads[_roomThreadBalancer];
|
|
||||||
_roomThreadCounter[roomThread] += 1;
|
|
||||||
_socketByRoomThreads.Add(evnt.PeerId, roomThread);
|
|
||||||
|
|
||||||
// TODO: Todo room manager matchmaking across all room threads
|
|
||||||
// TEMP_FIX: Remove this magical number
|
|
||||||
if (_roomThreadCounter[roomThread] > 20)
|
|
||||||
_roomThreadBalancer++;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (_socketByRoomThreads.TryGetValue(evnt.PeerId, out var existsRoomThread))
|
|
||||||
existsRoomThread.WriteInEvent(evnt);
|
|
||||||
|
|
||||||
if (evnt.Type == EventType.DISCONNECTED)
|
|
||||||
{
|
|
||||||
_socketByRoomThreads.Remove(evnt.PeerId, out var roomThread);
|
|
||||||
_roomThreadCounter[roomThread] =- 1;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (evnt.Type == EventType.TIMEOUT)
|
|
||||||
{
|
|
||||||
_socketByRoomThreads.Remove(evnt.PeerId, out var roomThread);
|
|
||||||
_roomThreadCounter[roomThread] =- 1;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
Thread.Sleep(1);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public void Dispose()
|
|
||||||
{
|
|
||||||
foreach (var roomThread in _roomThreads)
|
|
||||||
roomThread.Dispose();
|
|
||||||
|
|
||||||
_roomThreads.Clear();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,11 +0,0 @@
|
|||||||
using System;
|
|
||||||
|
|
||||||
namespace Ragon.Core;
|
|
||||||
|
|
||||||
public class AuthorizationManager
|
|
||||||
{
|
|
||||||
public virtual bool OnAuthorize(uint peerId, ref ReadOnlySpan<byte> payload)
|
|
||||||
{
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,17 +0,0 @@
|
|||||||
using NLog;
|
|
||||||
|
|
||||||
namespace Ragon.Core
|
|
||||||
{
|
|
||||||
public class Bootstrap
|
|
||||||
{
|
|
||||||
private ILogger _logger = LogManager.GetCurrentClassLogger();
|
|
||||||
|
|
||||||
public void Configure(PluginFactory factory)
|
|
||||||
{
|
|
||||||
_logger.Info("Configure application...");
|
|
||||||
var configuration = ConfigurationLoader.Load("config.json");
|
|
||||||
var app = new Application(factory, configuration, 2);
|
|
||||||
app.Start();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,18 +0,0 @@
|
|||||||
using System;
|
|
||||||
|
|
||||||
namespace Ragon.Core
|
|
||||||
{
|
|
||||||
[Serializable]
|
|
||||||
public struct Server
|
|
||||||
{
|
|
||||||
public ushort Port;
|
|
||||||
public ushort TickRate;
|
|
||||||
}
|
|
||||||
|
|
||||||
[Serializable]
|
|
||||||
public struct Configuration
|
|
||||||
{
|
|
||||||
public string Key;
|
|
||||||
public Server Server;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,38 +0,0 @@
|
|||||||
using System;
|
|
||||||
using System.IO;
|
|
||||||
using Newtonsoft.Json;
|
|
||||||
using NLog;
|
|
||||||
using Logger = NLog.Logger;
|
|
||||||
|
|
||||||
namespace Ragon.Core
|
|
||||||
{
|
|
||||||
public static class ConfigurationLoader
|
|
||||||
{
|
|
||||||
private static readonly Logger _logger = LogManager.GetCurrentClassLogger();
|
|
||||||
private static readonly string _serverVersion = "1.0.0-rc";
|
|
||||||
|
|
||||||
private static void CopyrightInfo()
|
|
||||||
{
|
|
||||||
_logger.Info($"Server Version: {_serverVersion}");
|
|
||||||
_logger.Info($"Machine Name: {Environment.MachineName}");
|
|
||||||
_logger.Info($"OS: {Environment.OSVersion}");
|
|
||||||
_logger.Info($"Processors: {Environment.ProcessorCount}");
|
|
||||||
_logger.Info($"Runtime Version: {Environment.Version}");
|
|
||||||
|
|
||||||
_logger.Info("==================================");
|
|
||||||
_logger.Info("= =");
|
|
||||||
_logger.Info($"={"Ragon".PadBoth(32)}=");
|
|
||||||
_logger.Info("= =");
|
|
||||||
_logger.Info("==================================");
|
|
||||||
}
|
|
||||||
|
|
||||||
public static Configuration Load(string filePath)
|
|
||||||
{
|
|
||||||
CopyrightInfo();
|
|
||||||
|
|
||||||
var data = File.ReadAllText(filePath);
|
|
||||||
var configuration = JsonConvert.DeserializeObject<Configuration>(data);
|
|
||||||
return configuration;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,22 +0,0 @@
|
|||||||
using Ragon.Common;
|
|
||||||
|
|
||||||
namespace Ragon.Core;
|
|
||||||
|
|
||||||
public class Entity
|
|
||||||
{
|
|
||||||
private static int _idGenerator = 0;
|
|
||||||
public int EntityId { get; private set; }
|
|
||||||
public uint OwnerId { get; private set; }
|
|
||||||
public ushort EntityType { get; private set; }
|
|
||||||
public RagonAuthority Authority { get; private set; }
|
|
||||||
public EntityState State { get; private set; }
|
|
||||||
|
|
||||||
public Entity(uint ownerId, ushort entityType, RagonAuthority stateAuthority, RagonAuthority eventAuthority)
|
|
||||||
{
|
|
||||||
OwnerId = ownerId;
|
|
||||||
EntityType = entityType;
|
|
||||||
EntityId = _idGenerator++;
|
|
||||||
State = new EntityState(stateAuthority);
|
|
||||||
Authority = eventAuthority;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,32 +0,0 @@
|
|||||||
using System;
|
|
||||||
using System.Runtime.Serialization;
|
|
||||||
using Ragon.Common;
|
|
||||||
|
|
||||||
namespace Ragon.Core;
|
|
||||||
|
|
||||||
public class EntityState
|
|
||||||
{
|
|
||||||
public bool isDirty { get; private set; }
|
|
||||||
public RagonAuthority Authority { get; private set; }
|
|
||||||
|
|
||||||
public byte[] Data
|
|
||||||
{
|
|
||||||
get => Data;
|
|
||||||
set
|
|
||||||
{
|
|
||||||
Data = value;
|
|
||||||
isDirty = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public EntityState(RagonAuthority ragonAuthority)
|
|
||||||
{
|
|
||||||
Authority = ragonAuthority;
|
|
||||||
isDirty = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void Clear()
|
|
||||||
{
|
|
||||||
isDirty = true;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,7 +0,0 @@
|
|||||||
namespace Ragon.Core;
|
|
||||||
|
|
||||||
public enum DeliveryType
|
|
||||||
{
|
|
||||||
Unreliable,
|
|
||||||
Reliable
|
|
||||||
}
|
|
||||||
@@ -1,12 +0,0 @@
|
|||||||
using System;
|
|
||||||
|
|
||||||
namespace Ragon.Core
|
|
||||||
{
|
|
||||||
public struct Event
|
|
||||||
{
|
|
||||||
public EventType Type;
|
|
||||||
public DeliveryType Delivery;
|
|
||||||
public byte[] Data;
|
|
||||||
public uint PeerId;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,39 +0,0 @@
|
|||||||
using System.Collections.Generic;
|
|
||||||
using System.Runtime.ExceptionServices;
|
|
||||||
using DisruptorUnity3d;
|
|
||||||
|
|
||||||
namespace Ragon.Core;
|
|
||||||
|
|
||||||
public class EventStream
|
|
||||||
{
|
|
||||||
private Stack<Event> _pool = new Stack<Event>(1024);
|
|
||||||
private RingBuffer<Event> _events = new RingBuffer<Event>(1024);
|
|
||||||
|
|
||||||
// public ref Event Reserve()
|
|
||||||
// {
|
|
||||||
// if (_pool.Count == 0)
|
|
||||||
// {
|
|
||||||
// var evnt = new Event();
|
|
||||||
// return ref evnt;
|
|
||||||
// }
|
|
||||||
// // var evnt = _pool.Pop();
|
|
||||||
// // ref Event evntRef = ref evnt;
|
|
||||||
// // return evntRef;
|
|
||||||
// }
|
|
||||||
|
|
||||||
// public void Retain(ref Event @event)
|
|
||||||
// {
|
|
||||||
//
|
|
||||||
// }
|
|
||||||
//
|
|
||||||
// public void WriteEvent(ref Event evnt)
|
|
||||||
// {
|
|
||||||
// // _pool.Push(evnt);
|
|
||||||
// _events.Enqueue(evnt);
|
|
||||||
// }
|
|
||||||
//
|
|
||||||
// public ref Event ReadEvent()
|
|
||||||
// {
|
|
||||||
//
|
|
||||||
// }
|
|
||||||
}
|
|
||||||
@@ -1,10 +0,0 @@
|
|||||||
namespace Ragon.Core
|
|
||||||
{
|
|
||||||
public enum EventType
|
|
||||||
{
|
|
||||||
CONNECTED,
|
|
||||||
DISCONNECTED,
|
|
||||||
TIMEOUT,
|
|
||||||
DATA,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,14 +0,0 @@
|
|||||||
using System;
|
|
||||||
|
|
||||||
namespace Ragon.Core
|
|
||||||
{
|
|
||||||
public static class StringExtensions
|
|
||||||
{
|
|
||||||
public static string PadBoth(this string str, int length)
|
|
||||||
{
|
|
||||||
int spaces = length - str.Length;
|
|
||||||
int padLeft = spaces / 2 + str.Length;
|
|
||||||
return str.PadLeft(padLeft).PadRight(length);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,14 +0,0 @@
|
|||||||
using System;
|
|
||||||
|
|
||||||
namespace Ragon.Core
|
|
||||||
{
|
|
||||||
public static class ValueExtensions
|
|
||||||
{
|
|
||||||
public static T Clamp<T>(this T val, T min, T max) where T : IComparable<T>
|
|
||||||
{
|
|
||||||
if (val.CompareTo(min) < 0) return min;
|
|
||||||
else if (val.CompareTo(max) > 0) return max;
|
|
||||||
else return val;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,15 +0,0 @@
|
|||||||
using System;
|
|
||||||
using System.Collections.Generic;
|
|
||||||
|
|
||||||
namespace Ragon.Core
|
|
||||||
{
|
|
||||||
public class Player
|
|
||||||
{
|
|
||||||
public uint PeerId { get; set; }
|
|
||||||
public string PlayerName { get; set; }
|
|
||||||
public bool IsLoaded { get; set; }
|
|
||||||
|
|
||||||
public List<Entity> Entities;
|
|
||||||
public List<int> EntitiesIds;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,289 +0,0 @@
|
|||||||
using System;
|
|
||||||
using System.Collections.Generic;
|
|
||||||
using System.Runtime.CompilerServices;
|
|
||||||
using NetStack.Serialization;
|
|
||||||
using NLog;
|
|
||||||
using Ragon.Common;
|
|
||||||
|
|
||||||
namespace Ragon.Core
|
|
||||||
{
|
|
||||||
public class PluginBase: IDisposable
|
|
||||||
{
|
|
||||||
private delegate void SubscribeDelegate(Player player, ref ReadOnlySpan<byte> data);
|
|
||||||
private delegate void SubscribeEntityDelegate(Player player, Entity entity, ref ReadOnlySpan<byte> data);
|
|
||||||
|
|
||||||
private Dictionary<ushort, SubscribeDelegate> _globalEvents = new();
|
|
||||||
private Dictionary<int, Dictionary<ushort, SubscribeEntityDelegate>> _entityEvents = new();
|
|
||||||
private BitBuffer _buffer = new BitBuffer(8192);
|
|
||||||
|
|
||||||
protected Room Room { get; private set; }
|
|
||||||
protected ILogger _logger;
|
|
||||||
|
|
||||||
public void Attach(Room room)
|
|
||||||
{
|
|
||||||
_logger = LogManager.GetLogger($"Plugin<{GetType().Name}>");
|
|
||||||
|
|
||||||
Room = room;
|
|
||||||
|
|
||||||
_globalEvents.Clear();
|
|
||||||
_entityEvents.Clear();
|
|
||||||
}
|
|
||||||
public void Dispose()
|
|
||||||
{
|
|
||||||
_globalEvents.Clear();
|
|
||||||
_entityEvents.Clear();
|
|
||||||
}
|
|
||||||
|
|
||||||
public void Subscribe<T>(ushort evntCode, Action<Player, T> action) where T : IRagonSerializable, new()
|
|
||||||
{
|
|
||||||
if (_globalEvents.ContainsKey(evntCode))
|
|
||||||
{
|
|
||||||
_logger.Warn($"Event subscriber already added {evntCode}");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
var data = new T();
|
|
||||||
_globalEvents.Add(evntCode, (Player player, ref ReadOnlySpan<byte> raw) =>
|
|
||||||
{
|
|
||||||
if (raw.Length == 0)
|
|
||||||
{
|
|
||||||
_logger.Warn($"Payload is empty for event {evntCode}");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
_buffer.Clear();
|
|
||||||
_buffer.FromSpan(ref raw, raw.Length);
|
|
||||||
data.Deserialize(_buffer);
|
|
||||||
action.Invoke(player, data);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
public void Subscribe(ushort evntCode, Action<Player> action)
|
|
||||||
{
|
|
||||||
if (_globalEvents.ContainsKey(evntCode))
|
|
||||||
{
|
|
||||||
_logger.Warn($"Event subscriber already added {evntCode}");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
_globalEvents.Add(evntCode, (Player player, ref ReadOnlySpan<byte> raw) =>
|
|
||||||
{
|
|
||||||
action.Invoke(player);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
public void Subscribe<T>(Entity entity, ushort evntCode, Action<Player, Entity, T> action) where T : IRagonSerializable, new()
|
|
||||||
{
|
|
||||||
if (_entityEvents.ContainsKey(entity.EntityId))
|
|
||||||
{
|
|
||||||
if (_entityEvents[entity.EntityId].ContainsKey(evntCode))
|
|
||||||
{
|
|
||||||
_logger.Warn($"Event subscriber already added {evntCode} for {entity.EntityId}");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
var data = new T();
|
|
||||||
_entityEvents[entity.EntityId].Add(evntCode, (Player player, Entity ent, ref ReadOnlySpan<byte> raw) =>
|
|
||||||
{
|
|
||||||
if (raw.Length == 0)
|
|
||||||
{
|
|
||||||
_logger.Warn($"Payload is empty for entity {ent.EntityId} event {evntCode}");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
_buffer.Clear();
|
|
||||||
_buffer.FromSpan(ref raw, raw.Length);
|
|
||||||
data.Deserialize(_buffer);
|
|
||||||
action.Invoke(player, ent, data);
|
|
||||||
});
|
|
||||||
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
{
|
|
||||||
var data = new T();
|
|
||||||
_entityEvents.Add(entity.EntityId, new Dictionary<ushort, SubscribeEntityDelegate>());
|
|
||||||
_entityEvents[entity.EntityId].Add(evntCode, (Player player, Entity ent, ref ReadOnlySpan<byte> raw) =>
|
|
||||||
{
|
|
||||||
if (raw.Length == 0)
|
|
||||||
{
|
|
||||||
_logger.Warn($"Payload is empty for entity {ent.EntityId} event {evntCode}");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
_buffer.Clear();
|
|
||||||
_buffer.FromSpan(ref raw, raw.Length);
|
|
||||||
data.Deserialize(_buffer);
|
|
||||||
action.Invoke(player, ent, data);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public void Subscribe(Entity entity, ushort evntCode, Action<Player, Entity> action)
|
|
||||||
{
|
|
||||||
if (_entityEvents.ContainsKey(entity.EntityId))
|
|
||||||
{
|
|
||||||
if (_entityEvents[entity.EntityId].ContainsKey(evntCode))
|
|
||||||
{
|
|
||||||
_logger.Warn($"Event subscriber already added {evntCode} for {entity.EntityId}");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
_entityEvents[entity.EntityId].Add(evntCode, (Player player, Entity ent, ref ReadOnlySpan<byte> raw) =>
|
|
||||||
{
|
|
||||||
action.Invoke(player, ent);
|
|
||||||
});
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
{
|
|
||||||
_entityEvents.Add(entity.EntityId, new Dictionary<ushort, SubscribeEntityDelegate>());
|
|
||||||
_entityEvents[entity.EntityId].Add(evntCode, (Player player, Entity ent, ref ReadOnlySpan<byte> raw) =>
|
|
||||||
{
|
|
||||||
action.Invoke(player, ent);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public void UnsubscribeAll()
|
|
||||||
{
|
|
||||||
_globalEvents.Clear();
|
|
||||||
_entityEvents.Clear();
|
|
||||||
}
|
|
||||||
|
|
||||||
public bool InternalHandle(uint peerId, int entityId, ushort evntCode, ref ReadOnlySpan<byte> payload)
|
|
||||||
{
|
|
||||||
if (!_entityEvents.ContainsKey(entityId))
|
|
||||||
return false;
|
|
||||||
|
|
||||||
if (!_entityEvents[entityId].ContainsKey(evntCode))
|
|
||||||
return false;
|
|
||||||
|
|
||||||
var player = Room.GetPlayerById(peerId);
|
|
||||||
var entity = Room.GetEntityById(entityId);
|
|
||||||
_entityEvents[entityId][evntCode].Invoke(player, entity, ref payload);
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
public bool InternalHandle(uint peerId, ushort evntCode, ref ReadOnlySpan<byte> payload)
|
|
||||||
{
|
|
||||||
if (_globalEvents.ContainsKey(evntCode))
|
|
||||||
{
|
|
||||||
var player = Room.GetPlayerById(peerId);
|
|
||||||
_globalEvents[evntCode].Invoke(player, ref payload);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void SendEvent(Player player, uint eventCode, IRagonSerializable payload)
|
|
||||||
{
|
|
||||||
_buffer.Clear();
|
|
||||||
payload.Serialize(_buffer);
|
|
||||||
|
|
||||||
var sendData = new byte[_buffer.Length + 4];
|
|
||||||
Span<byte> data = sendData.AsSpan();
|
|
||||||
Span<byte> operationData = data.Slice(0, 2);
|
|
||||||
Span<byte> eventCodeData = data.Slice(2, 2);
|
|
||||||
Span<byte> payloadData = data.Slice(4, data.Length - 4);
|
|
||||||
|
|
||||||
_buffer.ToSpan(ref payloadData);
|
|
||||||
|
|
||||||
RagonHeader.WriteUShort((ushort) RagonOperation.REPLICATE_EVENT, ref operationData);
|
|
||||||
RagonHeader.WriteUShort((ushort) eventCode, ref eventCodeData);
|
|
||||||
|
|
||||||
Room.Send(player.PeerId, sendData);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void SendEvent(ushort eventCode, IRagonSerializable payload)
|
|
||||||
{
|
|
||||||
_buffer.Clear();
|
|
||||||
payload.Serialize(_buffer);
|
|
||||||
|
|
||||||
var sendData = new byte[_buffer.Length + 4];
|
|
||||||
Span<byte> data = sendData.AsSpan();
|
|
||||||
Span<byte> operationData = data.Slice(0, 2);
|
|
||||||
Span<byte> eventCodeData = data.Slice(2, 2);
|
|
||||||
Span<byte> payloadData = data.Slice(4, _buffer.Length);
|
|
||||||
|
|
||||||
RagonHeader.WriteUShort((ushort) RagonOperation.REPLICATE_EVENT,ref operationData);
|
|
||||||
RagonHeader.WriteUShort( eventCode, ref eventCodeData);
|
|
||||||
|
|
||||||
_buffer.ToSpan(ref payloadData);
|
|
||||||
|
|
||||||
Room.Broadcast(sendData);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void SendEntityEvent(Player player, Entity entity, IRagonSerializable payload)
|
|
||||||
{
|
|
||||||
_buffer.Clear();
|
|
||||||
payload.Serialize(_buffer);
|
|
||||||
|
|
||||||
var sendData = new byte[_buffer.Length + 6];
|
|
||||||
Span<byte> data = sendData.AsSpan();
|
|
||||||
Span<byte> operationData = data.Slice(0, 2);
|
|
||||||
Span<byte> entityData = data.Slice(2, 4);
|
|
||||||
|
|
||||||
RagonHeader.WriteUShort((ushort) RagonOperation.REPLICATE_EVENT, ref operationData);
|
|
||||||
RagonHeader.WriteInt(entity.EntityId, ref entityData);
|
|
||||||
|
|
||||||
Room.Send(player.PeerId, sendData);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void SendEntityEvent(Entity entity, IRagonSerializable payload)
|
|
||||||
{
|
|
||||||
_buffer.Clear();
|
|
||||||
payload.Serialize(_buffer);
|
|
||||||
|
|
||||||
var sendData = new byte[_buffer.Length + 6];
|
|
||||||
Span<byte> data = sendData.AsSpan();
|
|
||||||
Span<byte> operationData = data.Slice(0, 2);
|
|
||||||
Span<byte> entityData = data.Slice(2, 4);
|
|
||||||
|
|
||||||
RagonHeader.WriteUShort((ushort) RagonOperation.REPLICATE_EVENT, ref operationData);
|
|
||||||
RagonHeader.WriteInt(entity.EntityId, ref entityData);
|
|
||||||
|
|
||||||
Room.Broadcast(sendData);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
#region VIRTUAL
|
|
||||||
|
|
||||||
public virtual void OnPlayerJoined(Player player)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
public virtual void OnPlayerLeaved(Player player)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
public virtual void OnOwnerChanged(Player player)
|
|
||||||
{
|
|
||||||
|
|
||||||
}
|
|
||||||
public virtual void OnEntityCreated(Player creator, Entity entity)
|
|
||||||
{
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
public virtual void OnEntityDestroyed(Player destoyer, Entity entity)
|
|
||||||
{
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
public virtual void OnStart()
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
public virtual void OnStop()
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
public virtual void OnTick(ulong ticks, float deltaTime)
|
|
||||||
{
|
|
||||||
}
|
|
||||||
|
|
||||||
#endregion
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,10 +0,0 @@
|
|||||||
namespace Ragon.Core
|
|
||||||
{
|
|
||||||
public interface PluginFactory
|
|
||||||
{
|
|
||||||
public PluginBase CreatePlugin(string map);
|
|
||||||
public AuthorizationManager CreateManager(Configuration configuration);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
}
|
|
||||||
@@ -1,393 +0,0 @@
|
|||||||
using System;
|
|
||||||
using System.Collections.Generic;
|
|
||||||
using System.Linq;
|
|
||||||
using System.Runtime.CompilerServices;
|
|
||||||
using System.Runtime.InteropServices;
|
|
||||||
using System.Text;
|
|
||||||
using NetStack.Serialization;
|
|
||||||
using NLog;
|
|
||||||
using Ragon.Common;
|
|
||||||
|
|
||||||
namespace Ragon.Core
|
|
||||||
{
|
|
||||||
public class Room : IDisposable
|
|
||||||
{
|
|
||||||
public int PlayersMin { get; private set; }
|
|
||||||
public int PlayersMax { get; private set; }
|
|
||||||
public int PlayersCount => _players.Count;
|
|
||||||
public string Id { get; private set; }
|
|
||||||
public string Map { get; private set; }
|
|
||||||
|
|
||||||
private ILogger _logger = LogManager.GetCurrentClassLogger();
|
|
||||||
private Dictionary<uint, Player> _players = new();
|
|
||||||
private Dictionary<int, Entity> _entities = new();
|
|
||||||
private uint _owner;
|
|
||||||
private uint _ticks;
|
|
||||||
|
|
||||||
private readonly PluginBase _plugin;
|
|
||||||
private readonly RoomThread _roomThread;
|
|
||||||
|
|
||||||
// Cache
|
|
||||||
private uint[] _readyPlayers = Array.Empty<uint>();
|
|
||||||
private uint[] _allPlayers = Array.Empty<uint>();
|
|
||||||
private Entity[] _entitiesAll = Array.Empty<Entity>();
|
|
||||||
|
|
||||||
public Room(RoomThread roomThread, PluginBase pluginBase, string map, int min, int max)
|
|
||||||
{
|
|
||||||
_roomThread = roomThread;
|
|
||||||
_plugin = pluginBase;
|
|
||||||
|
|
||||||
Map = map;
|
|
||||||
PlayersMin = min;
|
|
||||||
PlayersMax = max;
|
|
||||||
Id = Guid.NewGuid().ToString();
|
|
||||||
|
|
||||||
_logger.Info($"Room created with plugin: {_plugin.GetType().Name}");
|
|
||||||
_plugin.Attach(this);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void Joined(uint peerId, ReadOnlySpan<byte> payload)
|
|
||||||
{
|
|
||||||
if (_players.Count == 0)
|
|
||||||
{
|
|
||||||
_owner = peerId;
|
|
||||||
}
|
|
||||||
|
|
||||||
var player = new Player()
|
|
||||||
{
|
|
||||||
PlayerName = "Player " + peerId,
|
|
||||||
PeerId = peerId,
|
|
||||||
IsLoaded = false,
|
|
||||||
Entities = new List<Entity>(),
|
|
||||||
EntitiesIds = new List<int>(),
|
|
||||||
};
|
|
||||||
|
|
||||||
_players.Add(peerId, player);
|
|
||||||
_allPlayers = _players.Select(p => p.Key).ToArray();
|
|
||||||
|
|
||||||
{
|
|
||||||
var idRaw = Encoding.UTF8.GetBytes(Id).AsSpan();
|
|
||||||
|
|
||||||
var sendData = new byte[idRaw.Length + 18];
|
|
||||||
var data = sendData.AsSpan();
|
|
||||||
|
|
||||||
Span<byte> operationData = data.Slice(0, 2);
|
|
||||||
Span<byte> peerData = data.Slice(2, 4);
|
|
||||||
Span<byte> ownerData = data.Slice(4, 4);
|
|
||||||
Span<byte> minData = data.Slice(10, 4);
|
|
||||||
Span<byte> maxData = data.Slice(14, 4);
|
|
||||||
Span<byte> idData = data.Slice(18, idRaw.Length);
|
|
||||||
|
|
||||||
RagonHeader.WriteUShort((ushort) RagonOperation.JOIN_ROOM, ref operationData);
|
|
||||||
RagonHeader.WriteInt((int) peerId, ref peerData);
|
|
||||||
RagonHeader.WriteInt((int) _owner, ref ownerData);
|
|
||||||
RagonHeader.WriteInt(PlayersMin, ref minData);
|
|
||||||
RagonHeader.WriteInt(PlayersMax, ref maxData);
|
|
||||||
|
|
||||||
idRaw.CopyTo(idData);
|
|
||||||
|
|
||||||
Send(peerId, sendData);
|
|
||||||
}
|
|
||||||
|
|
||||||
{
|
|
||||||
var sceneRawData = Encoding.UTF8.GetBytes(Map).AsSpan();
|
|
||||||
var sendData = new byte[sceneRawData.Length + 2];
|
|
||||||
var data = sendData.AsSpan();
|
|
||||||
|
|
||||||
Span<byte> operationData = data.Slice(0, 2);
|
|
||||||
Span<byte> sceneData = data.Slice(2, sceneRawData.Length);
|
|
||||||
|
|
||||||
RagonHeader.WriteUShort((ushort) RagonOperation.LOAD_SCENE, ref operationData);
|
|
||||||
sceneRawData.CopyTo(sceneData);
|
|
||||||
|
|
||||||
Send(peerId, sendData, DeliveryType.Reliable);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public void Leave(uint peerId)
|
|
||||||
{
|
|
||||||
if (_players.Remove(peerId, out var player))
|
|
||||||
{
|
|
||||||
_allPlayers = _players.Select(p => p.Key).ToArray();
|
|
||||||
|
|
||||||
_plugin.OnPlayerLeaved(player);
|
|
||||||
|
|
||||||
foreach (var entityId in player.EntitiesIds)
|
|
||||||
{
|
|
||||||
var sendData = new byte[6];
|
|
||||||
var entityData = sendData.AsSpan();
|
|
||||||
var operationData = entityData.Slice(0, 2);
|
|
||||||
|
|
||||||
RagonHeader.WriteUShort((ushort) RagonOperation.DESTROY_ENTITY, ref operationData);
|
|
||||||
RagonHeader.WriteInt(entityId, ref entityData);
|
|
||||||
|
|
||||||
Broadcast(_allPlayers, sendData);
|
|
||||||
|
|
||||||
_entities.Remove(entityId);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public void ProcessEvent(RagonOperation operation, uint peerId, ReadOnlySpan<byte> rawData)
|
|
||||||
{
|
|
||||||
switch (operation)
|
|
||||||
{
|
|
||||||
case RagonOperation.REPLICATE_ENTITY_STATE:
|
|
||||||
{
|
|
||||||
var entityData = rawData.Slice(2, 4);
|
|
||||||
var entityId = RagonHeader.ReadInt(ref entityData);
|
|
||||||
if (_entities.TryGetValue(entityId, out var ent))
|
|
||||||
{
|
|
||||||
if (ent.State.Authority == RagonAuthority.OWNER_ONLY && ent.OwnerId != peerId)
|
|
||||||
return;
|
|
||||||
|
|
||||||
ent.State.Data = rawData.Slice(6, rawData.Length - 6).ToArray();
|
|
||||||
|
|
||||||
var data = new byte[rawData.Length];
|
|
||||||
|
|
||||||
rawData.CopyTo(data);
|
|
||||||
|
|
||||||
Broadcast(_readyPlayers, data);
|
|
||||||
}
|
|
||||||
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case RagonOperation.REPLICATE_ENTITY_EVENT:
|
|
||||||
{
|
|
||||||
var evntCodeData = rawData.Slice(2, 2);
|
|
||||||
var entityIdData = rawData.Slice(4, 4);
|
|
||||||
var evntId = RagonHeader.ReadUShort(ref evntCodeData);
|
|
||||||
var entityId = RagonHeader.ReadInt(ref entityIdData);
|
|
||||||
|
|
||||||
if (!_entities.TryGetValue(entityId, out var ent))
|
|
||||||
return;
|
|
||||||
|
|
||||||
if (ent.Authority == RagonAuthority.OWNER_ONLY && ent.OwnerId != peerId)
|
|
||||||
return;
|
|
||||||
|
|
||||||
var payload = rawData.Slice(8, rawData.Length - 8);
|
|
||||||
if (_plugin.InternalHandle(peerId, entityId, evntId, ref payload))
|
|
||||||
return;
|
|
||||||
|
|
||||||
var data = new byte[rawData.Length];
|
|
||||||
|
|
||||||
rawData.CopyTo(data);
|
|
||||||
|
|
||||||
Broadcast(_readyPlayers, data, DeliveryType.Reliable);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case RagonOperation.REPLICATE_EVENT:
|
|
||||||
{
|
|
||||||
var evntCodeData = rawData.Slice(2, 2);
|
|
||||||
var evntId = RagonHeader.ReadUShort(ref evntCodeData);
|
|
||||||
|
|
||||||
var payload = rawData.Slice(4, rawData.Length - 4);
|
|
||||||
if (_plugin.InternalHandle(peerId, evntId, ref payload))
|
|
||||||
return;
|
|
||||||
|
|
||||||
var data = new byte[rawData.Length];
|
|
||||||
|
|
||||||
rawData.CopyTo(data);
|
|
||||||
|
|
||||||
Broadcast(_readyPlayers, data, DeliveryType.Reliable);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case RagonOperation.CREATE_ENTITY:
|
|
||||||
{
|
|
||||||
var typeData = rawData.Slice(2, 2);
|
|
||||||
var authorityData = rawData.Slice(2, 2);
|
|
||||||
var entityPayloadData = rawData.Slice(4, rawData.Length - 4);
|
|
||||||
|
|
||||||
var entityType = RagonHeader.ReadUShort(ref typeData);
|
|
||||||
var stateAuthority = (RagonAuthority) authorityData[0];
|
|
||||||
var eventAuthority = (RagonAuthority) authorityData[1];
|
|
||||||
var entity = new Entity(peerId, entityType, stateAuthority, eventAuthority);
|
|
||||||
entity.State.Data = entityPayloadData.ToArray();
|
|
||||||
|
|
||||||
var player = _players[peerId];
|
|
||||||
player.Entities.Add(entity);
|
|
||||||
player.EntitiesIds.Add(entity.EntityId);
|
|
||||||
|
|
||||||
_entities.Add(entity.EntityId, entity);
|
|
||||||
_entitiesAll = _entities.Values.ToArray();
|
|
||||||
|
|
||||||
_plugin.OnEntityCreated(player, entity);
|
|
||||||
|
|
||||||
var data = new byte[entityPayloadData.Length + 14];
|
|
||||||
var sendData = data.AsSpan();
|
|
||||||
var operationData = sendData.Slice(0, 2);
|
|
||||||
var entityTypeData = sendData.Slice(2, 2);
|
|
||||||
var authority = sendData.Slice(4, 2);
|
|
||||||
var entityIdData = sendData.Slice(6, 4);
|
|
||||||
var peerData = sendData.Slice(10, 4);
|
|
||||||
var payload = sendData.Slice(14, entityPayloadData.Length);
|
|
||||||
|
|
||||||
entityPayloadData.CopyTo(payload);
|
|
||||||
|
|
||||||
authority[0] = authorityData[0];
|
|
||||||
authority[1] = authorityData[1];
|
|
||||||
|
|
||||||
RagonHeader.WriteUShort((ushort) RagonOperation.CREATE_ENTITY, ref operationData);
|
|
||||||
RagonHeader.WriteUShort(entityType, ref entityTypeData);
|
|
||||||
RagonHeader.WriteInt(entity.EntityId, ref entityIdData);
|
|
||||||
RagonHeader.WriteInt((int) peerId, ref peerData);
|
|
||||||
|
|
||||||
Broadcast(_allPlayers, data, DeliveryType.Reliable);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case RagonOperation.DESTROY_ENTITY:
|
|
||||||
{
|
|
||||||
var entityData = rawData.Slice(2, 4);
|
|
||||||
var entityId = RagonHeader.ReadInt(ref entityData);
|
|
||||||
if (_entities.TryGetValue(entityId, out var entity))
|
|
||||||
{
|
|
||||||
if (entity.Authority == RagonAuthority.OWNER_ONLY && entity.OwnerId != peerId)
|
|
||||||
return;
|
|
||||||
|
|
||||||
var player = _players[peerId];
|
|
||||||
|
|
||||||
player.Entities.Remove(entity);
|
|
||||||
player.EntitiesIds.Remove(entity.EntityId);
|
|
||||||
|
|
||||||
_entities.Remove(entityId);
|
|
||||||
_entitiesAll = _entities.Values.ToArray();
|
|
||||||
|
|
||||||
_plugin.OnEntityDestroyed(player, entity);
|
|
||||||
|
|
||||||
var data = new byte[rawData.Length];
|
|
||||||
Span<byte> sendData = data.AsSpan();
|
|
||||||
rawData.CopyTo(sendData);
|
|
||||||
|
|
||||||
Broadcast(_readyPlayers, data, DeliveryType.Reliable);
|
|
||||||
}
|
|
||||||
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case RagonOperation.SCENE_IS_LOADED:
|
|
||||||
{
|
|
||||||
Send(peerId, RagonOperation.RESTORE_BEGIN, DeliveryType.Reliable);
|
|
||||||
foreach (var entity in _entities.Values)
|
|
||||||
{
|
|
||||||
var entityState = entity.State.Data.AsSpan();
|
|
||||||
var data = new byte[entity.State.Data.Length + 12];
|
|
||||||
|
|
||||||
Span<byte> sendData = data.AsSpan();
|
|
||||||
Span<byte> operationData = sendData.Slice(0, 2);
|
|
||||||
Span<byte> entityTypeData = sendData.Slice(2, 2);
|
|
||||||
Span<byte> authorityData = sendData.Slice(4, 2);
|
|
||||||
Span<byte> entityData = sendData.Slice(6, 4);
|
|
||||||
Span<byte> ownerData = sendData.Slice(10, 4);
|
|
||||||
Span<byte> entityStateData = sendData.Slice(14, entity.State.Data.Length);
|
|
||||||
|
|
||||||
RagonHeader.WriteUShort((ushort) RagonOperation.CREATE_ENTITY, ref operationData);
|
|
||||||
RagonHeader.WriteUShort(entity.EntityType, ref entityTypeData);
|
|
||||||
RagonHeader.WriteInt(entity.EntityId, ref entityData);
|
|
||||||
RagonHeader.WriteInt((int) entity.OwnerId, ref ownerData);
|
|
||||||
|
|
||||||
authorityData[0] = (byte) entity.State.Authority;
|
|
||||||
authorityData[1] = (byte) entity.Authority;
|
|
||||||
|
|
||||||
entityState.CopyTo(entityStateData);
|
|
||||||
|
|
||||||
Send(peerId, data, DeliveryType.Reliable);
|
|
||||||
}
|
|
||||||
|
|
||||||
Send(peerId, RagonOperation.RESTORE_END, DeliveryType.Reliable);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case RagonOperation.RESTORED:
|
|
||||||
{
|
|
||||||
_players[peerId].IsLoaded = true;
|
|
||||||
_readyPlayers = _players.Where(p => p.Value.IsLoaded).Select(p => p.Key).ToArray();
|
|
||||||
|
|
||||||
_plugin.OnPlayerJoined(_players[peerId]);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public void Tick(float deltaTime)
|
|
||||||
{
|
|
||||||
_ticks++;
|
|
||||||
_plugin.OnTick(_ticks, deltaTime);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void Start()
|
|
||||||
{
|
|
||||||
_logger.Info("Room started");
|
|
||||||
_plugin.OnStart();
|
|
||||||
}
|
|
||||||
|
|
||||||
public void Stop()
|
|
||||||
{
|
|
||||||
_logger.Info("Room stopped");
|
|
||||||
_plugin.OnStop();
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
public void Dispose()
|
|
||||||
{
|
|
||||||
_logger.Info("Room destroyed");
|
|
||||||
_plugin.Dispose();
|
|
||||||
}
|
|
||||||
|
|
||||||
public Player GetPlayerById(uint peerId) => _players[peerId];
|
|
||||||
public Entity GetEntityById(int entityId) => _entities[entityId];
|
|
||||||
public Player GetOwner() => _players[_owner];
|
|
||||||
|
|
||||||
public void Send(uint peerId, RagonOperation operation, DeliveryType deliveryType = DeliveryType.Unreliable)
|
|
||||||
{
|
|
||||||
var rawData = new byte[2];
|
|
||||||
var rawDataSpan = new Span<byte>(rawData);
|
|
||||||
|
|
||||||
RagonHeader.WriteUShort((ushort) operation, ref rawDataSpan);
|
|
||||||
|
|
||||||
_roomThread.WriteOutEvent(new Event()
|
|
||||||
{
|
|
||||||
PeerId = peerId,
|
|
||||||
Data = rawData,
|
|
||||||
Type = EventType.DATA,
|
|
||||||
Delivery = deliveryType,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
public void Send(uint peerId, byte[] rawData, DeliveryType deliveryType = DeliveryType.Unreliable)
|
|
||||||
{
|
|
||||||
_roomThread.WriteOutEvent(new Event()
|
|
||||||
{
|
|
||||||
PeerId = peerId,
|
|
||||||
Data = rawData,
|
|
||||||
Type = EventType.DATA,
|
|
||||||
Delivery = deliveryType,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
public void Broadcast(uint[] peersIds, byte[] rawData, DeliveryType deliveryType = DeliveryType.Unreliable)
|
|
||||||
{
|
|
||||||
foreach (var peer in peersIds)
|
|
||||||
{
|
|
||||||
_roomThread.WriteOutEvent(new Event()
|
|
||||||
{
|
|
||||||
PeerId = peer,
|
|
||||||
Data = rawData,
|
|
||||||
Type = EventType.DATA,
|
|
||||||
Delivery = deliveryType,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public void Broadcast(byte[] rawData, DeliveryType deliveryType = DeliveryType.Unreliable)
|
|
||||||
{
|
|
||||||
foreach (var player in _players.Values.ToArray())
|
|
||||||
{
|
|
||||||
_roomThread.WriteOutEvent(new Event()
|
|
||||||
{
|
|
||||||
PeerId = player.PeerId,
|
|
||||||
Data = rawData,
|
|
||||||
Type = EventType.DATA,
|
|
||||||
Delivery = deliveryType,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,166 +0,0 @@
|
|||||||
using System;
|
|
||||||
using System.Collections.Generic;
|
|
||||||
using System.Text;
|
|
||||||
using NLog;
|
|
||||||
using Ragon.Common;
|
|
||||||
|
|
||||||
namespace Ragon.Core
|
|
||||||
{
|
|
||||||
public class RoomManager
|
|
||||||
{
|
|
||||||
private readonly Logger _logger = LogManager.GetCurrentClassLogger();
|
|
||||||
private List<Room> _rooms;
|
|
||||||
private Dictionary<uint, Room> _peersByRoom;
|
|
||||||
private PluginFactory _factory;
|
|
||||||
private AuthorizationManager _manager;
|
|
||||||
private RoomThread _roomThread;
|
|
||||||
|
|
||||||
public Action<(uint, Room)> OnJoined;
|
|
||||||
public Action<(uint, Room)> OnLeaved;
|
|
||||||
|
|
||||||
public RoomManager(RoomThread roomThread, PluginFactory factory)
|
|
||||||
{
|
|
||||||
_roomThread = roomThread;
|
|
||||||
_factory = factory;
|
|
||||||
|
|
||||||
_manager = _factory.CreateManager(roomThread.Configuration);
|
|
||||||
_rooms = new List<Room>();
|
|
||||||
_peersByRoom = new Dictionary<uint, Room>();
|
|
||||||
}
|
|
||||||
|
|
||||||
public void ProcessEvent(RagonOperation operation, uint peerId, ReadOnlySpan<byte> payload)
|
|
||||||
{
|
|
||||||
switch (operation)
|
|
||||||
{
|
|
||||||
case RagonOperation.AUTHORIZE:
|
|
||||||
{
|
|
||||||
OnAuthorize(peerId, payload);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case RagonOperation.JOIN_ROOM:
|
|
||||||
{
|
|
||||||
var room = Join(peerId, payload);
|
|
||||||
OnJoined?.Invoke((peerId, room));
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case RagonOperation.LEAVE_ROOM:
|
|
||||||
{
|
|
||||||
var room = Left(peerId, payload);
|
|
||||||
OnLeaved((peerId, room));
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public void OnAuthorize(uint peerId, ReadOnlySpan<byte> payload)
|
|
||||||
{
|
|
||||||
if (_manager.OnAuthorize(peerId, ref payload))
|
|
||||||
{
|
|
||||||
var sendData = new byte[2];
|
|
||||||
Span<byte> data = sendData.AsSpan();
|
|
||||||
|
|
||||||
RagonHeader.WriteUShort((ushort) RagonOperation.AUTHORIZED_SUCCESS, ref data);
|
|
||||||
|
|
||||||
_roomThread.WriteOutEvent(new Event()
|
|
||||||
{
|
|
||||||
Delivery = DeliveryType.Reliable,
|
|
||||||
Type = EventType.DATA,
|
|
||||||
Data = sendData,
|
|
||||||
PeerId = peerId,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
var sendData = new byte[2];
|
|
||||||
var data = sendData.AsSpan();
|
|
||||||
RagonHeader.WriteUShort((ushort) RagonOperation.AUTHORIZED_FAILED, ref data);
|
|
||||||
|
|
||||||
_roomThread.WriteOutEvent(new Event()
|
|
||||||
{
|
|
||||||
Delivery = DeliveryType.Reliable,
|
|
||||||
Type = EventType.DATA,
|
|
||||||
Data = sendData,
|
|
||||||
PeerId = peerId,
|
|
||||||
});
|
|
||||||
|
|
||||||
_roomThread.WriteOutEvent(new Event()
|
|
||||||
{
|
|
||||||
Delivery = DeliveryType.Reliable,
|
|
||||||
Type = EventType.DISCONNECTED,
|
|
||||||
Data = Array.Empty<byte>(),
|
|
||||||
PeerId = peerId,
|
|
||||||
});
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public Room Join(uint peerId, ReadOnlySpan<byte> payload)
|
|
||||||
{
|
|
||||||
var minData = payload.Slice(0, 2);
|
|
||||||
var maxData = payload.Slice(2, 2);
|
|
||||||
var mapData = payload.Slice(4, payload.Length - 4);
|
|
||||||
|
|
||||||
var map = Encoding.UTF8.GetString(mapData);
|
|
||||||
var min = RagonHeader.ReadUShort(ref minData);
|
|
||||||
var max = RagonHeader.ReadUShort(ref maxData);
|
|
||||||
|
|
||||||
Room room = null;
|
|
||||||
if (_rooms.Count > 0)
|
|
||||||
{
|
|
||||||
foreach (var existRoom in _rooms)
|
|
||||||
{
|
|
||||||
if (existRoom.Map == map && existRoom.PlayersCount < existRoom.PlayersMax)
|
|
||||||
{
|
|
||||||
room = existRoom;
|
|
||||||
room.Joined(peerId, payload);
|
|
||||||
|
|
||||||
_peersByRoom.Add(peerId, room);
|
|
||||||
|
|
||||||
return room;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
var plugin = _factory.CreatePlugin(map);
|
|
||||||
if (plugin == null)
|
|
||||||
throw new NullReferenceException($"Plugin for map {map} is null");
|
|
||||||
|
|
||||||
room = new Room(_roomThread, plugin, map, min, max);
|
|
||||||
room.Joined(peerId, payload);
|
|
||||||
room.Start();
|
|
||||||
|
|
||||||
_peersByRoom.Add(peerId, room);
|
|
||||||
_rooms.Add(room);
|
|
||||||
|
|
||||||
return room;
|
|
||||||
}
|
|
||||||
|
|
||||||
public Room Left(uint peerId, ReadOnlySpan<byte> payload)
|
|
||||||
{
|
|
||||||
_peersByRoom.Remove(peerId, out var room);
|
|
||||||
|
|
||||||
return room;
|
|
||||||
}
|
|
||||||
|
|
||||||
public void Disconnected(uint peerId)
|
|
||||||
{
|
|
||||||
_peersByRoom.Remove(peerId, out var room);
|
|
||||||
if (room != null)
|
|
||||||
{
|
|
||||||
room.Leave(peerId);
|
|
||||||
if (room.PlayersCount <= 0)
|
|
||||||
{
|
|
||||||
_rooms.Remove(room);
|
|
||||||
|
|
||||||
room.Stop();
|
|
||||||
room.Dispose();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public void Tick(float deltaTime)
|
|
||||||
{
|
|
||||||
foreach (Room room in _rooms)
|
|
||||||
room.Tick(deltaTime);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,110 +0,0 @@
|
|||||||
using System;
|
|
||||||
using System.Collections.Generic;
|
|
||||||
using System.Diagnostics;
|
|
||||||
using System.Threading;
|
|
||||||
using DisruptorUnity3d;
|
|
||||||
using NLog;
|
|
||||||
using Ragon.Common;
|
|
||||||
|
|
||||||
namespace Ragon.Core
|
|
||||||
{
|
|
||||||
public class RoomThread : IDisposable
|
|
||||||
{
|
|
||||||
private readonly RoomManager _roomManager;
|
|
||||||
private readonly Dictionary<uint, Room> _socketByRooms;
|
|
||||||
private readonly Thread _thread;
|
|
||||||
private readonly Stopwatch _timer;
|
|
||||||
private readonly ILogger _logger = LogManager.GetCurrentClassLogger();
|
|
||||||
private readonly float _deltaTime = 0.0f;
|
|
||||||
private RingBuffer<Event> _receiveBuffer = new RingBuffer<Event>(8192 + 8192);
|
|
||||||
private RingBuffer<Event> _sendBuffer = new RingBuffer<Event>(8192 + 8192);
|
|
||||||
|
|
||||||
public Configuration Configuration { get; private set; }
|
|
||||||
public bool ReadOutEvent(out Event evnt) => _sendBuffer.TryDequeue(out evnt);
|
|
||||||
public void WriteOutEvent(Event evnt) => _sendBuffer.Enqueue(evnt);
|
|
||||||
|
|
||||||
public bool ReadIntEvent(out Event evnt) => _receiveBuffer.TryDequeue(out evnt);
|
|
||||||
public void WriteInEvent(Event evnt) => _receiveBuffer.Enqueue(evnt);
|
|
||||||
|
|
||||||
public RoomThread(PluginFactory factory, Configuration configuration)
|
|
||||||
{
|
|
||||||
_thread = new Thread(Execute);
|
|
||||||
_thread.IsBackground = true;
|
|
||||||
_timer = new Stopwatch();
|
|
||||||
_socketByRooms = new Dictionary<uint, Room>();
|
|
||||||
|
|
||||||
Configuration = configuration;
|
|
||||||
_deltaTime = 1000.0f / Configuration.Server.TickRate;
|
|
||||||
|
|
||||||
_roomManager = new RoomManager(this, factory);
|
|
||||||
_roomManager.OnJoined += (tuple) => _socketByRooms.Add(tuple.Item1, tuple.Item2);
|
|
||||||
_roomManager.OnLeaved += (tuple) => _socketByRooms.Remove(tuple.Item1);
|
|
||||||
}
|
|
||||||
|
|
||||||
public void Start()
|
|
||||||
{
|
|
||||||
_timer.Start();
|
|
||||||
_thread.Start();
|
|
||||||
}
|
|
||||||
|
|
||||||
public void Stop()
|
|
||||||
{
|
|
||||||
_thread.Interrupt();
|
|
||||||
}
|
|
||||||
|
|
||||||
private void Execute()
|
|
||||||
{
|
|
||||||
while (true)
|
|
||||||
{
|
|
||||||
while (_receiveBuffer.TryDequeue(out var evnt))
|
|
||||||
{
|
|
||||||
if (evnt.Type == EventType.DISCONNECTED || evnt.Type == EventType.TIMEOUT)
|
|
||||||
{
|
|
||||||
if (_socketByRooms.ContainsKey(evnt.PeerId))
|
|
||||||
{
|
|
||||||
_roomManager.Disconnected(evnt.PeerId);
|
|
||||||
_socketByRooms.Remove(evnt.PeerId);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (evnt.Type == EventType.DATA)
|
|
||||||
{
|
|
||||||
var data = new ReadOnlySpan<byte>(evnt.Data);
|
|
||||||
var operationData = data.Slice(0, 2);
|
|
||||||
var operation = (RagonOperation) RagonHeader.ReadUShort(ref operationData);
|
|
||||||
if (_socketByRooms.TryGetValue(evnt.PeerId, out var room))
|
|
||||||
{
|
|
||||||
try
|
|
||||||
{
|
|
||||||
room.ProcessEvent(operation, evnt.PeerId, data);
|
|
||||||
}
|
|
||||||
catch (Exception exception)
|
|
||||||
{
|
|
||||||
_logger.Error(exception);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
var payload = data.Slice(2, data.Length - 2);
|
|
||||||
_roomManager.ProcessEvent(operation, evnt.PeerId, payload);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
var elapsedMilliseconds = _timer.ElapsedMilliseconds;
|
|
||||||
if (elapsedMilliseconds > _deltaTime)
|
|
||||||
{
|
|
||||||
_roomManager.Tick(elapsedMilliseconds / 1000.0f);
|
|
||||||
_timer.Restart();
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
Thread.Sleep(15);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public void Dispose()
|
|
||||||
{
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,153 +0,0 @@
|
|||||||
using System;
|
|
||||||
using System.Collections.Generic;
|
|
||||||
using System.Diagnostics;
|
|
||||||
using System.Threading;
|
|
||||||
using DisruptorUnity3d;
|
|
||||||
using ENet;
|
|
||||||
using NLog;
|
|
||||||
|
|
||||||
namespace Ragon.Core
|
|
||||||
{
|
|
||||||
public enum Status
|
|
||||||
{
|
|
||||||
Stopped,
|
|
||||||
Listening,
|
|
||||||
Disconnecting,
|
|
||||||
Connecting,
|
|
||||||
Assigning,
|
|
||||||
Connected
|
|
||||||
}
|
|
||||||
|
|
||||||
public class ENetServer : IDisposable
|
|
||||||
{
|
|
||||||
public Status Status { get; private set; }
|
|
||||||
|
|
||||||
private ILogger _logger = LogManager.GetCurrentClassLogger();
|
|
||||||
private Thread _thread;
|
|
||||||
private Host _host;
|
|
||||||
private Address _address;
|
|
||||||
private ENet.Event _netEvent;
|
|
||||||
private Peer[] _peers;
|
|
||||||
|
|
||||||
private RingBuffer<Event> _receiveBuffer;
|
|
||||||
private RingBuffer<Event> _sendBuffer;
|
|
||||||
|
|
||||||
public void WriteEvent(Event evnt) => _sendBuffer.Enqueue(evnt);
|
|
||||||
public bool ReadEvent(out Event evnt) => _receiveBuffer.TryDequeue(out evnt);
|
|
||||||
|
|
||||||
public void Start(ushort port)
|
|
||||||
{
|
|
||||||
Library.Initialize();
|
|
||||||
|
|
||||||
_address = default;
|
|
||||||
_address.Port = port;
|
|
||||||
|
|
||||||
_host = new Host();
|
|
||||||
_host.Create(_address, 4095, 2, 0, 0, 1024 * 1024);
|
|
||||||
|
|
||||||
_peers = new Peer[4095];
|
|
||||||
_sendBuffer = new RingBuffer<Event>(8192 + 8192);
|
|
||||||
_receiveBuffer = new RingBuffer<Event>(8192 + 8192);
|
|
||||||
|
|
||||||
Status = Status.Listening;
|
|
||||||
|
|
||||||
_thread = new Thread(Execute);
|
|
||||||
_thread.Name = "NetworkThread";
|
|
||||||
_thread.Start();
|
|
||||||
_logger.Info($"ENet Server Started at port {port}");
|
|
||||||
}
|
|
||||||
|
|
||||||
private void Execute()
|
|
||||||
{
|
|
||||||
while (true)
|
|
||||||
{
|
|
||||||
while (_sendBuffer.TryDequeue(out var data))
|
|
||||||
{
|
|
||||||
if (data.Type == EventType.DATA)
|
|
||||||
{
|
|
||||||
var newPacket = new Packet();
|
|
||||||
var packetFlags = PacketFlags.Instant;
|
|
||||||
byte channel = 1;
|
|
||||||
|
|
||||||
if (data.Delivery == DeliveryType.Reliable)
|
|
||||||
{
|
|
||||||
packetFlags = PacketFlags.Reliable;
|
|
||||||
channel = 0;
|
|
||||||
}
|
|
||||||
else if (data.Delivery == DeliveryType.Unreliable)
|
|
||||||
{
|
|
||||||
channel = 1;
|
|
||||||
packetFlags = PacketFlags.Instant;
|
|
||||||
}
|
|
||||||
|
|
||||||
newPacket.Create(data.Data, data.Data.Length, packetFlags);
|
|
||||||
_peers[data.PeerId].Send(channel, ref newPacket);
|
|
||||||
}
|
|
||||||
else if (data.Type == EventType.DISCONNECTED)
|
|
||||||
{
|
|
||||||
_peers[data.PeerId].DisconnectNow(0);
|
|
||||||
_receiveBuffer.Enqueue(data);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
bool polled = false;
|
|
||||||
while (!polled)
|
|
||||||
{
|
|
||||||
if (_host.CheckEvents(out _netEvent) <= 0)
|
|
||||||
{
|
|
||||||
if (_host.Service(16, out _netEvent) <= 0)
|
|
||||||
break;
|
|
||||||
|
|
||||||
polled = true;
|
|
||||||
}
|
|
||||||
|
|
||||||
switch (_netEvent.Type)
|
|
||||||
{
|
|
||||||
case ENet.EventType.None:
|
|
||||||
Console.WriteLine("None event");
|
|
||||||
break;
|
|
||||||
|
|
||||||
case ENet.EventType.Connect:
|
|
||||||
{
|
|
||||||
var @event = new Event {PeerId = _netEvent.Peer.ID, Type = EventType.CONNECTED};
|
|
||||||
_peers[_netEvent.Peer.ID] = _netEvent.Peer;
|
|
||||||
_receiveBuffer.Enqueue(@event);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case ENet.EventType.Disconnect:
|
|
||||||
{
|
|
||||||
var @event = new Event {PeerId = _netEvent.Peer.ID, Type = EventType.DISCONNECTED};
|
|
||||||
_receiveBuffer.Enqueue(@event);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case ENet.EventType.Timeout:
|
|
||||||
{
|
|
||||||
var @event = new Event {PeerId = _netEvent.Peer.ID, Type = EventType.TIMEOUT};
|
|
||||||
_receiveBuffer.Enqueue(@event);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
case ENet.EventType.Receive:
|
|
||||||
{
|
|
||||||
var data = new byte[_netEvent.Packet.Length];
|
|
||||||
|
|
||||||
_netEvent.Packet.CopyTo(data);
|
|
||||||
_netEvent.Packet.Dispose();
|
|
||||||
|
|
||||||
var @event = new Event {PeerId = _netEvent.Peer.ID, Type = EventType.DATA, Data = data };
|
|
||||||
|
|
||||||
_receiveBuffer.Enqueue(@event);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public void Dispose()
|
|
||||||
{
|
|
||||||
Library.Deinitialize();
|
|
||||||
|
|
||||||
_host?.Dispose();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,60 +0,0 @@
|
|||||||
using System;
|
|
||||||
using System.Net;
|
|
||||||
using System.Net.WebSockets;
|
|
||||||
using System.Threading;
|
|
||||||
using DisruptorUnity3d;
|
|
||||||
using NLog;
|
|
||||||
|
|
||||||
namespace Ragon.Core;
|
|
||||||
|
|
||||||
public class WebsocketServer : IDisposable
|
|
||||||
{
|
|
||||||
private HttpListener _httpListener;
|
|
||||||
private ILogger _logger = LogManager.GetCurrentClassLogger();
|
|
||||||
private Thread _thread;
|
|
||||||
private ENet.Event _netEvent;
|
|
||||||
|
|
||||||
private RingBuffer<Event> _receiveBuffer;
|
|
||||||
private RingBuffer<Event> _sendBuffer;
|
|
||||||
|
|
||||||
public void WriteEvent(Event evnt) => _sendBuffer.Enqueue(evnt);
|
|
||||||
public bool ReadEvent(out Event evnt) => _receiveBuffer.TryDequeue(out evnt);
|
|
||||||
|
|
||||||
public void Start(ushort port)
|
|
||||||
{
|
|
||||||
// _httpListener = new HttpListener();
|
|
||||||
// _httpListener.Prefixes.Add("http://localhost/");
|
|
||||||
// _httpListener.Start();
|
|
||||||
//
|
|
||||||
// _thread = new Thread(Execute);
|
|
||||||
// _thread.Name = "NetworkThread";
|
|
||||||
// _thread.Start();
|
|
||||||
// _logger.Info($"Socket Server Started at port {port}");
|
|
||||||
}
|
|
||||||
|
|
||||||
public void Execute()
|
|
||||||
{
|
|
||||||
|
|
||||||
}
|
|
||||||
|
|
||||||
public async void ExecuteAsync()
|
|
||||||
{
|
|
||||||
// while (true)
|
|
||||||
// {
|
|
||||||
// HttpListenerContext context = await _httpListener.GetContextAsync();
|
|
||||||
// if (context.Request.IsWebSocketRequest)
|
|
||||||
// {
|
|
||||||
// HttpListenerWebSocketContext webSocketContext = await context.AcceptWebSocketAsync(null);
|
|
||||||
// WebSocket webSocket = webSocketContext.WebSocket;
|
|
||||||
// while (webSocket.State == WebSocketState.Open)
|
|
||||||
// {
|
|
||||||
// await webSocket.SendAsync(... );
|
|
||||||
// }
|
|
||||||
// }
|
|
||||||
// }
|
|
||||||
}
|
|
||||||
|
|
||||||
public void Dispose()
|
|
||||||
{
|
|
||||||
}
|
|
||||||
}
|
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user