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:
Gabriel Tofvesson 2018-04-20 13:23:23 +02:00
parent 54f0df852c
commit 80399f387d

View File

@ -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)