mirror of
https://github.com/Relintai/pandemonium_engine.git
synced 2025-01-25 10:29:18 +01:00
343 lines
9.6 KiB
C++
343 lines
9.6 KiB
C++
|
/*************************************************************************/
|
||
|
/* net_utilities.h */
|
||
|
/*************************************************************************/
|
||
|
/* This file is part of: */
|
||
|
/* GODOT ENGINE */
|
||
|
/* https://godotengine.org */
|
||
|
/*************************************************************************/
|
||
|
/* Copyright (c) 2007-2021 Juan Linietsky, Ariel Manzur. */
|
||
|
/* Copyright (c) 2014-2021 Godot Engine contributors (cf. AUTHORS.md). */
|
||
|
/* */
|
||
|
/* Permission is hereby granted, free of charge, to any person obtaining */
|
||
|
/* a copy of this software and associated documentation files (the */
|
||
|
/* "Software"), to deal in the Software without restriction, including */
|
||
|
/* without limitation the rights to use, copy, modify, merge, publish, */
|
||
|
/* distribute, sublicense, and/or sell copies of the Software, and to */
|
||
|
/* permit persons to whom the Software is furnished to do so, subject to */
|
||
|
/* the following conditions: */
|
||
|
/* */
|
||
|
/* The above copyright notice and this permission notice shall be */
|
||
|
/* included in all copies or substantial portions of the Software. */
|
||
|
/* */
|
||
|
/* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, */
|
||
|
/* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF */
|
||
|
/* MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT.*/
|
||
|
/* IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY */
|
||
|
/* CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, */
|
||
|
/* TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE */
|
||
|
/* SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. */
|
||
|
/*************************************************************************/
|
||
|
|
||
|
/**
|
||
|
@author AndreaCatania
|
||
|
*/
|
||
|
|
||
|
#ifndef NET_UTILITIES_H
|
||
|
#define NET_UTILITIES_H
|
||
|
|
||
|
#include "core/math/math_funcs.h"
|
||
|
#include "core/templates/local_vector.h"
|
||
|
#include "core/variant/variant.h"
|
||
|
|
||
|
#ifdef DEBUG_ENABLED
|
||
|
#define NET_DEBUG_PRINT(msg) \
|
||
|
print_line(String("[Net] ") + msg)
|
||
|
#define NET_DEBUG_WARN(msg) \
|
||
|
WARN_PRINT(String("[Net] ") + msg)
|
||
|
#define NET_DEBUG_ERR(msg) \
|
||
|
ERR_PRINT(String("[Net] ") + msg)
|
||
|
#else
|
||
|
#define NET_DEBUG_PRINT(msg)
|
||
|
#define NET_DEBUG_WARN(msg)
|
||
|
#define NET_DEBUG_ERR(msg)
|
||
|
#endif
|
||
|
|
||
|
typedef uint32_t NetNodeId;
|
||
|
typedef uint32_t NetVarId;
|
||
|
|
||
|
/// Flags used to control when an event is executed.
|
||
|
enum NetEventFlag {
|
||
|
|
||
|
// ~~ Flags ~~ //
|
||
|
EMPTY = 0,
|
||
|
|
||
|
/// Called at the end of the frame, if the value is different.
|
||
|
/// It's also called when a variable is modified by the
|
||
|
/// `apply_scene_changes` function.
|
||
|
CHANGE = 1 << 0,
|
||
|
|
||
|
/// Called when the variable is modified by the `NetworkSynchronizer`
|
||
|
/// because not in sync with the server.
|
||
|
SYNC_RECOVER = 1 << 1,
|
||
|
|
||
|
/// Called when the variable is modified by the `NetworkSynchronizer`
|
||
|
/// because it's preparing the node for the rewinding.
|
||
|
SYNC_RESET = 1 << 2,
|
||
|
|
||
|
/// Called when the variable is modified during the rewinding phase.
|
||
|
SYNC_REWIND = 1 << 3,
|
||
|
|
||
|
/// Called at the end of the recovering phase, if the value was modified
|
||
|
/// during the rewinding.
|
||
|
END_SYNC = 1 << 4,
|
||
|
|
||
|
// ~~ Preconfigured ~~ //
|
||
|
|
||
|
DEFAULT = CHANGE | END_SYNC,
|
||
|
SYNC = SYNC_RECOVER | SYNC_RESET | SYNC_REWIND,
|
||
|
ALWAYS = CHANGE | SYNC_RECOVER | SYNC_RESET | SYNC_REWIND | END_SYNC
|
||
|
};
|
||
|
|
||
|
namespace NetUtility {
|
||
|
|
||
|
template <class T>
|
||
|
class StatisticalRingBuffer {
|
||
|
LocalVector<T> data;
|
||
|
uint32_t index = 0;
|
||
|
|
||
|
T avg_sum = 0;
|
||
|
|
||
|
public:
|
||
|
StatisticalRingBuffer(uint32_t p_size, T p_default);
|
||
|
void resize(uint32_t p_size, T p_default);
|
||
|
void reset(T p_default);
|
||
|
|
||
|
void push(T p_value);
|
||
|
|
||
|
/// Maximum value.
|
||
|
T max() const;
|
||
|
|
||
|
/// Minumum value.
|
||
|
T min(uint32_t p_consider_last) const;
|
||
|
|
||
|
/// Median value.
|
||
|
T average() const;
|
||
|
|
||
|
T get_deviation(T p_mean) const;
|
||
|
|
||
|
private:
|
||
|
// Used to avoid accumulate precision loss.
|
||
|
void force_recompute_avg_sum();
|
||
|
};
|
||
|
|
||
|
template <class T>
|
||
|
StatisticalRingBuffer<T>::StatisticalRingBuffer(uint32_t p_size, T p_default) {
|
||
|
resize(p_size, p_default);
|
||
|
}
|
||
|
|
||
|
template <class T>
|
||
|
void StatisticalRingBuffer<T>::resize(uint32_t p_size, T p_default) {
|
||
|
data.resize(p_size);
|
||
|
|
||
|
reset(p_default);
|
||
|
}
|
||
|
|
||
|
template <class T>
|
||
|
void StatisticalRingBuffer<T>::reset(T p_default) {
|
||
|
for (uint32_t i = 0; i < data.size(); i += 1) {
|
||
|
data[i] = p_default;
|
||
|
}
|
||
|
|
||
|
index = 0;
|
||
|
force_recompute_avg_sum();
|
||
|
}
|
||
|
|
||
|
template <class T>
|
||
|
void StatisticalRingBuffer<T>::push(T p_value) {
|
||
|
avg_sum -= data[index];
|
||
|
avg_sum += p_value;
|
||
|
data[index] = p_value;
|
||
|
|
||
|
index = (index + 1) % data.size();
|
||
|
if (index == 0) {
|
||
|
// Each cycle recompute the sum.
|
||
|
force_recompute_avg_sum();
|
||
|
}
|
||
|
}
|
||
|
|
||
|
template <class T>
|
||
|
T StatisticalRingBuffer<T>::max() const {
|
||
|
CRASH_COND(data.size() == 0);
|
||
|
|
||
|
T a = data[0];
|
||
|
for (uint32_t i = 1; i < data.size(); i += 1) {
|
||
|
a = MAX(a, data[i]);
|
||
|
}
|
||
|
return a;
|
||
|
}
|
||
|
|
||
|
template <class T>
|
||
|
T StatisticalRingBuffer<T>::min(uint32_t p_consider_last) const {
|
||
|
CRASH_COND(data.size() == 0);
|
||
|
p_consider_last = MIN(p_consider_last, data.size());
|
||
|
|
||
|
const uint32_t youngest = (index == 0 ? data.size() : index) - 1;
|
||
|
const uint32_t oldest = (index + (data.size() - p_consider_last)) % data.size();
|
||
|
|
||
|
T a = data[oldest];
|
||
|
|
||
|
uint32_t i = oldest;
|
||
|
do {
|
||
|
i = (i + 1) % data.size();
|
||
|
a = MIN(a, data[i]);
|
||
|
} while (i != youngest);
|
||
|
|
||
|
return a;
|
||
|
}
|
||
|
|
||
|
template <class T>
|
||
|
T StatisticalRingBuffer<T>::average() const {
|
||
|
CRASH_COND(data.size() == 0);
|
||
|
|
||
|
#ifdef DEBUG_ENABLED
|
||
|
T a = data[0];
|
||
|
for (uint32_t i = 1; i < data.size(); i += 1) {
|
||
|
a += data[i];
|
||
|
}
|
||
|
a = a / T(data.size());
|
||
|
T b = avg_sum / T(data.size());
|
||
|
const T difference = a > b ? a - b : b - a;
|
||
|
ERR_FAIL_COND_V_MSG(difference > (CMP_EPSILON * 4.0), b, "The `avg_sum` accumulated a sensible precision loss: " + rtos(difference));
|
||
|
return b;
|
||
|
#else
|
||
|
// Divide it by the buffer size is wrong when the buffer is not yet fully
|
||
|
// initialized. However, this is wrong just for the first run.
|
||
|
// I'm leaving it as is because solve it mean do more operations. All this
|
||
|
// just to get the right value for the first few frames.
|
||
|
return avg_sum / T(data.size());
|
||
|
#endif
|
||
|
}
|
||
|
|
||
|
template <class T>
|
||
|
T StatisticalRingBuffer<T>::get_deviation(T p_mean) const {
|
||
|
if (data.size() <= 0) {
|
||
|
return T();
|
||
|
}
|
||
|
|
||
|
double r = 0;
|
||
|
for (uint32_t i = 0; i < data.size(); i += 1) {
|
||
|
r += Math::pow(double(data[i]) - double(p_mean), 2.0);
|
||
|
}
|
||
|
|
||
|
return Math::sqrt(r / double(data.size()));
|
||
|
}
|
||
|
|
||
|
template <class T>
|
||
|
void StatisticalRingBuffer<T>::force_recompute_avg_sum() {
|
||
|
#ifdef DEBUG_ENABLED
|
||
|
// This class is not supposed to be used with 0 size.
|
||
|
CRASH_COND(data.size() <= 0);
|
||
|
#endif
|
||
|
avg_sum = data[0];
|
||
|
for (uint32_t i = 1; i < data.size(); i += 1) {
|
||
|
avg_sum += data[i];
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/// Specific node listener. Alone this doesn't do much, but allows the
|
||
|
/// `ChangeListener` to know and keep track of the node events.
|
||
|
struct NodeChangeListener {
|
||
|
struct NodeData *node_data = nullptr;
|
||
|
NetVarId var_id = UINT32_MAX;
|
||
|
|
||
|
bool old_set = false;
|
||
|
Variant old_value;
|
||
|
|
||
|
bool operator==(const NodeChangeListener &p_other) const;
|
||
|
};
|
||
|
|
||
|
/// Change listener that rapresents a pair of Object and Method.
|
||
|
/// This can track the changes of many nodes and variables. It's dispatched
|
||
|
/// if one or more tracked variable change during the tracked phase, specified
|
||
|
/// by the flag.
|
||
|
struct ChangeListener {
|
||
|
// TODO use a callable instead??
|
||
|
ObjectID object_id = ObjectID();
|
||
|
StringName method;
|
||
|
uint32_t method_argument_count;
|
||
|
NetEventFlag flag;
|
||
|
|
||
|
LocalVector<NodeChangeListener> watching_vars;
|
||
|
LocalVector<Variant> old_values;
|
||
|
bool emitted = true;
|
||
|
|
||
|
bool operator==(const ChangeListener &p_other) const;
|
||
|
};
|
||
|
|
||
|
struct Var {
|
||
|
StringName name;
|
||
|
Variant value;
|
||
|
};
|
||
|
|
||
|
struct VarData {
|
||
|
NetVarId id = UINT32_MAX;
|
||
|
Var var;
|
||
|
bool skip_rewinding = false;
|
||
|
bool enabled = false;
|
||
|
Vector<uint32_t> change_listeners;
|
||
|
|
||
|
VarData() = default;
|
||
|
VarData(const StringName &p_name);
|
||
|
VarData(NetVarId p_id, const StringName &p_name, const Variant &p_val, bool p_skip_rewinding, bool p_enabled);
|
||
|
|
||
|
bool operator==(const VarData &p_other) const;
|
||
|
bool operator<(const VarData &p_other) const;
|
||
|
};
|
||
|
|
||
|
struct NodeData {
|
||
|
// ID used to reference this Node in the networked calls.
|
||
|
uint32_t id = 0;
|
||
|
ObjectID instance_id = ObjectID();
|
||
|
NodeData *controlled_by = nullptr;
|
||
|
|
||
|
/// When `false`, this node is not sync. It's usefult to locally pause sync
|
||
|
/// of specific nodes.
|
||
|
bool sync_enabled = true;
|
||
|
|
||
|
bool is_controller = false;
|
||
|
LocalVector<NodeData *> controlled_nodes;
|
||
|
LocalVector<NodeData *> dependency_nodes;
|
||
|
LocalVector<uint32_t> dependency_nodes_end;
|
||
|
|
||
|
/// The sync variables of this node. The order of this vector matters
|
||
|
/// because the index is the `NetVarId`.
|
||
|
LocalVector<VarData> vars;
|
||
|
LocalVector<StringName> functions;
|
||
|
|
||
|
// This is valid to use only inside the process function.
|
||
|
Node *node = nullptr;
|
||
|
|
||
|
NodeData() = default;
|
||
|
|
||
|
void process(const real_t p_delta) const;
|
||
|
};
|
||
|
|
||
|
struct PeerData {
|
||
|
NetNodeId controller_id = UINT32_MAX;
|
||
|
// For new peers notify the state as soon as possible.
|
||
|
bool force_notify_snapshot = true;
|
||
|
// For new peers a full snapshot is needed.
|
||
|
bool need_full_snapshot = true;
|
||
|
// Used to know if the peer is enabled.
|
||
|
bool enabled = true;
|
||
|
};
|
||
|
|
||
|
struct Snapshot {
|
||
|
uint32_t input_id;
|
||
|
/// The Node variables in a particular frame. The order of this vector
|
||
|
/// matters because the index is the `NetNodeId`.
|
||
|
/// The variable array order also matter.
|
||
|
Vector<Vector<Var>> node_vars;
|
||
|
|
||
|
operator String() const;
|
||
|
};
|
||
|
|
||
|
struct PostponedRecover {
|
||
|
NodeData *node_data = nullptr;
|
||
|
Vector<Var> vars;
|
||
|
};
|
||
|
|
||
|
} // namespace NetUtility
|
||
|
|
||
|
#endif
|