Implemented floating point value serialization
Implemented floating point value deserialization Separated Elliptic Curve Point class to its own file
This commit is contained in:
parent
de2a53c1cf
commit
1da1b7878d
@ -14,6 +14,9 @@ namespace Client
|
|||||||
// Collects reusable
|
// Collects reusable
|
||||||
private static readonly List<WeakReference<object[]>> expired = new List<WeakReference<object[]>>();
|
private static readonly List<WeakReference<object[]>> expired = new List<WeakReference<object[]>>();
|
||||||
|
|
||||||
|
private static readonly byte[] holder = new byte[8];
|
||||||
|
private static readonly float[] holder_f = new float[1];
|
||||||
|
private static readonly double[] holder_d = new double[1];
|
||||||
private static readonly List<Type> supportedTypes = new List<Type>()
|
private static readonly List<Type> supportedTypes = new List<Type>()
|
||||||
{
|
{
|
||||||
typeof(bool),
|
typeof(bool),
|
||||||
@ -93,10 +96,11 @@ namespace Client
|
|||||||
private static void Serialize<T>(T t, byte[] writeTo, ref long bitOffset)
|
private static void Serialize<T>(T t, byte[] writeTo, ref long bitOffset)
|
||||||
{
|
{
|
||||||
Type type = t.GetType();
|
Type type = t.GetType();
|
||||||
|
bool size = false;
|
||||||
if (type.IsArray)
|
if (type.IsArray)
|
||||||
{
|
{
|
||||||
var array = t as Array;
|
var array = t as Array;
|
||||||
Serialize(array.Length, writeTo, ref bitOffset);
|
Serialize((short)array.Length, writeTo, ref bitOffset);
|
||||||
foreach (var element in array)
|
foreach (var element in array)
|
||||||
Serialize(element, writeTo, ref bitOffset);
|
Serialize(element, writeTo, ref bitOffset);
|
||||||
}
|
}
|
||||||
@ -111,9 +115,25 @@ namespace Client
|
|||||||
WriteDynamic(writeTo, dec_hi.GetValue(t), 4, bitOffset + 64);
|
WriteDynamic(writeTo, dec_hi.GetValue(t), 4, bitOffset + 64);
|
||||||
WriteDynamic(writeTo, dec_flags.GetValue(t), 4, bitOffset + 96);
|
WriteDynamic(writeTo, dec_flags.GetValue(t), 4, bitOffset + 96);
|
||||||
}
|
}
|
||||||
else if(type == typeof(float))
|
else if((size = type == typeof(float)) || type == typeof(double))
|
||||||
{
|
{
|
||||||
|
int bytes = size ? 4 : 8;
|
||||||
|
Array type_holder = size ? holder_f as Array : holder_d as Array; // Fetch the preallocated array
|
||||||
|
lock (type_holder)
|
||||||
|
lock (holder)
|
||||||
|
{
|
||||||
|
type_holder.SetValue(t, 0); // Insert the value to convert into the preallocated holder array
|
||||||
|
Buffer.BlockCopy(type_holder, 0, holder, 0, bytes); // Perform an internal copy to the byte-based holder
|
||||||
|
for (int i = 0; i < bytes; ++i)
|
||||||
|
WriteByte(writeTo, holder[i], bitOffset + (i * 8)); // Write the converted value to the output array
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
dynamic value = t;
|
||||||
|
int type_size = Marshal.SizeOf(typeof(T));
|
||||||
|
for (int i = 0; i < type_size; ++i)
|
||||||
|
WriteByte(writeTo, (byte)(value >> (int)(i * 8)), bitOffset + (i * 8));
|
||||||
}
|
}
|
||||||
bitOffset += offset;
|
bitOffset += offset;
|
||||||
}
|
}
|
||||||
@ -126,25 +146,25 @@ namespace Client
|
|||||||
if (type.IsArray)
|
if (type.IsArray)
|
||||||
{
|
{
|
||||||
Type elementType = type.GetElementType();
|
Type elementType = type.GetElementType();
|
||||||
long allocSize = GetBitAllocation(elementType);
|
|
||||||
var array = t as Array;
|
|
||||||
|
|
||||||
count += 2; // Int16 array size. Arrays shouldn't be syncing more than 65k elements
|
count += 16; // Int16 array size. Arrays shouldn't be syncing more than 65k elements
|
||||||
|
foreach (var element in t as Array)
|
||||||
if (allocSize != 0) // The array contents is known: compute the data size
|
count += GetBitCount(element);
|
||||||
count += allocSize * array.Length;
|
}
|
||||||
else // Unknown array contents type: iteratively assess serialization size
|
else if (IsSupportedType(type))
|
||||||
foreach (var element in t as Array)
|
{
|
||||||
count += GetBitCount(element);
|
long ba = GetBitAllocation(type);
|
||||||
|
if (ba == 0) count += Encoding.UTF8.GetByteCount(t as string);
|
||||||
|
else if (t is bool || t is float || t is double || t is decimal) count += ba;
|
||||||
|
else count += BytesToRead(t, Marshal.SizeOf(t)) * 8;
|
||||||
}
|
}
|
||||||
else if(IsSupportedType(type)) count += GetBitAllocation(type);
|
|
||||||
//else
|
//else
|
||||||
// Debug.LogWarning("MLAPI: The type \"" + b.GetType() + "\" is not supported by the Binary Serializer. It will be ignored");
|
// Debug.LogWarning("MLAPI: The type \"" + b.GetType() + "\" is not supported by the Binary Serializer. It will be ignored");
|
||||||
return count;
|
return count;
|
||||||
}
|
}
|
||||||
|
|
||||||
private static void WriteBit(byte[] b, bool bit, long index)
|
private static void WriteBit(byte[] b, bool bit, long index)
|
||||||
=> b[index / 8] = (byte)((b[index / 8] & (1 << (int)(index % 8))) | (bit ? 1 << (int)(index % 8) : 0));
|
=> b[index / 8] = (byte)((b[index / 8] & ~(1 << (int)(index % 8))) | (bit ? 1 << (int)(index % 8) : 0));
|
||||||
private static void WriteByte(byte[] b, byte value, long index)
|
private static void WriteByte(byte[] b, byte value, long index)
|
||||||
{
|
{
|
||||||
int byteIndex = (int)(index / 8);
|
int byteIndex = (int)(index / 8);
|
||||||
@ -154,7 +174,7 @@ namespace Client
|
|||||||
|
|
||||||
b[byteIndex] = (byte)((b[byteIndex] & lower_mask) | (value << shift));
|
b[byteIndex] = (byte)((b[byteIndex] & lower_mask) | (value << shift));
|
||||||
if(shift != 0 && byteIndex + 1 < b.Length)
|
if(shift != 0 && byteIndex + 1 < b.Length)
|
||||||
b[byteIndex + 1] = (byte)((b[byteIndex + 1] & upper_mask) | (value << (8 - shift)));
|
b[byteIndex + 1] = (byte)((b[byteIndex + 1] & upper_mask) | (value >> (8 - shift)));
|
||||||
}
|
}
|
||||||
private static void WriteDynamic(byte[] b, dynamic value, int byteCount, long index)
|
private static void WriteDynamic(byte[] b, dynamic value, int byteCount, long index)
|
||||||
{
|
{
|
||||||
@ -162,6 +182,14 @@ namespace Client
|
|||||||
WriteByte(b, (byte)((value >> (8 * i)) & 0xFF), index + (8 * i));
|
WriteByte(b, (byte)((value >> (8 * i)) & 0xFF), index + (8 * i));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private static int BytesToRead(dynamic integer, int maxBytes)
|
||||||
|
{
|
||||||
|
for (int i = 0; i < maxBytes; ++i)
|
||||||
|
if ((integer >> (8 * i)) & 0xFF <= 127)
|
||||||
|
return i + 1;
|
||||||
|
return maxBytes;
|
||||||
|
}
|
||||||
|
|
||||||
// Supported datatypes for serialization
|
// Supported datatypes for serialization
|
||||||
private static bool IsSupportedType(Type t) => supportedTypes.Contains(t);
|
private static bool IsSupportedType(Type t) => supportedTypes.Contains(t);
|
||||||
|
|
||||||
|
68
Client/BinaryDistributor.cs
Normal file
68
Client/BinaryDistributor.cs
Normal file
@ -0,0 +1,68 @@
|
|||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Runtime.InteropServices;
|
||||||
|
using System.Text;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
|
||||||
|
namespace Client
|
||||||
|
{
|
||||||
|
public class BinaryDistributor
|
||||||
|
{
|
||||||
|
private static readonly byte[] holder = new byte[8];
|
||||||
|
private static readonly float[] holder_f = new float[1];
|
||||||
|
private static readonly double[] holder_d = new double[1];
|
||||||
|
|
||||||
|
private readonly byte[] readFrom;
|
||||||
|
private long bitCount = 0;
|
||||||
|
public BinaryDistributor(byte[] readFrom) => this.readFrom = readFrom;
|
||||||
|
|
||||||
|
public bool ReadBit()
|
||||||
|
{
|
||||||
|
bool result = (readFrom[bitCount / 8] & (byte)(1 << (int)(bitCount % 8))) != 0;
|
||||||
|
++bitCount;
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
public byte ReadByte()
|
||||||
|
{
|
||||||
|
int shift = (int)(bitCount % 8);
|
||||||
|
int index = (int)(bitCount / 8);
|
||||||
|
byte lower_mask = (byte)(0xFF << shift);
|
||||||
|
byte upper_mask = (byte)~lower_mask;
|
||||||
|
byte result = (byte)(((readFrom[index] & lower_mask) >> shift) | (shift == 0 ? 0 : (readFrom[index + 1] & upper_mask) << (8 - shift)));
|
||||||
|
bitCount += 8;
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
public float ReadFloat() => ReadFloating<float>();
|
||||||
|
public double ReadDouble() => ReadFloating<double>();
|
||||||
|
public float[] ReadFloatArray() => ReadFloatingArray<float>();
|
||||||
|
public double[] ReadDoubleArray() => ReadFloatingArray<double>();
|
||||||
|
|
||||||
|
private T[] ReadFloatingArray<T>()
|
||||||
|
{
|
||||||
|
short size = (short)(ReadByte() | (ReadByte() << 8));
|
||||||
|
T[] result = new T[size];
|
||||||
|
for (short s = 0; s < size; ++s)
|
||||||
|
result[s] = ReadFloating<T>();
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
|
||||||
|
private T ReadFloating<T>()
|
||||||
|
{
|
||||||
|
int size = Marshal.SizeOf(typeof(T));
|
||||||
|
Array type_holder = size == 4 ? holder_f as Array: holder_d as Array;
|
||||||
|
T result;
|
||||||
|
lock(type_holder)
|
||||||
|
lock (holder)
|
||||||
|
{
|
||||||
|
for (int i = 0; i < size; ++i)
|
||||||
|
holder.SetValue(ReadByte(), i);
|
||||||
|
Buffer.BlockCopy(holder, 0, type_holder, 0, size);
|
||||||
|
result = (T) type_holder.GetValue(0);
|
||||||
|
}
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
@ -44,6 +44,7 @@
|
|||||||
</ItemGroup>
|
</ItemGroup>
|
||||||
<ItemGroup>
|
<ItemGroup>
|
||||||
<Compile Include="BinaryCollector.cs" />
|
<Compile Include="BinaryCollector.cs" />
|
||||||
|
<Compile Include="BinaryDistributor.cs" />
|
||||||
<Compile Include="ConsoleForms\CancellationPipe.cs" />
|
<Compile Include="ConsoleForms\CancellationPipe.cs" />
|
||||||
<Compile Include="ConsoleForms\ConsoleController.cs" />
|
<Compile Include="ConsoleForms\ConsoleController.cs" />
|
||||||
<Compile Include="ConsoleForms\Context.cs" />
|
<Compile Include="ConsoleForms\Context.cs" />
|
||||||
|
@ -18,14 +18,20 @@ namespace ConsoleForms
|
|||||||
// Set up timestamps in debug output
|
// Set up timestamps in debug output
|
||||||
DebugStream = new TimeStampWriter(DebugStream, "HH:mm:ss.fff");
|
DebugStream = new TimeStampWriter(DebugStream, "HH:mm:ss.fff");
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
byte[] serialized;
|
byte[] serialized;
|
||||||
|
|
||||||
using (BinaryCollector collector = new BinaryCollector(1))
|
using (BinaryCollector collector = new BinaryCollector(2))
|
||||||
{
|
{
|
||||||
collector.Push(5f);
|
collector.Push(true);
|
||||||
|
collector.Push(new double[] { 6.0, 5.0 });
|
||||||
serialized = collector.ToArray();
|
serialized = collector.ToArray();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
BinaryDistributor bd = new BinaryDistributor(serialized);
|
||||||
|
bool bit = bd.ReadBit();
|
||||||
|
double[] result = bd.ReadDoubleArray();
|
||||||
|
|
||||||
Padding p = new AbsolutePadding(2, 2, 1, 1);
|
Padding p = new AbsolutePadding(2, 2, 1, 1);
|
||||||
|
|
||||||
|
@ -70,6 +70,7 @@
|
|||||||
<Compile Include="Cryptography\KeyExchange\DiffieHellman.cs" />
|
<Compile Include="Cryptography\KeyExchange\DiffieHellman.cs" />
|
||||||
<Compile Include="Cryptography\KeyExchange\EllipticDiffieHellman.cs" />
|
<Compile Include="Cryptography\KeyExchange\EllipticDiffieHellman.cs" />
|
||||||
<Compile Include="Cryptography\KeyExchange\IKeyExchange.cs" />
|
<Compile Include="Cryptography\KeyExchange\IKeyExchange.cs" />
|
||||||
|
<Compile Include="Cryptography\Point.cs" />
|
||||||
<Compile Include="KDF.cs" />
|
<Compile Include="KDF.cs" />
|
||||||
<Compile Include="NetClient.cs" />
|
<Compile Include="NetClient.cs" />
|
||||||
<Compile Include="NetServer.cs" />
|
<Compile Include="NetServer.cs" />
|
||||||
|
@ -1,4 +1,6 @@
|
|||||||
using System;
|
//#define SAFE_MATH
|
||||||
|
|
||||||
|
using System;
|
||||||
using System.Collections.Generic;
|
using System.Collections.Generic;
|
||||||
using System.Linq;
|
using System.Linq;
|
||||||
using System.Numerics;
|
using System.Numerics;
|
||||||
@ -8,24 +10,6 @@ using Tofvesson.Crypto;
|
|||||||
|
|
||||||
namespace Common.Cryptography
|
namespace Common.Cryptography
|
||||||
{
|
{
|
||||||
public class Point
|
|
||||||
{
|
|
||||||
public static readonly Point POINT_AT_INFINITY = new Point();
|
|
||||||
public BigInteger X { get; private set; }
|
|
||||||
public BigInteger Y { get; private set; }
|
|
||||||
private bool pai = false;
|
|
||||||
public Point(BigInteger x, BigInteger y)
|
|
||||||
{
|
|
||||||
X = x;
|
|
||||||
Y = y;
|
|
||||||
}
|
|
||||||
private Point() { pai = true; } // Accessing corrdinates causes undocumented behaviour
|
|
||||||
public override string ToString()
|
|
||||||
{
|
|
||||||
return pai ? "(POINT_AT_INFINITY)" : "(" + X + ", " + Y + ")";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
public class EllipticCurve
|
public class EllipticCurve
|
||||||
{
|
{
|
||||||
public enum CurveType { Weierstrass, Montgomery }
|
public enum CurveType { Weierstrass, Montgomery }
|
||||||
|
27
Common/Cryptography/Point.cs
Normal file
27
Common/Cryptography/Point.cs
Normal file
@ -0,0 +1,27 @@
|
|||||||
|
using System;
|
||||||
|
using System.Collections.Generic;
|
||||||
|
using System.Linq;
|
||||||
|
using System.Numerics;
|
||||||
|
using System.Text;
|
||||||
|
using System.Threading.Tasks;
|
||||||
|
|
||||||
|
namespace Common.Cryptography
|
||||||
|
{
|
||||||
|
public class Point
|
||||||
|
{
|
||||||
|
public static readonly Point POINT_AT_INFINITY = new Point();
|
||||||
|
public BigInteger X { get; private set; }
|
||||||
|
public BigInteger Y { get; private set; }
|
||||||
|
private bool pai = false;
|
||||||
|
public Point(BigInteger x, BigInteger y)
|
||||||
|
{
|
||||||
|
X = x;
|
||||||
|
Y = y;
|
||||||
|
}
|
||||||
|
private Point() { pai = true; } // Accessing corrdinates causes undocumented behaviour
|
||||||
|
public override string ToString()
|
||||||
|
{
|
||||||
|
return pai ? "(POINT_AT_INFINITY)" : "(" + X + ", " + Y + ")";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user