NcEngine
SerializeCpo.h
1#pragma once
2
3#include "BinarySerializationDetail.h"
4
6namespace nc::serialize::cpo
7{
8// Helper to allow static_assert(false) prior to c++23.
9template <class>
10inline constexpr bool g_alwaysFalse = false;
11
12// Indicates how a (de)serialize call will be resolved.
13enum class Dispatch { None, Member, Adl, Default };
14
15// Satisfied for types that have a Serialize member function.
16template <class T>
17concept HasSerializeMember = requires(std::ostream& stream, const T& obj)
18{
19 { obj.Serialize(stream) } -> std::same_as<void>;
20};
21
22// Satisfied for types that have a Serialize function in their namespace.
23template <class T>
24concept HasSerializeAdl = requires(std::ostream& stream, const T& obj)
25{
26 { Serialize(stream, obj) } -> std::same_as<void>;
27};
28
29// Satisfied for types that have a compatible Serialize function internally.
30template <class T>
31concept HasSerializeDefault = requires(std::ostream& stream, const T& obj)
32{
33 { nc::serialize::binary::Serialize(stream, obj) } -> std::same_as<void>;
34};
35
36// CPO for nc::serialize::Serialize - dispatches to a `Serialize()` function that is either
37// a member of T, non-member found via adl, or internal non- member depending on what is
38// available. Resolution is attempted in that order.
39struct SerializeFn
40{
41 private:
42 template<class T>
43 static consteval auto GetDispatch() -> Dispatch
44 {
45 if constexpr(HasSerializeMember<T>)
46 return Dispatch::Member;
47 else if constexpr(HasSerializeAdl<T>)
48 return Dispatch::Adl;
49 else if constexpr(HasSerializeDefault<T>)
50 return Dispatch::Default;
51 else
52 return Dispatch::None;
53 }
54
55 template<class T>
56 static constexpr auto Strategy = GetDispatch<T>();
57
58 public:
59 template<class T>
60 requires (Strategy<T> != Dispatch::None)
61 auto operator()(std::ostream& stream, const T& obj) const
62 {
63 constexpr auto dispatch = Strategy<T>;
64 if constexpr(dispatch == Dispatch::Member)
65 obj.Serialize(stream);
66 else if constexpr(dispatch == Dispatch::Adl)
67 Serialize(stream, obj);
68 else if constexpr(dispatch == Dispatch::Default)
69 nc::serialize::binary::Serialize(stream, obj);
70 else
71 static_assert(g_alwaysFalse<T>, "Unreachable");
72 }
73};
74
75// Satisfied for types that have a Deserialize member function.
76template <class T>
77concept HasDeserializeMember = requires(std::istream& stream, T& obj)
78{
79 { obj.Deserialize(stream) } -> std::same_as<void>;
80};
81
82// Satisfied for types that have a Deserialize function in their namespace.
83template <class T>
84concept HasDeserializeAdl = requires(std::istream& stream, T& obj)
85{
86 { Deserialize(stream, obj) } -> std::same_as<void>; // intentional ADL
87};
88
89// Satisfied for types that have a compatible Deserialize function internally.
90template <class T>
91concept HasDeserializeDefault = requires(std::istream& stream, T& obj)
92{
93 { nc::serialize::binary::Deserialize(stream, obj) } -> std::same_as<void>;
94};
95
96// CPO for nc::serialize::Deserialize - dispatches to a `Deserialize()` function that is either
97// a member of T, non-member found via adl, or internal non- member depending on what is
98// available. Resolution is attempted in that order.
99struct DeserializeFn
100{
101 private:
102 template<class T>
103 static consteval auto GetDispatch() -> Dispatch
104 {
105 if constexpr(HasDeserializeMember<T>)
106 return Dispatch::Member;
107 else if constexpr(HasDeserializeAdl<T>)
108 return Dispatch::Adl;
109 else if constexpr(HasDeserializeDefault<T>)
110 return Dispatch::Default;
111 else
112 return Dispatch::None;
113 }
114
115 template<class T>
116 static constexpr auto Strategy = GetDispatch<T>();
117
118 public:
119 template<class T>
120 requires (Strategy<T> != Dispatch::None)
121 auto operator()(std::istream& stream, T& obj) const
122 {
123 constexpr auto dispatch = Strategy<T>;
124 if constexpr(dispatch == Dispatch::Member)
125 obj.Deserialize(stream);
126 else if constexpr(dispatch == Dispatch::Adl)
127 Deserialize(stream, obj);
128 else if constexpr(dispatch == Dispatch::Default)
129 nc::serialize::binary::Deserialize(stream, obj);
130 else
131 static_assert(g_alwaysFalse<T>, "Unreachable");
132 }
133};
134} // namespace nc::serialize::cpo
constexpr nc::serialize::cpo::DeserializeFn Deserialize
Deserialize an object from a stream.
Definition: BinarySerialization.h:47
constexpr nc::serialize::cpo::SerializeFn Serialize
Serialize an object to a stream.
Definition: BinarySerialization.h:41