1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
|
#pragma once
#include <cstddef>
#include <cstdint>
#include <span>
class DataStream
{
public:
enum Endianness
{
BigEndian,
LittleEndian,
};
private:
Endianness mEndian;
public:
DataStream();
~DataStream();
Endianness GetEndianness() const;
void SetEndianness(Endianness endianness);
void ReadBytes(size_t byteCount, std::byte* buffer);
void ReadBytes(size_t byteCount, char* buffer);
void ReadBytes(size_t byteCount, signed char* buffer);
void ReadBytes(size_t byteCount, unsigned char* buffer);
template <class TInserter>
void ReadBytes(size_t byteCount, TInserter&& inserter)
{
for (size_t i = 0; i < byteCount; ++i) {
uint8_t byte;
Read(byte);
inserter = byte;
}
}
void WriteBytes(size_t byteCount, const std::byte* buffer);
void WriteBytes(size_t byteCount, const char* buffer);
void WriteBytes(size_t byteCount, const signed char* buffer);
void WriteBytes(size_t byteCount, const unsigned char* buffer);
template <class TIterator>
void WriteBytes(TIterator&& begin, TIterator&& end)
{
for (; begin != end; ++begin) {
uint8_t byte = *begin;
Write(byte);
}
}
void Write(int8_t n);
void Write(int16_t n);
void Write(int32_t n);
void Write(int64_t n);
void Read(int8_t& n);
void Read(int16_t& n);
void Read(int32_t& n);
void Read(int64_t& n);
void Write(uint8_t n);
void Write(uint16_t n);
void Write(uint32_t n);
void Write(uint64_t n);
void Read(uint8_t& n);
void Read(uint16_t& n);
void Read(uint32_t& n);
void Read(uint64_t& n);
void Write(float n);
void Write(double n);
void Read(float& n);
void Read(double& n);
template <class TObject>
requires requires(TObject obj)
{
obj.ReadFromDataStream(std::declval<DataStream>());
}
void ReadObject(TObject& obj)
{
obj.ReadFromDataStream(*this);
}
template <class TObject>
requires requires(TObject obj)
{
// Let ADL happen
ReadFromDataStream(std::declval<DataStream>(), obj);
}
void ReadObject(TObject& obj)
{
ReadFromDataStream(*this, obj);
}
template <class TObject>
requires requires(TObject obj)
{
obj.WriteToDataStream(std::declval<DataStream>());
}
void WriteObject(TObject& obj)
{
obj.ReadFromDataStream(*this);
}
template <class TObject>
requires requires(TObject obj)
{
// Let ADL happen
WriteToDataStream(std::declval<DataStream>(), obj);
}
void WriteObject(TObject& obj)
{
WriteToDataStream(*this, obj);
}
};
|