Improved Partial write system
- Partial writes now only incur one alignment check per write - Upgraded WriteByte method to support rudimentary boundary checks
This commit is contained in:
parent
54f0df852c
commit
80399f387d
@ -10,6 +10,19 @@ namespace MLAPI.NetworkingManagerComponents.Binary
|
|||||||
{
|
{
|
||||||
public sealed class BitWriter : IDisposable
|
public sealed class BitWriter : IDisposable
|
||||||
{
|
{
|
||||||
|
private struct Partial
|
||||||
|
{
|
||||||
|
public byte value;
|
||||||
|
public byte count;
|
||||||
|
public static readonly FieldInfo value_info = typeof(Partial).GetField("value");
|
||||||
|
public static readonly FieldInfo count_info = typeof(Partial).GetField("count");
|
||||||
|
|
||||||
|
public Partial(byte value, byte count)
|
||||||
|
{
|
||||||
|
this.value = value;
|
||||||
|
this.count = count;
|
||||||
|
}
|
||||||
|
}
|
||||||
private static readonly Queue<List<object>> listPool = new Queue<List<object>>();
|
private static readonly Queue<List<object>> listPool = new Queue<List<object>>();
|
||||||
|
|
||||||
private static readonly float[] holder_f = new float[1];
|
private static readonly float[] holder_f = new float[1];
|
||||||
@ -105,10 +118,7 @@ namespace MLAPI.NetworkingManagerComponents.Binary
|
|||||||
public void WriteShortArray(short[] s, bool known = false) => PushArray(s, known);
|
public void WriteShortArray(short[] s, bool known = false) => PushArray(s, known);
|
||||||
public void WriteIntArray(int[] i, bool known = false) => PushArray(i, known);
|
public void WriteIntArray(int[] i, bool known = false) => PushArray(i, known);
|
||||||
public void WriteLongArray(long[] l, bool known = false) => PushArray(l, known);
|
public void WriteLongArray(long[] l, bool known = false) => PushArray(l, known);
|
||||||
public void WriteBits(byte value, int bits)
|
public void WriteBits(byte value, int bits) => Push(new Partial(ReadNBits(value, 0, bits % 8), (byte)(bits%8))); // Suggestion: store (bits % 8) result?
|
||||||
{
|
|
||||||
for (int i = 0; i < bits; ++i) WriteBool((value & (1 << i)) != 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
private void PushArray<T>(T[] t, bool knownSize = false)
|
private void PushArray<T>(T[] t, bool knownSize = false)
|
||||||
{
|
{
|
||||||
@ -197,6 +207,14 @@ namespace MLAPI.NetworkingManagerComponents.Binary
|
|||||||
foreach (var element in array)
|
foreach (var element in array)
|
||||||
Serialize(element, writeTo, ref bitOffset, ref isAligned);
|
Serialize(element, writeTo, ref bitOffset, ref isAligned);
|
||||||
}
|
}
|
||||||
|
else if (type == typeof(Partial))
|
||||||
|
{
|
||||||
|
byte count;
|
||||||
|
WriteByte(writeTo, (byte)Partial.value_info.GetValue(t), bitOffset, isAligned, count = (byte)Partial.count_info.GetValue(t));
|
||||||
|
bitOffset += count;
|
||||||
|
isAligned = bitOffset % 8 == 0;
|
||||||
|
return;
|
||||||
|
}
|
||||||
else if (IsSupportedType(type))
|
else if (IsSupportedType(type))
|
||||||
{
|
{
|
||||||
long offset = t is bool ? 1 : BytesToRead(t) * 8;
|
long offset = t is bool ? 1 : BytesToRead(t) * 8;
|
||||||
@ -230,7 +248,7 @@ namespace MLAPI.NetworkingManagerComponents.Binary
|
|||||||
|
|
||||||
// Since floating point flag bits are seemingly the highest bytes of the floating point values
|
// Since floating point flag bits are seemingly the highest bytes of the floating point values
|
||||||
// and even very small values have them, we swap the endianness in the hopes of reducing the size
|
// and even very small values have them, we swap the endianness in the hopes of reducing the size
|
||||||
if(size) Serialize(BinaryHelpers.SwapEndian((uint)result_holder.GetValue(0)), writeTo, ref bitOffset, ref isAligned);
|
if (size) Serialize(BinaryHelpers.SwapEndian((uint)result_holder.GetValue(0)), writeTo, ref bitOffset, ref isAligned);
|
||||||
else Serialize(BinaryHelpers.SwapEndian((ulong)result_holder.GetValue(0)), writeTo, ref bitOffset, ref isAligned);
|
else Serialize(BinaryHelpers.SwapEndian((ulong)result_holder.GetValue(0)), writeTo, ref bitOffset, ref isAligned);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -338,7 +356,7 @@ namespace MLAPI.NetworkingManagerComponents.Binary
|
|||||||
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, ulong value, long index, bool isAligned) => WriteByte(b, (byte)value, index, isAligned);
|
private static void WriteByte(byte[] b, ulong value, long index, bool isAligned) => WriteByte(b, (byte)value, index, isAligned);
|
||||||
private static void WriteByte(byte[] b, byte value, long index, bool isAligned)
|
private static void WriteByte(byte[] b, byte value, long index, bool isAligned, byte bits = 8)
|
||||||
{
|
{
|
||||||
if (isAligned) b[index / 8] = value;
|
if (isAligned) b[index / 8] = value;
|
||||||
else
|
else
|
||||||
@ -348,7 +366,7 @@ namespace MLAPI.NetworkingManagerComponents.Binary
|
|||||||
byte upper_mask = (byte)(0xFF << shift);
|
byte upper_mask = (byte)(0xFF << shift);
|
||||||
|
|
||||||
b[byteIndex] = (byte)((b[byteIndex] & (byte)~upper_mask) | (value << shift));
|
b[byteIndex] = (byte)((b[byteIndex] & (byte)~upper_mask) | (value << shift));
|
||||||
b[byteIndex + 1] = (byte)((b[byteIndex + 1] & upper_mask) | (value >> (8 - shift)));
|
if((8-shift)<bits) b[byteIndex + 1] = (byte)((b[byteIndex + 1] & upper_mask) | (value >> (8 - shift)));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
private static void WriteDynamic(byte[] b, int value, int byteCount, long index, bool isAligned)
|
private static void WriteDynamic(byte[] b, int value, int byteCount, long index, bool isAligned)
|
||||||
|
Loading…
x
Reference in New Issue
Block a user