mirror of
https://github.com/OpenMW/openmw.git
synced 2025-01-22 09:53:54 +00:00
a58f1a94e3
To construct serializer from given entities: * Data source/destination - any value that has to be serialized/deserialized, usually already existing type. * Format - functional object to define high level serialization logic to define specific format and data schema. Like order of fields, allocation. * Visitor - functional object to define low level serialization logic to operator on given data part. * BinaryWriter - copies given value into provided buffer. * BinaryReader - copies value into given destination from provided buffer. * SizeAccumulator - calculates required buffer size for given data.
80 lines
2.4 KiB
C++
80 lines
2.4 KiB
C++
#ifndef OPENMW_COMPONENTS_DETOURNAVIGATOR_SERIALIZATION_FORMAT_H
|
|
#define OPENMW_COMPONENTS_DETOURNAVIGATOR_SERIALIZATION_FORMAT_H
|
|
|
|
#include <algorithm>
|
|
#include <cstddef>
|
|
#include <functional>
|
|
#include <type_traits>
|
|
#include <utility>
|
|
#include <vector>
|
|
|
|
namespace DetourNavigator::Serialization
|
|
{
|
|
enum class Mode
|
|
{
|
|
Read,
|
|
Write,
|
|
};
|
|
|
|
template <class>
|
|
struct IsContiguousContainer : std::false_type {};
|
|
|
|
template <class ... Args>
|
|
struct IsContiguousContainer<std::vector<Args ...>> : std::true_type {};
|
|
|
|
template <class T>
|
|
constexpr bool isContiguousContainer = IsContiguousContainer<std::decay_t<T>>::value;
|
|
|
|
template <Mode mode, class Derived>
|
|
struct Format
|
|
{
|
|
template <class Visitor, class T>
|
|
void operator()(Visitor&& visitor, T* data, std::size_t size) const
|
|
{
|
|
if constexpr (std::is_arithmetic_v<T>)
|
|
{
|
|
visitor(self(), data, size);
|
|
}
|
|
else if constexpr (std::is_enum_v<T>)
|
|
{
|
|
if constexpr (mode == Mode::Write)
|
|
visitor(self(), reinterpret_cast<const std::underlying_type_t<T>*>(data), size);
|
|
else
|
|
{
|
|
static_assert(mode == Mode::Read);
|
|
visitor(self(), reinterpret_cast<std::underlying_type_t<T>*>(data), size);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
std::for_each(data, data + size, [&] (auto& v) { visitor(self(), v); });
|
|
}
|
|
}
|
|
|
|
template <class Visitor, class T, std::size_t size>
|
|
void operator()(Visitor&& visitor, T(& data)[size]) const
|
|
{
|
|
self()(std::forward<Visitor>(visitor), data, size);
|
|
}
|
|
|
|
template <class Visitor, class T>
|
|
auto operator()(Visitor&& visitor, T&& value) const
|
|
-> std::enable_if_t<isContiguousContainer<T>>
|
|
{
|
|
if constexpr (mode == Mode::Write)
|
|
visitor(self(), value.size());
|
|
else
|
|
{
|
|
static_assert(mode == Mode::Read);
|
|
std::size_t size = 0;
|
|
visitor(self(), size);
|
|
value.resize(size);
|
|
}
|
|
self()(std::forward<Visitor>(visitor), value.data(), value.size());
|
|
}
|
|
|
|
const Derived& self() const { return static_cast<const Derived&>(*this); }
|
|
};
|
|
}
|
|
|
|
#endif
|