refactor bitfield io
This commit is contained in:
90
include/sp/io/BitBuffer.h
Normal file
90
include/sp/io/BitBuffer.h
Normal file
@@ -0,0 +1,90 @@
|
||||
#pragma once
|
||||
|
||||
#include <sp/common/DataBuffer.h>
|
||||
#include <sp/protocol/BitField.h>
|
||||
|
||||
namespace sp {
|
||||
|
||||
// TODO: flush if offset exceeds 64
|
||||
class BitBuffer {
|
||||
private:
|
||||
using Data = std::uint64_t;
|
||||
|
||||
private:
|
||||
DataBuffer& m_Buffer;
|
||||
Data m_Data;
|
||||
std::size_t m_Offset;
|
||||
bool m_WasBitField;
|
||||
|
||||
public:
|
||||
BitBuffer(DataBuffer& a_Buffer) : m_Buffer(a_Buffer), m_Data(0), m_Offset(0), m_WasBitField(false) {}
|
||||
|
||||
void UpdateWrite(bool a_IsBitField) {
|
||||
if (m_Offset == 0)
|
||||
return;
|
||||
if ((m_Offset % 8 == 0) || (!a_IsBitField && m_WasBitField)) {
|
||||
Flush();
|
||||
}
|
||||
m_WasBitField = a_IsBitField;
|
||||
}
|
||||
|
||||
void UpdateRead(bool a_IsBitField) {
|
||||
if (m_Offset == 0)
|
||||
return;
|
||||
if ((m_Offset % 8 == 0) || (!a_IsBitField && m_WasBitField)) {
|
||||
MoveReadOffset();
|
||||
}
|
||||
m_WasBitField = a_IsBitField;
|
||||
}
|
||||
|
||||
template <typename T, std::size_t BitSize>
|
||||
void Append(T a_Data) {
|
||||
Data bin = static_cast<Data>(a_Data);
|
||||
bin &= ((1 << BitSize) - 1); // prevents overflow
|
||||
std::size_t pushCount = sizeof(Data) * 8 - m_Offset - BitSize;
|
||||
m_Data |= bin << pushCount;
|
||||
m_Offset += BitSize;
|
||||
}
|
||||
|
||||
template <typename T, std::size_t BitSize>
|
||||
void Read(T& a_Data) {
|
||||
std::size_t byteCount = GetByteCount(m_Offset + BitSize);
|
||||
constexpr Data dataMask = (1 << BitSize) - 1;
|
||||
m_Buffer.ReadSome(reinterpret_cast<std::uint8_t*>(&m_Data), byteCount);
|
||||
SwapBytes(reinterpret_cast<std::uint8_t*>(&m_Data), reinterpret_cast<std::uint8_t*>(&m_Data) + byteCount);
|
||||
m_Data >>= byteCount * 8 - m_Offset - BitSize;
|
||||
m_Data &= dataMask;
|
||||
a_Data = T(m_Data);
|
||||
m_Buffer.SetReadOffset(m_Buffer.GetReadOffset() - byteCount);
|
||||
m_Offset += BitSize;
|
||||
}
|
||||
|
||||
private:
|
||||
void Flush() {
|
||||
std::size_t byteCount = GetByteCount();
|
||||
m_Data >>= (sizeof(Data) - byteCount) * 8;
|
||||
SwapBytes(reinterpret_cast<std::uint8_t*>(&m_Data), reinterpret_cast<std::uint8_t*>(&m_Data) + byteCount);
|
||||
m_Buffer.WriteSome(reinterpret_cast<std::uint8_t*>(&m_Data), byteCount);
|
||||
m_Offset = 0;
|
||||
m_WasBitField = false;
|
||||
m_Data = 0;
|
||||
}
|
||||
|
||||
void MoveReadOffset() {
|
||||
std::size_t byteCount = GetByteCount();
|
||||
m_Buffer.SetReadOffset(m_Buffer.GetReadOffset() + byteCount);
|
||||
m_Offset = 0;
|
||||
m_WasBitField = false;
|
||||
m_Data = 0;
|
||||
}
|
||||
|
||||
std::size_t GetByteCount(std::size_t a_Offset = -1) const {
|
||||
if (a_Offset == static_cast<std::size_t>(-1))
|
||||
a_Offset = m_Offset;
|
||||
if (a_Offset <= 8)
|
||||
return 1;
|
||||
return (a_Offset - 1) / 8 + 1;
|
||||
}
|
||||
};
|
||||
|
||||
} // namespace sp
|
||||
@@ -1,78 +1,53 @@
|
||||
#pragma once
|
||||
|
||||
#include <boost/pfr.hpp>
|
||||
#include <sp/common/ByteSwapping.h>
|
||||
#include <sp/common/DataBuffer.h>
|
||||
#include <sp/protocol/BitField.h>
|
||||
#include <sp/io/BitBuffer.h>
|
||||
|
||||
namespace sp {
|
||||
namespace details {
|
||||
|
||||
template <typename T, std::size_t BitSize>
|
||||
void WriteField(DataBuffer& a_Buffer, const BitField<T, BitSize>& a_Data, BitBuffer& a_BitBuffer) {
|
||||
a_BitBuffer.Append<T, BitSize>(*a_Data);
|
||||
a_BitBuffer.UpdateWrite(true);
|
||||
}
|
||||
|
||||
template <typename T>
|
||||
void WriteBitField(DataBuffer& a_Buffer, std::uint64_t& a_DataRaw, std::size_t& a_Offset) {
|
||||
T filled = static_cast<T>(a_DataRaw);
|
||||
ToNetwork(filled);
|
||||
a_Buffer << filled;
|
||||
a_Offset = 0;
|
||||
a_DataRaw = 0;
|
||||
void WriteField(DataBuffer& a_Buffer, const T& a_Data, BitBuffer& a_BitBuffer) {
|
||||
a_Buffer << a_Data;
|
||||
a_BitBuffer.UpdateWrite(false);
|
||||
}
|
||||
|
||||
template <typename T, std::size_t BitSize>
|
||||
void WriteField(DataBuffer& a_Buffer, const BitField<T, BitSize>& a_Data, std::uint64_t& a_DataRaw, std::size_t& a_Offset) {
|
||||
T cut = *a_Data & ((1 << a_Data.GetBitSize()) - 1);
|
||||
std::size_t pushCount = sizeof(T) * 8 - a_Offset - a_Data.GetBitSize();
|
||||
a_DataRaw |= cut << pushCount;
|
||||
a_Offset += a_Data.GetBitSize();
|
||||
if (a_Offset == sizeof(T) * 8) {
|
||||
WriteBitField<T>(a_Buffer, a_DataRaw, a_Offset);
|
||||
}
|
||||
void ReadField(DataBuffer& a_Buffer, BitField<T, BitSize>& a_Data, BitBuffer& a_BitBuffer) {
|
||||
a_BitBuffer.Read<T, BitSize>(*a_Data);
|
||||
a_BitBuffer.UpdateRead(true);
|
||||
}
|
||||
|
||||
template <typename T>
|
||||
void WriteField(DataBuffer& a_Buffer, const T& a_Data, std::uint64_t& a_DataRaw, std::size_t& a_Offset) {
|
||||
T swapped = a_Data;
|
||||
ToNetwork(swapped);
|
||||
a_Buffer << swapped;
|
||||
}
|
||||
|
||||
template <typename T, std::size_t BitSize>
|
||||
void ReadField(DataBuffer& a_Buffer, BitField<T, BitSize>& a_Data, std::size_t& a_Offset) {
|
||||
a_Buffer >> *a_Data;
|
||||
FromNetwork(*a_Data);
|
||||
|
||||
*a_Data >>= sizeof(T) * 8 - a_Offset - a_Data.GetBitSize();
|
||||
*a_Data &= (1 << a_Data.GetBitSize()) - 1;
|
||||
|
||||
if (a_Offset != sizeof(T) * 8) {
|
||||
a_Buffer.SetReadOffset(a_Buffer.GetReadOffset() - sizeof(T));
|
||||
a_Offset += a_Data.GetBitSize();
|
||||
} else {
|
||||
a_Offset = 0;
|
||||
}
|
||||
}
|
||||
|
||||
template <typename T>
|
||||
void ReadField(DataBuffer& a_Buffer, T& a_Data, std::size_t& a_Offset) {
|
||||
void ReadField(DataBuffer& a_Buffer, T& a_Data, BitBuffer& a_BitBuffer) {
|
||||
a_Buffer >> a_Data;
|
||||
FromNetwork(a_Data);
|
||||
a_BitBuffer.UpdateRead(false);
|
||||
}
|
||||
|
||||
} // namespace details
|
||||
|
||||
template <typename TData>
|
||||
DataBuffer WriteMessage(const TData& a_MessageData) {
|
||||
DataBuffer buffer;
|
||||
std::size_t currentOffset = 0;
|
||||
std::uint64_t dataRaw = 0;
|
||||
boost::pfr::for_each_field(a_MessageData,
|
||||
[&buffer, &dataRaw, ¤tOffset](const auto& a_Field) { WriteField(buffer, a_Field, dataRaw, currentOffset); });
|
||||
BitBuffer bitBuffer(buffer);
|
||||
boost::pfr::for_each_field(
|
||||
a_MessageData, [&buffer, &bitBuffer](const auto& a_Field) { details::WriteField(buffer, a_Field, bitBuffer); });
|
||||
bitBuffer.UpdateWrite(false);
|
||||
return buffer;
|
||||
}
|
||||
|
||||
template <typename TData>
|
||||
void ReadMessage(DataBuffer& a_Buffer, TData& a_MessageData) {
|
||||
std::size_t currentOffset = 0;
|
||||
BitBuffer bitBuffer(a_Buffer);
|
||||
boost::pfr::for_each_field(
|
||||
a_MessageData, [&a_Buffer, ¤tOffset](auto& a_Field) { ReadField(a_Buffer, a_Field, currentOffset); });
|
||||
a_MessageData, [&a_Buffer, &bitBuffer](auto& a_Field) { details::ReadField(a_Buffer, a_Field, bitBuffer); });
|
||||
bitBuffer.UpdateRead(false);
|
||||
}
|
||||
|
||||
} // namespace details
|
||||
} // namespace sp
|
||||
|
||||
Reference in New Issue
Block a user