2021-10-30 15:01:50 +00:00
|
|
|
#ifndef OPENMW_COMPONENTS_SERIALIZATION_BINARYWRITER_H
|
|
|
|
#define OPENMW_COMPONENTS_SERIALIZATION_BINARYWRITER_H
|
2021-10-22 22:06:10 +00:00
|
|
|
|
2021-11-08 18:28:04 +00:00
|
|
|
#include <components/misc/endianness.hpp>
|
|
|
|
|
|
|
|
#include <algorithm>
|
2021-10-22 22:06:10 +00:00
|
|
|
#include <cassert>
|
|
|
|
#include <cstddef>
|
|
|
|
#include <cstring>
|
|
|
|
#include <stdexcept>
|
|
|
|
#include <type_traits>
|
|
|
|
|
2021-10-30 15:01:50 +00:00
|
|
|
namespace Serialization
|
2021-10-22 22:06:10 +00:00
|
|
|
{
|
2021-12-19 21:49:41 +00:00
|
|
|
struct NotEnoughSpace : std::runtime_error
|
|
|
|
{
|
|
|
|
NotEnoughSpace()
|
|
|
|
: std::runtime_error("Not enough space")
|
|
|
|
{
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2021-10-22 22:06:10 +00:00
|
|
|
struct BinaryWriter
|
|
|
|
{
|
|
|
|
public:
|
|
|
|
explicit BinaryWriter(std::byte* dest, const std::byte* end)
|
|
|
|
: mDest(dest)
|
|
|
|
, mEnd(end)
|
|
|
|
{
|
|
|
|
assert(mDest <= mEnd);
|
|
|
|
}
|
|
|
|
|
|
|
|
BinaryWriter(const BinaryWriter&) = delete;
|
|
|
|
|
|
|
|
template <class Format, class T>
|
|
|
|
void operator()(Format&& format, const T& value)
|
|
|
|
{
|
2021-11-08 18:25:42 +00:00
|
|
|
if constexpr (std::is_enum_v<T>)
|
|
|
|
(*this)(std::forward<Format>(format), static_cast<std::underlying_type_t<T>>(value));
|
|
|
|
else if constexpr (std::is_arithmetic_v<T>)
|
2021-10-22 22:06:10 +00:00
|
|
|
{
|
2021-11-08 18:25:42 +00:00
|
|
|
if (mEnd - mDest < static_cast<std::ptrdiff_t>(sizeof(T)))
|
2021-12-19 21:49:41 +00:00
|
|
|
throw NotEnoughSpace();
|
2021-11-08 18:28:04 +00:00
|
|
|
writeValue(value);
|
2021-10-22 22:06:10 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
format(*this, value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
template <class Format, class T>
|
|
|
|
auto operator()(Format&& format, const T* data, std::size_t count)
|
|
|
|
{
|
2021-11-08 18:25:42 +00:00
|
|
|
if constexpr (std::is_enum_v<T>)
|
|
|
|
(*this)(std::forward<Format>(format), reinterpret_cast<const std::underlying_type_t<T>*>(data), count);
|
|
|
|
else if constexpr (std::is_arithmetic_v<T>)
|
2021-10-22 22:06:10 +00:00
|
|
|
{
|
|
|
|
const std::size_t size = sizeof(T) * count;
|
|
|
|
if (mEnd - mDest < static_cast<std::ptrdiff_t>(size))
|
2021-12-19 21:49:41 +00:00
|
|
|
throw NotEnoughSpace();
|
2021-11-08 18:28:04 +00:00
|
|
|
if constexpr (Misc::IS_LITTLE_ENDIAN)
|
|
|
|
{
|
|
|
|
std::memcpy(mDest, data, size);
|
|
|
|
mDest += size;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
std::for_each(data, data + count, [&](const T& v) { writeValue(v); });
|
2021-10-22 22:06:10 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
format(*this, data, count);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
std::byte* mDest;
|
|
|
|
const std::byte* const mEnd;
|
2021-11-08 18:28:04 +00:00
|
|
|
|
|
|
|
template <class T>
|
|
|
|
void writeValue(const T& value) noexcept
|
|
|
|
{
|
|
|
|
T coverted = Misc::toLittleEndian(value);
|
|
|
|
std::memcpy(mDest, &coverted, sizeof(T));
|
|
|
|
mDest += sizeof(T);
|
|
|
|
}
|
2021-10-22 22:06:10 +00:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|