8 #include <unordered_map> 14 static size_t instanceCount();
15 mutable uint64_t counter = 0;
19 struct Snapshot : std::enable_if<fromStringSupported((T *)nullptr) ||
20 propertiesSupported((T *)nullptr),
23 static std::shared_ptr<const Snapshot<T>>
24 save(
const T &v,
const std::shared_ptr<
const Snapshot<T>> &s,
25 const SnapshotFilter &filter) {
26 if (s && s->value == v) {
29 auto ret = std::make_shared<Snapshot>();
34 void apply(T &v)
const { v = value; }
39 std::shared_ptr<const Type> type;
41 std::vector<std::shared_ptr<const SnapshotBase>> property_snapshots;
42 static std::shared_ptr<const Type> findType(
const std::shared_ptr<T> &p) {
46 return Type::find(
typeid(*p));
48 static std::shared_ptr<const Snapshot<std::shared_ptr<T>>>
49 save(
const std::shared_ptr<T> &v,
50 const std::shared_ptr<
const Snapshot<std::shared_ptr<T>>> &s0,
51 const SnapshotFilter &filter) {
52 std::shared_ptr<Snapshot<std::shared_ptr<T>>> ret;
53 if (s0 ==
nullptr || s0->nonzero != (
bool)v || (v && (s0->id != v->id())) ||
54 (v && s0->type->typeId() != std::type_index(
typeid(*v)))) {
55 ret = std::make_shared<Snapshot<std::shared_ptr<T>>>();
57 ret->property_snapshots.resize(v->properties().size());
59 ret->id = (v ? v->id() : 0);
60 ret->type = findType(v);
61 ret->nonzero = (bool)v;
64 auto properties = v->properties();
65 for (
size_t property_index = 0; property_index < properties.size();
67 auto &
property = properties[property_index];
68 auto previous_snapshot =
69 ((s0 !=
nullptr && property_index < s0->property_snapshots.size() &&
70 s0->nonzero && s0->type->typeId() == std::type_index(
typeid(*v)))
71 ? s0->property_snapshots.at(property_index)
73 auto property_snapshot =
74 ((
property.canContainObjects() || filter ==
nullptr ||
75 previous_snapshot ==
nullptr || filter(v))
76 ? property.save(previous_snapshot, filter)
81 if (property_snapshot ==
nullptr) {
82 throw std::runtime_error(
"failed to save property");
84 if (property_snapshot != previous_snapshot && ret ==
nullptr) {
85 ret = std::make_shared<Snapshot<std::shared_ptr<T>>>(*s0);
86 ret->property_snapshots.resize(v->properties().size());
89 ret->property_snapshots.at(property_index) = property_snapshot;
94 ret->property_snapshots.clear();
97 return ret ? ret : s0;
99 void apply(std::shared_ptr<T> &v)
const {
104 if (v ==
nullptr || type != Type::find(
typeid(*v))) {
105 v = type->instantiate<T>();
109 auto properties = v->properties();
110 for (
size_t property_index = 0; property_index < properties.size();
112 auto &
property = properties[property_index];
113 property.applySnapshot(property_snapshots.at(property_index));
121 std::vector<std::shared_ptr<const Snapshot<T>>> snapshots;
122 static std::shared_ptr<const Snapshot>
123 save(
const std::vector<T> &v,
const std::shared_ptr<const Snapshot> &s0,
124 const SnapshotFilter &filter) {
125 std::shared_ptr<Snapshot<std::vector<T>>> ret;
126 if (s0 ==
nullptr || (s0->snapshots.size() != v.size())) {
127 ret = std::make_shared<Snapshot<std::vector<T>>>();
128 ret->snapshots.resize(v.size());
130 for (
size_t i = 0; i < v.size(); i++) {
132 (s0 !=
nullptr && i < s0->snapshots.size())
136 if (ret !=
nullptr || i >= s0->snapshots.size() ||
137 shot != s0->snapshots[i]) {
138 if (ret ==
nullptr) {
140 ret = std::make_shared<Snapshot<std::vector<T>>>(*s0);
142 ret = std::make_shared<Snapshot<std::vector<T>>>();
144 ret->snapshots.resize(v.size());
146 ret->snapshots[i] = shot;
149 return ret ? ret : s0;
151 inline void shuffle(...)
const {}
152 template <
class X>
void shuffle(std::vector<std::shared_ptr<X>> &v)
const {
154 std::unordered_map<uint64_t, std::shared_ptr<X>> id_map;
155 id_map.reserve(v.size());
161 v.resize(snapshots.size());
162 for (
size_t i = 0; i < snapshots.size(); i++) {
163 auto iter = id_map.find(snapshots[i]->
id);
164 if (iter != id_map.end()) {
172 void apply(std::vector<T> &v)
const {
174 v.resize(snapshots.size());
175 for (
size_t i = 0; i < snapshots.size(); i++) {
176 snapshots[i]->apply(v[i]);
183 static std::shared_ptr<const Snapshot>
184 save(
const Handle<T> &v,
const std::shared_ptr<const Snapshot> &s0,
185 const SnapshotFilter &filter) {
186 if (s0 && s0->id == v.id()) {
189 auto ret = std::make_shared<Snapshot>();
194 void apply(
Handle<T> &v)
const { v.reset(
id); }
198 std::shared_ptr<const SnapshotBase>
200 const std::shared_ptr<const SnapshotBase> &x,
201 const SnapshotFilter &filter)
const {
202 auto snapshot = std::dynamic_pointer_cast<
const Snapshot<T>>(x);
203 if (x && !snapshot) {
204 throw std::runtime_error(
"snapshot type mismatch");
211 void *value,
const std::shared_ptr<const SnapshotBase> &x)
const {
212 std::dynamic_pointer_cast<
const Snapshot<T>>(x)->apply(*(T *)value);