This commit is contained in:
@@ -3,19 +3,51 @@
|
|||||||
#include <boost/pfr.hpp>
|
#include <boost/pfr.hpp>
|
||||||
#include <sp/common/ByteSwapping.h>
|
#include <sp/common/ByteSwapping.h>
|
||||||
#include <sp/common/DataBuffer.h>
|
#include <sp/common/DataBuffer.h>
|
||||||
|
#include <sp/protocol/BitField.h>
|
||||||
|
|
||||||
namespace sp {
|
namespace sp {
|
||||||
namespace details {
|
namespace details {
|
||||||
|
|
||||||
|
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) {
|
||||||
|
T filled = static_cast<T>(a_DataRaw);
|
||||||
|
ToNetwork(filled);
|
||||||
|
a_Buffer << filled;
|
||||||
|
a_Offset = 0;
|
||||||
|
a_DataRaw = 0;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
template <typename T>
|
template <typename T>
|
||||||
void WriteField(DataBuffer& a_Buffer, const T& a_Data) {
|
void WriteField(DataBuffer& a_Buffer, const T& a_Data, std::uint64_t& a_DataRaw, std::size_t& a_Offset) {
|
||||||
T swapped = a_Data;
|
T swapped = a_Data;
|
||||||
ToNetwork(swapped);
|
ToNetwork(swapped);
|
||||||
a_Buffer << 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>
|
template <typename T>
|
||||||
void ReadField(DataBuffer& a_Buffer, T& a_Data) {
|
void ReadField(DataBuffer& a_Buffer, T& a_Data, std::size_t& a_Offset) {
|
||||||
a_Buffer >> a_Data;
|
a_Buffer >> a_Data;
|
||||||
FromNetwork(a_Data);
|
FromNetwork(a_Data);
|
||||||
}
|
}
|
||||||
@@ -23,13 +55,17 @@ void ReadField(DataBuffer& a_Buffer, T& a_Data) {
|
|||||||
template <typename TData>
|
template <typename TData>
|
||||||
DataBuffer WriteMessage(const TData& a_MessageData) {
|
DataBuffer WriteMessage(const TData& a_MessageData) {
|
||||||
DataBuffer buffer;
|
DataBuffer buffer;
|
||||||
boost::pfr::for_each_field(a_MessageData, [&buffer](const auto& a_Field) { WriteField(buffer, a_Field); });
|
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); });
|
||||||
return buffer;
|
return buffer;
|
||||||
}
|
}
|
||||||
|
|
||||||
template <typename TData>
|
template <typename TData>
|
||||||
void ReadMessage(DataBuffer& a_Buffer, TData& a_MessageData) {
|
void ReadMessage(DataBuffer& a_Buffer, TData& a_MessageData) {
|
||||||
boost::pfr::for_each_field(a_MessageData, [&a_Buffer](auto& a_Field) { ReadField(a_Buffer, a_Field); });
|
std::size_t currentOffset = 0;
|
||||||
|
boost::pfr::for_each_field(a_MessageData, [&a_Buffer, ¤tOffset](auto& a_Field) { ReadField(a_Buffer, a_Field, currentOffset); });
|
||||||
}
|
}
|
||||||
|
|
||||||
} // namespace details
|
} // namespace details
|
||||||
|
|||||||
37
include/sp/protocol/BitField.h
Normal file
37
include/sp/protocol/BitField.h
Normal file
@@ -0,0 +1,37 @@
|
|||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <cstdint>
|
||||||
|
|
||||||
|
namespace sp {
|
||||||
|
|
||||||
|
template <typename T, std::size_t BitSize>
|
||||||
|
class BitField {
|
||||||
|
private:
|
||||||
|
static constexpr int BITS_IN_CHAR = 8;
|
||||||
|
static_assert(sizeof(T) * BITS_IN_CHAR > BitSize, "The bit count must be lower than the actual type size !");
|
||||||
|
|
||||||
|
T m_Data;
|
||||||
|
|
||||||
|
public:
|
||||||
|
BitField() : m_Data{} {};
|
||||||
|
BitField(T a_Data) : m_Data(a_Data) {}
|
||||||
|
|
||||||
|
BitField& operator=(T a_Data) {
|
||||||
|
m_Data = a_Data;
|
||||||
|
return *this;
|
||||||
|
}
|
||||||
|
|
||||||
|
constexpr std::size_t GetBitSize() const {
|
||||||
|
return BitSize;
|
||||||
|
}
|
||||||
|
|
||||||
|
T& operator*() {
|
||||||
|
return m_Data;
|
||||||
|
}
|
||||||
|
|
||||||
|
const T& operator*() const {
|
||||||
|
return m_Data;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
} // namespace sp
|
||||||
@@ -4,6 +4,7 @@
|
|||||||
#include <sp/protocol/MessageHandler.h>
|
#include <sp/protocol/MessageHandler.h>
|
||||||
#include <sp/io/MessageStream.h>
|
#include <sp/io/MessageStream.h>
|
||||||
#include <sp/io/StdIo.h>
|
#include <sp/io/StdIo.h>
|
||||||
|
#include <sp/protocol/BitField.h>
|
||||||
|
|
||||||
#include <sp/extensions/Compress.h>
|
#include <sp/extensions/Compress.h>
|
||||||
|
|
||||||
@@ -21,8 +22,8 @@ template <typename TData, PacketID ID>
|
|||||||
using Message = sp::ConcreteMessage<TData, PacketID, ID, PacketHandler>;
|
using Message = sp::ConcreteMessage<TData, PacketID, ID, PacketHandler>;
|
||||||
|
|
||||||
struct KeepAlivePacket {
|
struct KeepAlivePacket {
|
||||||
std::uint64_t m_KeepAlive;
|
sp::BitField<std::uint16_t, 12> one;
|
||||||
std::string mdc;
|
sp::BitField<std::uint16_t, 4> two;
|
||||||
};
|
};
|
||||||
|
|
||||||
using KeepAliveMessage = Message<KeepAlivePacket, PacketID::KeepAlive>;
|
using KeepAliveMessage = Message<KeepAlivePacket, PacketID::KeepAlive>;
|
||||||
@@ -34,7 +35,7 @@ class PacketHandler : public sp::MessageHandler<AllMessages> {};
|
|||||||
class MyHandler : public PacketHandler {
|
class MyHandler : public PacketHandler {
|
||||||
public:
|
public:
|
||||||
virtual void Handle(const KeepAlivePacket& msg) {
|
virtual void Handle(const KeepAlivePacket& msg) {
|
||||||
std::cout << "I recieved a keep alive : " << msg.m_KeepAlive << " : " << msg.mdc << "\n";
|
std::cout << "I recieved a keep alive : " << *msg.one << " : " << *msg.two << "\n";
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
@@ -45,7 +46,7 @@ using PacketFactory = sp::MessageFactory<PacketBase, AllMessages>;
|
|||||||
using PacketStream = sp::MessageStream<PacketFactory>;
|
using PacketStream = sp::MessageStream<PacketFactory>;
|
||||||
|
|
||||||
int main() {
|
int main() {
|
||||||
KeepAliveMessage m{69UL, "ceci est une mdc aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa"};
|
KeepAliveMessage m{69, 5};
|
||||||
|
|
||||||
// dispatch tests
|
// dispatch tests
|
||||||
|
|
||||||
@@ -79,6 +80,8 @@ int main() {
|
|||||||
|
|
||||||
d.Dispatch(*message2);
|
d.Dispatch(*message2);
|
||||||
|
|
||||||
|
// Todo : verify bitfields
|
||||||
|
|
||||||
// message->Write(file);
|
// message->Write(file);
|
||||||
// file << std::endl;
|
// file << std::endl;
|
||||||
// m.Write(file);
|
// m.Write(file);
|
||||||
|
|||||||
Reference in New Issue
Block a user