Updated NetworkSynchronizer from https://github.com/GameNetworking/network_synchronizer/tree/godot-3.x . f12fee66a7e3eb2013d6c5437837770455105b91 (Merge commit: cc3c7d244adea8b25822f43963618fd2dbf18ec8)

This commit is contained in:
Relintai 2023-12-25 19:27:07 +01:00
parent 1f291939e9
commit eb63a4d682
45 changed files with 5624 additions and 2717 deletions

View File

@ -0,0 +1,189 @@
# Commented out parameters are those with the same value as base LLVM style.
# We can uncomment them if we want to change their value, or enforce the
# chosen value in case the base style changes (last sync: Clang 13.0).
---
### General config, applies to all languages ###
BasedOnStyle: LLVM
AccessModifierOffset: -4
AlignAfterOpenBracket: DontAlign
# AlignArrayOfStructures: None
# AlignConsecutiveMacros: None
# AlignConsecutiveAssignments: None
# AlignConsecutiveBitFields: None
# AlignConsecutiveDeclarations: None
# AlignEscapedNewlines: Right
AlignOperands: DontAlign
AlignTrailingComments: false
# AllowAllArgumentsOnNextLine: true
# AllowAllConstructorInitializersOnNextLine: true
AllowAllParametersOfDeclarationOnNextLine: false
# AllowShortEnumsOnASingleLine: true
# AllowShortBlocksOnASingleLine: Never
# AllowShortCaseLabelsOnASingleLine: false
AllowShortFunctionsOnASingleLine: Inline
# AllowShortLambdasOnASingleLine: All
# AllowShortIfStatementsOnASingleLine: Never
# AllowShortLoopsOnASingleLine: false
# AlwaysBreakAfterDefinitionReturnType: None
# AlwaysBreakAfterReturnType: None
# AlwaysBreakBeforeMultilineStrings: false
# AlwaysBreakTemplateDeclarations: MultiLine
# AttributeMacros:
# - __capability
# BinPackArguments: true
# BinPackParameters: true
# BraceWrapping:
# AfterCaseLabel: false
# AfterClass: false
# AfterControlStatement: Never
# AfterEnum: false
# AfterFunction: false
# AfterNamespace: false
# AfterObjCDeclaration: false
# AfterStruct: false
# AfterUnion: false
# AfterExternBlock: false
# BeforeCatch: false
# BeforeElse: false
# BeforeLambdaBody: false
# BeforeWhile: false
# IndentBraces: false
# SplitEmptyFunction: true
# SplitEmptyRecord: true
# SplitEmptyNamespace: true
# BreakBeforeBinaryOperators: None
# BreakBeforeConceptDeclarations: true
# BreakBeforeBraces: Attach
# BreakBeforeInheritanceComma: false
# BreakInheritanceList: BeforeColon
# BreakBeforeTernaryOperators: true
# BreakConstructorInitializersBeforeComma: false
BreakConstructorInitializers: AfterColon
# BreakStringLiterals: true
ColumnLimit: 0
# CommentPragmas: '^ IWYU pragma:'
# CompactNamespaces: false
ConstructorInitializerAllOnOneLineOrOnePerLine: true
ConstructorInitializerIndentWidth: 8
ContinuationIndentWidth: 8
Cpp11BracedListStyle: false
# DeriveLineEnding: true
# DerivePointerAlignment: false
# DisableFormat: false
# EmptyLineAfterAccessModifier: Never
# EmptyLineBeforeAccessModifier: LogicalBlock
# ExperimentalAutoDetectBinPacking: false
# FixNamespaceComments: true
# ForEachMacros:
# - foreach
# - Q_FOREACH
# - BOOST_FOREACH
# IfMacros:
# - KJ_IF_MAYBE
# IncludeBlocks: Preserve
IncludeCategories:
- Regex: '".*"'
Priority: 1
- Regex: '^<.*\.h>'
Priority: 2
- Regex: '^<.*'
Priority: 3
# IncludeIsMainRegex: '(Test)?$'
# IncludeIsMainSourceRegex: ''
# IndentAccessModifiers: false
IndentCaseLabels: true
# IndentCaseBlocks: false
# IndentGotoLabels: true
# IndentPPDirectives: None
# IndentExternBlock: AfterExternBlock
# IndentRequires: false
IndentWidth: 4
# IndentWrappedFunctionNames: false
# InsertTrailingCommas: None
# JavaScriptQuotes: Leave
# JavaScriptWrapImports: true
KeepEmptyLinesAtTheStartOfBlocks: false
# LambdaBodyIndentation: Signature
# MacroBlockBegin: ''
# MacroBlockEnd: ''
# MaxEmptyLinesToKeep: 1
# NamespaceIndentation: None
# PenaltyBreakAssignment: 2
# PenaltyBreakBeforeFirstCallParameter: 19
# PenaltyBreakComment: 300
# PenaltyBreakFirstLessLess: 120
# PenaltyBreakString: 1000
# PenaltyBreakTemplateDeclaration: 10
# PenaltyExcessCharacter: 1000000
# PenaltyReturnTypeOnItsOwnLine: 60
# PenaltyIndentedWhitespace: 0
# PointerAlignment: Right
# PPIndentWidth: -1
# ReferenceAlignment: Pointer
# ReflowComments: true
# ShortNamespaceLines: 1
# SortIncludes: CaseSensitive
# SortJavaStaticImport: Before
# SortUsingDeclarations: true
# SpaceAfterCStyleCast: false
# SpaceAfterLogicalNot: false
# SpaceAfterTemplateKeyword: true
# SpaceBeforeAssignmentOperators: true
# SpaceBeforeCaseColon: false
# SpaceBeforeCpp11BracedList: false
# SpaceBeforeCtorInitializerColon: true
# SpaceBeforeInheritanceColon: true
# SpaceBeforeParens: ControlStatements
# SpaceAroundPointerQualifiers: Default
# SpaceBeforeRangeBasedForLoopColon: true
# SpaceInEmptyParentheses: false
# SpacesBeforeTrailingComments: 1
# SpaceInEmptyBlock: false
# SpaceInEmptyParentheses: false
# SpacesBeforeTrailingComments: 1
# SpacesInAngles: Never
# SpacesInContainerLiterals: true
# SpacesInConditionalStatement: false
# SpacesInContainerLiterals: true
# SpacesInCStyleCastParentheses: false
## Godot TODO: We'll want to use a min of 1, but we need to see how to fix
## our comment capitalization at the same time.
SpacesInLineCommentPrefix:
Minimum: 0
Maximum: -1
# SpacesInParentheses: false
# SpacesInSquareBrackets: false
# SpaceBeforeSquareBrackets: false
# BitFieldColonSpacing: Both
# StatementAttributeLikeMacros:
# - Q_EMIT
# StatementMacros:
# - Q_UNUSED
# - QT_REQUIRE_VERSION
TabWidth: 4
# UseCRLF: false
UseTab: Always
# WhitespaceSensitiveMacros:
# - STRINGIZE
# - PP_STRINGIZE
# - BOOST_PP_STRINGIZE
# - NS_SWIFT_NAME
# - CF_SWIFT_NAME
---
### C++ specific config ###
Language: Cpp
Standard: c++14
---
### ObjC specific config ###
Language: ObjC
# ObjCBinPackProtocolList: Auto
ObjCBlockIndentWidth: 4
# ObjCBreakBeforeNestedBlockParam: true
# ObjCSpaceAfterProperty: false
# ObjCSpaceBeforeProtocolList: true
---
### Java specific config ###
Language: Java
# BreakAfterJavaFieldAnnotations: false
JavaImportGroups: ['org.godotengine', 'android', 'androidx', 'com.android', 'com.google', 'java', 'javax']
...

View File

@ -0,0 +1,2 @@
# Normalize EOL for all files that Git considers text files
* text=auto eol=lf

45
modules/network_synchronizer/.gitignore vendored Normal file
View File

@ -0,0 +1,45 @@
__generated__debugger_ui.h
# Godot-specific ignores
.import/
.godot/
export.cfg
export_presets.cfg
# Mono-specific ignores
.mono/
data_*/
# General c++ generated files
*.lib
*.o
*.obj
*.ox
*.a
*.ax
*.d
*.so
*.os
*.Plo
*.lo
# for projects that use SCons for building: http://http://www.scons.org/
.sconf_temp
.sconsign*.dblite
*.pyc
# vscode
.vscode
*.code-workspace
# QtCreator
*.cflags
*.config
*.creator
*.creator.user
*.cxxflags
*.files
*.includes
# CLion
.idea/

View File

@ -1,5 +1,9 @@
#!/usr/bin/env python
from debugger_ui import cpplize_debugger
cpplize_debugger.create_debugger_header()
Import("env")
Import("env_modules")

View File

@ -1,13 +1,12 @@
/*************************************************************************/
/* bit_array.cpp */
/*************************************************************************/
/* This file is part of: */
/* PANDEMONIUM ENGINE */
/* https://github.com/Relintai/pandemonium_engine */
/* This file is part of: */
/* GODOT ENGINE */
/* https://godotengine.org */
/*************************************************************************/
/* Copyright (c) 2022-present Péter Magyar. */
/* Copyright (c) 2014-2022 Godot Engine contributors (cf. AUTHORS.md). */
/* Copyright (c) 2007-2022 Juan Linietsky, Ariel Manzur. */
/* 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 */
@ -36,13 +35,13 @@
#include "bit_array.h"
#include "core/math/math_funcs.h"
#include "core/string/ustring.h"
#include "core/ustring.h"
BitArray::BitArray(uint32_t p_initial_size_in_bit) {
resize_in_bits(p_initial_size_in_bit);
}
BitArray::BitArray(const PoolByteArray &p_bytes) :
BitArray::BitArray(const Vector<uint8_t> &p_bytes) :
bytes(p_bytes) {
}
@ -74,8 +73,6 @@ void BitArray::store_bits(int p_bit_offset, uint64_t p_value, int p_bits) {
int bit_offset = p_bit_offset;
uint64_t val = p_value;
PoolByteArray::Write w = bytes.write();
while (bits > 0) {
const int bits_to_write = MIN(bits, 8 - bit_offset % 8);
const int bits_to_jump = bit_offset % 8;
@ -87,18 +84,16 @@ void BitArray::store_bits(int p_bit_offset, uint64_t p_value, int p_bits) {
uint8_t byte_clear = 0xFF >> bits_to_jump;
byte_clear = byte_clear << (bits_to_jump + bits_to_skip);
byte_clear = ~(byte_clear >> bits_to_skip);
w[byte_offset] &= byte_clear;
bytes.write[byte_offset] &= byte_clear;
// Now we can continue to write bits
w[byte_offset] |= (val & 0xFF) << bits_to_jump;
bytes.write[byte_offset] |= (val & 0xFF) << bits_to_jump;
bits -= bits_to_write;
bit_offset += bits_to_write;
val >>= bits_to_write;
}
w.release();
}
uint64_t BitArray::read_bits(int p_bit_offset, int p_bits) const {
@ -109,8 +104,7 @@ uint64_t BitArray::read_bits(int p_bit_offset, int p_bits) const {
int bit_offset = p_bit_offset;
uint64_t val = 0;
PoolByteArray::Read r = bytes.read();
const uint8_t *bytes_ptr = r.ptr();
const uint8_t *bytes_ptr = bytes.ptr();
int val_bits_to_jump = 0;
while (bits > 0) {
@ -130,15 +124,11 @@ uint64_t BitArray::read_bits(int p_bit_offset, int p_bits) const {
val_bits_to_jump += bits_to_read;
}
r.release();
return val;
}
void BitArray::zero() {
if (bytes.size() > 0) {
PoolByteArray::Write w = bytes.write();
memset(w.ptr(), 0, sizeof(uint8_t) * bytes.size());
w.release();
memset(bytes.ptrw(), 0, sizeof(uint8_t) * bytes.size());
}
}

View File

@ -1,16 +1,12 @@
#ifndef BITARRAY_H
#define BITARRAY_H
/*************************************************************************/
/* bit_array.h */
/*************************************************************************/
/* This file is part of: */
/* PANDEMONIUM ENGINE */
/* https://github.com/Relintai/pandemonium_engine */
/* This file is part of: */
/* GODOT ENGINE */
/* https://godotengine.org */
/*************************************************************************/
/* Copyright (c) 2022-present Péter Magyar. */
/* Copyright (c) 2014-2022 Godot Engine contributors (cf. AUTHORS.md). */
/* Copyright (c) 2007-2022 Juan Linietsky, Ariel Manzur. */
/* 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 */
@ -36,21 +32,24 @@
@author AndreaCatania
*/
#include "core/variant/variant.h"
#include "core/vector.h"
#ifndef BITARRAY_H
#define BITARRAY_H
class BitArray {
PoolByteArray bytes;
Vector<uint8_t> bytes;
public:
BitArray() = default;
BitArray(uint32_t p_initial_size_in_bit);
BitArray(const PoolByteArray &p_bytes);
BitArray(const Vector<uint8_t> &p_bytes);
const PoolByteArray &get_bytes() const {
const Vector<uint8_t> &get_bytes() const {
return bytes;
}
PoolByteArray &get_bytes_mut() {
Vector<uint8_t> &get_bytes_mut() {
return bytes;
}

View File

@ -6,19 +6,17 @@ def configure(env):
pass
def get_doc_classes():
return [
"DataBuffer",
"SceneDiff",
"Interpolator",
"NetworkedController",
"SceneSynchronizer",
]
def get_doc_path():
return "doc_classes"
def get_doc_classes():
return [
"SceneSynchronizer",
"NetworkedController",
"DataBuffer",
]
def is_enabled():
return True

View File

@ -1,13 +1,12 @@
/*************************************************************************/
/* data_buffer.cpp */
/*************************************************************************/
/* This file is part of: */
/* PANDEMONIUM ENGINE */
/* https://github.com/Relintai/pandemonium_engine */
/* This file is part of: */
/* GODOT ENGINE */
/* https://godotengine.org */
/*************************************************************************/
/* Copyright (c) 2022-present Péter Magyar. */
/* Copyright (c) 2014-2022 Godot Engine contributors (cf. AUTHORS.md). */
/* Copyright (c) 2007-2022 Juan Linietsky, Ariel Manzur. */
/* 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 */
@ -36,18 +35,48 @@
#include "data_buffer.h"
#include "core/io/marshalls.h"
#include "scene_synchronizer_debugger.h"
// Beware that this macros was written to make sure nested function call doesn't add debug calls,
// making the log unreadable.
#ifdef DEBUG_ENABLED
#define DEB_WRITE(dt, compression, input) \
if (debug_enabled) { \
SceneSynchronizerDebugger::singleton()->databuffer_write(dt, compression, input); \
}
#define DEB_READ(dt, compression, input) \
if (debug_enabled) { \
SceneSynchronizerDebugger::singleton()->databuffer_read(dt, compression, input); \
}
#define DEB_DISABLE \
const bool was_debug_enabled = debug_enabled; \
debug_enabled = false;
#define DEB_ENABLE debug_enabled = was_debug_enabled;
#else
#define DEB_WRITE(dt, compression, input)
#define DEB_READ(dt, compression, input)
#define DEB_DISABLE
#define DEB_ENABLE
#endif
// TODO improve the allocation mechanism.
void DataBuffer::_bind_methods() {
BIND_ENUM_CONSTANT(DATA_TYPE_BOOL);
BIND_ENUM_CONSTANT(DATA_TYPE_INT);
BIND_ENUM_CONSTANT(DATA_TYPE_UINT);
BIND_ENUM_CONSTANT(DATA_TYPE_REAL);
BIND_ENUM_CONSTANT(DATA_TYPE_POSITIVE_UNIT_REAL);
BIND_ENUM_CONSTANT(DATA_TYPE_UNIT_REAL);
BIND_ENUM_CONSTANT(DATA_TYPE_VECTOR2);
BIND_ENUM_CONSTANT(DATA_TYPE_NORMALIZED_VECTOR2);
BIND_ENUM_CONSTANT(DATA_TYPE_VECTOR3);
BIND_ENUM_CONSTANT(DATA_TYPE_NORMALIZED_VECTOR3);
BIND_ENUM_CONSTANT(DATA_TYPE_VARIANT);
BIND_ENUM_CONSTANT(COMPRESSION_LEVEL_0);
BIND_ENUM_CONSTANT(COMPRESSION_LEVEL_1);
@ -58,6 +87,7 @@ void DataBuffer::_bind_methods() {
ClassDB::bind_method(D_METHOD("add_bool", "value"), &DataBuffer::add_bool);
ClassDB::bind_method(D_METHOD("add_int", "value", "compression_level"), &DataBuffer::add_int, DEFVAL(COMPRESSION_LEVEL_1));
ClassDB::bind_method(D_METHOD("add_uint", "value", "compression_level"), &DataBuffer::add_uint, DEFVAL(COMPRESSION_LEVEL_1));
ClassDB::bind_method(D_METHOD("add_real", "value", "compression_level"), &DataBuffer::add_real, DEFVAL(COMPRESSION_LEVEL_1));
ClassDB::bind_method(D_METHOD("add_positive_unit_real", "value", "compression_level"), &DataBuffer::add_positive_unit_real, DEFVAL(COMPRESSION_LEVEL_1));
ClassDB::bind_method(D_METHOD("add_unit_real", "value", "compression_level"), &DataBuffer::add_unit_real, DEFVAL(COMPRESSION_LEVEL_1));
@ -69,7 +99,9 @@ void DataBuffer::_bind_methods() {
ClassDB::bind_method(D_METHOD("read_bool"), &DataBuffer::read_bool);
ClassDB::bind_method(D_METHOD("read_int", "compression_level"), &DataBuffer::read_int, DEFVAL(COMPRESSION_LEVEL_1));
ClassDB::bind_method(D_METHOD("read_uint", "compression_level"), &DataBuffer::read_uint, DEFVAL(COMPRESSION_LEVEL_1));
ClassDB::bind_method(D_METHOD("read_real", "compression_level"), &DataBuffer::read_real, DEFVAL(COMPRESSION_LEVEL_1));
ClassDB::bind_method(D_METHOD("read_positive_unit_real", "compression_level"), &DataBuffer::read_positive_unit_real, DEFVAL(COMPRESSION_LEVEL_1));
ClassDB::bind_method(D_METHOD("read_unit_real", "compression_level"), &DataBuffer::read_unit_real, DEFVAL(COMPRESSION_LEVEL_1));
ClassDB::bind_method(D_METHOD("read_vector2", "compression_level"), &DataBuffer::read_vector2, DEFVAL(COMPRESSION_LEVEL_1));
ClassDB::bind_method(D_METHOD("read_normalized_vector2", "compression_level"), &DataBuffer::read_normalized_vector2, DEFVAL(COMPRESSION_LEVEL_1));
@ -79,6 +111,7 @@ void DataBuffer::_bind_methods() {
ClassDB::bind_method(D_METHOD("skip_bool"), &DataBuffer::skip_bool);
ClassDB::bind_method(D_METHOD("skip_int", "compression_level"), &DataBuffer::skip_int, DEFVAL(COMPRESSION_LEVEL_1));
ClassDB::bind_method(D_METHOD("skip_uint", "compression_level"), &DataBuffer::skip_uint, DEFVAL(COMPRESSION_LEVEL_1));
ClassDB::bind_method(D_METHOD("skip_real", "compression_level"), &DataBuffer::skip_real, DEFVAL(COMPRESSION_LEVEL_1));
ClassDB::bind_method(D_METHOD("skip_unit_real", "compression_level"), &DataBuffer::skip_unit_real, DEFVAL(COMPRESSION_LEVEL_1));
ClassDB::bind_method(D_METHOD("skip_vector2", "compression_level"), &DataBuffer::skip_vector2, DEFVAL(COMPRESSION_LEVEL_1));
@ -88,6 +121,7 @@ void DataBuffer::_bind_methods() {
ClassDB::bind_method(D_METHOD("get_bool_size"), &DataBuffer::get_bool_size);
ClassDB::bind_method(D_METHOD("get_int_size", "compression_level"), &DataBuffer::get_int_size, DEFVAL(COMPRESSION_LEVEL_1));
ClassDB::bind_method(D_METHOD("get_uint_size", "compression_level"), &DataBuffer::get_uint_size, DEFVAL(COMPRESSION_LEVEL_1));
ClassDB::bind_method(D_METHOD("get_real_size", "compression_level"), &DataBuffer::get_real_size, DEFVAL(COMPRESSION_LEVEL_1));
ClassDB::bind_method(D_METHOD("get_unit_real_size", "compression_level"), &DataBuffer::get_unit_real_size, DEFVAL(COMPRESSION_LEVEL_1));
ClassDB::bind_method(D_METHOD("get_vector2_size", "compression_level"), &DataBuffer::get_vector2_size, DEFVAL(COMPRESSION_LEVEL_1));
@ -97,6 +131,7 @@ void DataBuffer::_bind_methods() {
ClassDB::bind_method(D_METHOD("read_bool_size"), &DataBuffer::read_bool_size);
ClassDB::bind_method(D_METHOD("read_int_size", "compression_level"), &DataBuffer::read_int_size, DEFVAL(COMPRESSION_LEVEL_1));
ClassDB::bind_method(D_METHOD("read_uint_size", "compression_level"), &DataBuffer::read_uint_size, DEFVAL(COMPRESSION_LEVEL_1));
ClassDB::bind_method(D_METHOD("read_real_size", "compression_level"), &DataBuffer::read_real_size, DEFVAL(COMPRESSION_LEVEL_1));
ClassDB::bind_method(D_METHOD("read_unit_real_size", "compression_level"), &DataBuffer::read_unit_real_size, DEFVAL(COMPRESSION_LEVEL_1));
ClassDB::bind_method(D_METHOD("read_vector2_size", "compression_level"), &DataBuffer::read_vector2_size, DEFVAL(COMPRESSION_LEVEL_1));
@ -124,6 +159,22 @@ DataBuffer::DataBuffer(const BitArray &p_buffer) :
is_reading(true),
buffer(p_buffer) {}
void DataBuffer::copy(const DataBuffer &p_other) {
metadata_size = p_other.metadata_size;
bit_offset = p_other.bit_offset;
bit_size = p_other.bit_size;
is_reading = p_other.is_reading;
buffer = p_other.buffer;
}
void DataBuffer::copy(const BitArray &p_buffer) {
metadata_size = 0;
bit_offset = 0;
bit_size = p_buffer.size_in_bits();
is_reading = true;
buffer = p_buffer;
}
void DataBuffer::begin_write(int p_metadata_size) {
CRASH_COND_MSG(p_metadata_size < 0, "Metadata size can't be negative");
metadata_size = p_metadata_size;
@ -178,6 +229,8 @@ void DataBuffer::begin_read() {
bool DataBuffer::add_bool(bool p_input) {
ERR_FAIL_COND_V(is_reading == true, p_input);
DEB_WRITE(DATA_TYPE_BOOL, COMPRESSION_LEVEL_0, p_input);
const int bits = get_bit_taken(DATA_TYPE_BOOL, COMPRESSION_LEVEL_0);
make_room_in_bits(bits);
@ -198,12 +251,17 @@ bool DataBuffer::read_bool() {
const int bits = get_bit_taken(DATA_TYPE_BOOL, COMPRESSION_LEVEL_0);
const bool d = buffer.read_bits(bit_offset, bits);
bit_offset += bits;
DEB_READ(DATA_TYPE_BOOL, COMPRESSION_LEVEL_0, d);
return d;
}
int64_t DataBuffer::add_int(int64_t p_input, CompressionLevel p_compression_level) {
ERR_FAIL_COND_V(is_reading == true, p_input);
DEB_WRITE(DATA_TYPE_INT, p_compression_level, p_input);
const int bits = get_bit_taken(DATA_TYPE_INT, p_compression_level);
int64_t value = p_input;
@ -220,6 +278,73 @@ int64_t DataBuffer::add_int(int64_t p_input, CompressionLevel p_compression_leve
}
make_room_in_bits(bits);
// Safely convert int to uint.
uint64_t uvalue;
memcpy(&uvalue, &value, sizeof(uint64_t));
buffer.store_bits(bit_offset, uvalue, bits);
bit_offset += bits;
#ifdef DEBUG_ENABLED
// Can't never happen because the buffer size is correctly handled.
CRASH_COND((metadata_size + bit_size) > buffer.size_in_bits() && bit_offset > buffer.size_in_bits());
#endif
return value;
}
int64_t DataBuffer::read_int(CompressionLevel p_compression_level) {
ERR_FAIL_COND_V(is_reading == false, 0);
const int bits = get_bit_taken(DATA_TYPE_INT, p_compression_level);
const uint64_t uvalue = buffer.read_bits(bit_offset, bits);
bit_offset += bits;
int64_t value;
memcpy(&value, &uvalue, sizeof(uint64_t));
if (bits == 8) {
DEB_READ(DATA_TYPE_INT, p_compression_level, static_cast<int8_t>(value));
return static_cast<int8_t>(value);
} else if (bits == 16) {
DEB_READ(DATA_TYPE_INT, p_compression_level, static_cast<int16_t>(value));
return static_cast<int16_t>(value);
} else if (bits == 32) {
DEB_READ(DATA_TYPE_INT, p_compression_level, static_cast<int32_t>(value));
return static_cast<int32_t>(value);
} else {
DEB_READ(DATA_TYPE_INT, p_compression_level, value);
return value;
}
}
uint64_t DataBuffer::add_uint(uint64_t p_input, CompressionLevel p_compression_level) {
ERR_FAIL_COND_V(is_reading == true, p_input);
DEB_WRITE(DATA_TYPE_UINT, p_compression_level, p_input);
const int bits = get_bit_taken(DATA_TYPE_UINT, p_compression_level);
uint64_t value = p_input;
// Clamp the value to the max that the bit can store.
if (bits == 8) {
value = MIN(value, UINT8_MAX);
} else if (bits == 16) {
value = MIN(value, UINT16_MAX);
} else if (bits == 32) {
value = MIN(value, UINT32_MAX);
} else {
// Nothing to do here
}
make_room_in_bits(bits);
buffer.store_bits(bit_offset, value, bits);
bit_offset += bits;
@ -228,39 +353,27 @@ int64_t DataBuffer::add_int(int64_t p_input, CompressionLevel p_compression_leve
CRASH_COND((metadata_size + bit_size) > buffer.size_in_bits() && bit_offset > buffer.size_in_bits());
#endif
if (bits == 8) {
return static_cast<int8_t>(value);
} else if (bits == 16) {
return static_cast<int16_t>(value);
} else if (bits == 32) {
return static_cast<int32_t>(value);
} else {
return value;
}
return value;
}
int64_t DataBuffer::read_int(CompressionLevel p_compression_level) {
uint64_t DataBuffer::read_uint(CompressionLevel p_compression_level) {
ERR_FAIL_COND_V(is_reading == false, 0);
const int bits = get_bit_taken(DATA_TYPE_INT, p_compression_level);
const int bits = get_bit_taken(DATA_TYPE_UINT, p_compression_level);
const uint64_t value = buffer.read_bits(bit_offset, bits);
bit_offset += bits;
if (bits == 8) {
return static_cast<int8_t>(value);
} else if (bits == 16) {
return static_cast<int16_t>(value);
} else if (bits == 32) {
return static_cast<int32_t>(value);
} else {
return static_cast<int64_t>(value);
}
DEB_READ(DATA_TYPE_UINT, p_compression_level, value);
return value;
}
double DataBuffer::add_real(double p_input, CompressionLevel p_compression_level) {
ERR_FAIL_COND_V(is_reading == true, p_input);
DEB_WRITE(DATA_TYPE_REAL, p_compression_level, p_input);
// Clamp the input value according to the compression level
// Minifloat (compression level 0) have a special bias
const int exponent_bits = get_exponent_bits(p_compression_level);
@ -331,7 +444,11 @@ double DataBuffer::read_real(CompressionLevel p_compression_level) {
const double mantissa_scale = Math::pow(2.0, exponent <= 0 ? mantissa_bits - 1 : mantissa_bits);
const double mantissa = exponent <= 0 ? integer_mantissa / mantissa_scale / Math::pow(2.0, exponent) : integer_mantissa / mantissa_scale + 0.5;
return ldexp(sign ? -mantissa : mantissa, exponent);
const double value = ldexp(sign ? -mantissa : mantissa, exponent);
DEB_READ(DATA_TYPE_REAL, p_compression_level, value);
return value;
}
real_t DataBuffer::add_positive_unit_real(real_t p_input, CompressionLevel p_compression_level) {
@ -340,7 +457,9 @@ real_t DataBuffer::add_positive_unit_real(real_t p_input, CompressionLevel p_com
#endif
ERR_FAIL_COND_V(is_reading == true, p_input);
const int bits = get_bit_taken(DATA_TYPE_UNIT_REAL, p_compression_level);
DEB_WRITE(DATA_TYPE_POSITIVE_UNIT_REAL, p_compression_level, p_input);
const int bits = get_bit_taken(DATA_TYPE_POSITIVE_UNIT_REAL, p_compression_level);
const double max_value = static_cast<double>(~(UINT64_MAX << bits));
@ -361,19 +480,25 @@ real_t DataBuffer::add_positive_unit_real(real_t p_input, CompressionLevel p_com
real_t DataBuffer::read_positive_unit_real(CompressionLevel p_compression_level) {
ERR_FAIL_COND_V(is_reading == false, 0.0);
const int bits = get_bit_taken(DATA_TYPE_UNIT_REAL, p_compression_level);
const int bits = get_bit_taken(DATA_TYPE_POSITIVE_UNIT_REAL, p_compression_level);
const double max_value = static_cast<double>(~(UINT64_MAX << bits));
const uint64_t compressed_val = buffer.read_bits(bit_offset, bits);
bit_offset += bits;
return decompress_unit_float(compressed_val, max_value);
const real_t value = decompress_unit_float(compressed_val, max_value);
DEB_READ(DATA_TYPE_POSITIVE_UNIT_REAL, p_compression_level, value);
return value;
}
real_t DataBuffer::add_unit_real(real_t p_input, CompressionLevel p_compression_level) {
ERR_FAIL_COND_V(is_reading == true, p_input);
DEB_WRITE(DATA_TYPE_UNIT_REAL, p_compression_level, p_input);
const real_t added_real = add_positive_unit_real(ABS(p_input), p_compression_level);
const int bits_for_sign = 1;
@ -399,7 +524,11 @@ real_t DataBuffer::read_unit_real(CompressionLevel p_compression_level) {
const bool is_negative = buffer.read_bits(bit_offset, bits_for_sign);
bit_offset += bits_for_sign;
return is_negative ? -value : value;
const real_t ret = is_negative ? -value : value;
DEB_READ(DATA_TYPE_UNIT_REAL, p_compression_level, ret);
return ret;
}
Vector2 DataBuffer::add_vector2(Vector2 p_input, CompressionLevel p_compression_level) {
@ -413,9 +542,16 @@ Vector2 DataBuffer::add_vector2(Vector2 p_input, CompressionLevel p_compression_
}
#endif
DEB_WRITE(DATA_TYPE_VECTOR2, p_compression_level, p_input);
DEB_DISABLE
Vector2 r;
r[0] = add_real(p_input[0], p_compression_level);
r[1] = add_real(p_input[1], p_compression_level);
DEB_ENABLE
return r;
}
@ -430,24 +566,35 @@ Vector2 DataBuffer::read_vector2(CompressionLevel p_compression_level) {
}
#endif
DEB_DISABLE
Vector2 r;
r[0] = read_real(p_compression_level);
r[1] = read_real(p_compression_level);
DEB_ENABLE
DEB_READ(DATA_TYPE_VECTOR2, p_compression_level, r);
return r;
}
Vector2 DataBuffer::add_normalized_vector2(Vector2 p_input, CompressionLevel p_compression_level) {
const uint32_t is_not_zero = p_input.length_squared() > CMP_EPSILON;
#ifdef DEBUG_ENABLED
ERR_FAIL_COND_V(p_input.is_normalized() == false, p_input);
ERR_FAIL_COND_V_MSG(p_input.is_normalized() == false && is_not_zero, p_input, "[FATAL] The encoding failed because this function expects a normalized vector.");
#endif
ERR_FAIL_COND_V(is_reading == true, p_input);
DEB_WRITE(DATA_TYPE_NORMALIZED_VECTOR2, p_compression_level, p_input);
const int bits = get_bit_taken(DATA_TYPE_NORMALIZED_VECTOR2, p_compression_level);
const int bits_for_the_angle = bits - 1;
const int bits_for_zero = 1;
const double angle = p_input.angle();
const uint32_t is_not_zero = p_input.length_squared() > CMP_EPSILON;
const double max_value = static_cast<double>(~(UINT64_MAX << bits_for_the_angle));
@ -487,7 +634,10 @@ Vector2 DataBuffer::read_normalized_vector2(CompressionLevel p_compression_level
const real_t x = Math::cos(decompressed_angle);
const real_t y = Math::sin(decompressed_angle);
return Vector2(x, y) * is_not_zero;
const Vector2 value = Vector2(x, y) * is_not_zero;
DEB_READ(DATA_TYPE_NORMALIZED_VECTOR2, p_compression_level, value);
return value;
}
Vector3 DataBuffer::add_vector3(Vector3 p_input, CompressionLevel p_compression_level) {
@ -501,10 +651,17 @@ Vector3 DataBuffer::add_vector3(Vector3 p_input, CompressionLevel p_compression_
}
#endif
DEB_WRITE(DATA_TYPE_VECTOR3, p_compression_level, p_input);
DEB_DISABLE
Vector3 r;
r[0] = add_real(p_input[0], p_compression_level);
r[1] = add_real(p_input[1], p_compression_level);
r[2] = add_real(p_input[2], p_compression_level);
DEB_ENABLE
return r;
}
@ -519,34 +676,56 @@ Vector3 DataBuffer::read_vector3(CompressionLevel p_compression_level) {
}
#endif
DEB_DISABLE
Vector3 r;
r[0] = read_real(p_compression_level);
r[1] = read_real(p_compression_level);
r[2] = read_real(p_compression_level);
DEB_ENABLE
DEB_READ(DATA_TYPE_VECTOR3, p_compression_level, r);
return r;
}
Vector3 DataBuffer::add_normalized_vector3(Vector3 p_input, CompressionLevel p_compression_level) {
#ifdef DEBUG_ENABLED
ERR_FAIL_COND_V(p_input.is_normalized() == false, p_input);
const uint32_t is_not_zero = p_input.length_squared() > CMP_EPSILON;
ERR_FAIL_COND_V_MSG(p_input.is_normalized() == false && is_not_zero, p_input, "[FATAL] This function expects a normalized vector.");
#endif
ERR_FAIL_COND_V(is_reading == true, p_input);
DEB_WRITE(DATA_TYPE_NORMALIZED_VECTOR3, p_compression_level, p_input);
DEB_DISABLE
const real_t x_axis = add_unit_real(p_input.x, p_compression_level);
const real_t y_axis = add_unit_real(p_input.y, p_compression_level);
const real_t z_axis = add_unit_real(p_input.z, p_compression_level);
DEB_ENABLE
return Vector3(x_axis, y_axis, z_axis);
}
Vector3 DataBuffer::read_normalized_vector3(CompressionLevel p_compression_level) {
ERR_FAIL_COND_V(is_reading == false, Vector3());
DEB_DISABLE
const real_t x_axis = read_unit_real(p_compression_level);
const real_t y_axis = read_unit_real(p_compression_level);
const real_t z_axis = read_unit_real(p_compression_level);
return Vector3(x_axis, y_axis, z_axis);
DEB_ENABLE
const Vector3 value(x_axis, y_axis, z_axis);
DEB_READ(DATA_TYPE_NORMALIZED_VECTOR3, p_compression_level, value);
return value;
}
Variant DataBuffer::add_variant(const Variant &p_input) {
@ -556,6 +735,8 @@ Variant DataBuffer::add_variant(const Variant &p_input) {
// Get the variant size.
int len = 0;
DEB_WRITE(DATA_TYPE_VARIANT, COMPRESSION_LEVEL_0, p_input);
const Error len_err = encode_variant(
p_input,
nullptr,
@ -580,7 +761,7 @@ Variant DataBuffer::add_variant(const Variant &p_input) {
const Error write_err = encode_variant(
p_input,
buffer.get_bytes_mut().write().ptr() + (bit_offset / 8),
buffer.get_bytes_mut().ptrw() + (bit_offset / 8),
len,
false);
@ -611,7 +792,7 @@ Variant DataBuffer::read_variant() {
const Error read_err = decode_variant(
ret,
buffer.get_bytes_mut().write().ptr() + (bit_offset / 8),
buffer.get_bytes().ptr() + (bit_offset / 8),
buffer.size_in_bytes() - (bit_offset / 8),
&len,
false);
@ -623,6 +804,8 @@ Variant DataBuffer::read_variant() {
bit_offset += len * 8;
DEB_READ(DATA_TYPE_VARIANT, COMPRESSION_LEVEL_0, ret);
return ret;
}
@ -640,11 +823,21 @@ void DataBuffer::skip_int(CompressionLevel p_compression) {
skip(bits);
}
void DataBuffer::skip_uint(CompressionLevel p_compression) {
const int bits = get_uint_size(p_compression);
skip(bits);
}
void DataBuffer::skip_real(CompressionLevel p_compression) {
const int bits = get_real_size(p_compression);
skip(bits);
}
void DataBuffer::skip_positive_unit_real(CompressionLevel p_compression) {
const int bits = get_positive_unit_real_size(p_compression);
skip(bits);
}
void DataBuffer::skip_unit_real(CompressionLevel p_compression) {
const int bits = get_unit_real_size(p_compression);
skip(bits);
@ -678,10 +871,18 @@ int DataBuffer::get_int_size(CompressionLevel p_compression) const {
return DataBuffer::get_bit_taken(DATA_TYPE_INT, p_compression);
}
int DataBuffer::get_uint_size(CompressionLevel p_compression) const {
return DataBuffer::get_bit_taken(DATA_TYPE_UINT, p_compression);
}
int DataBuffer::get_real_size(CompressionLevel p_compression) const {
return DataBuffer::get_bit_taken(DATA_TYPE_REAL, p_compression);
}
int DataBuffer::get_positive_unit_real_size(CompressionLevel p_compression) const {
return DataBuffer::get_bit_taken(DATA_TYPE_POSITIVE_UNIT_REAL, p_compression);
}
int DataBuffer::get_unit_real_size(CompressionLevel p_compression) const {
return DataBuffer::get_bit_taken(DATA_TYPE_UNIT_REAL, p_compression);
}
@ -714,12 +915,24 @@ int DataBuffer::read_int_size(CompressionLevel p_compression) {
return bits;
}
int DataBuffer::read_uint_size(CompressionLevel p_compression) {
const int bits = get_uint_size(p_compression);
skip(bits);
return bits;
}
int DataBuffer::read_real_size(CompressionLevel p_compression) {
const int bits = get_real_size(p_compression);
skip(bits);
return bits;
}
int DataBuffer::read_positive_unit_real_size(CompressionLevel p_compression) {
const int bits = get_positive_unit_real_size(p_compression);
skip(bits);
return bits;
}
int DataBuffer::read_unit_real_size(CompressionLevel p_compression) {
const int bits = get_unit_real_size(p_compression);
skip(bits);
@ -767,7 +980,7 @@ int DataBuffer::read_variant_size() {
const Error read_err = decode_variant(
ret,
buffer.get_bytes_mut().write().ptr() + (bit_offset / 8),
buffer.get_bytes().ptr() + (bit_offset / 8),
buffer.size_in_bytes() - (bit_offset / 8),
&len,
false);
@ -802,6 +1015,21 @@ int DataBuffer::get_bit_taken(DataType p_data_type, CompressionLevel p_compressi
CRASH_NOW_MSG("Compression level not supported!");
}
} break;
case DATA_TYPE_UINT: {
switch (p_compression) {
case COMPRESSION_LEVEL_0:
return 64;
case COMPRESSION_LEVEL_1:
return 32;
case COMPRESSION_LEVEL_2:
return 16;
case COMPRESSION_LEVEL_3:
return 8;
default:
// Unreachable
CRASH_NOW_MSG("Compression level not supported!");
}
} break;
case DATA_TYPE_REAL: {
return get_mantissa_bits(p_compression) +
get_exponent_bits(p_compression);
@ -844,16 +1072,7 @@ int DataBuffer::get_bit_taken(DataType p_data_type, CompressionLevel p_compressi
return get_bit_taken(DATA_TYPE_REAL, p_compression) * 3;
} break;
case DATA_TYPE_NORMALIZED_VECTOR3: {
switch (p_compression) {
case CompressionLevel::COMPRESSION_LEVEL_0:
return 11 * 3;
case CompressionLevel::COMPRESSION_LEVEL_1:
return 10 * 3;
case CompressionLevel::COMPRESSION_LEVEL_2:
return 8 * 3;
case CompressionLevel::COMPRESSION_LEVEL_3:
return 6 * 3;
}
return get_bit_taken(DATA_TYPE_UNIT_REAL, p_compression) * 3;
} break;
case DATA_TYPE_VARIANT: {
ERR_FAIL_V_MSG(0, "The variant size is dynamic and can't be know at compile time.");
@ -869,6 +1088,14 @@ int DataBuffer::get_bit_taken(DataType p_data_type, CompressionLevel p_compressi
}
int DataBuffer::get_mantissa_bits(CompressionLevel p_compression) {
#ifndef REAL_T_IS_DOUBLE
// Fallback to compression level 1 if real_t is float
if (p_compression == DataBuffer::COMPRESSION_LEVEL_0) {
WARN_PRINT_ONCE("Compression level 0 is not supported for a binary compiled with single precision float. Falling back to compression level 1");
p_compression = DataBuffer::COMPRESSION_LEVEL_1;
}
#endif
// https://en.wikipedia.org/wiki/IEEE_754#Basic_and_interchange_formats
switch (p_compression) {
case CompressionLevel::COMPRESSION_LEVEL_0:
@ -887,6 +1114,14 @@ int DataBuffer::get_mantissa_bits(CompressionLevel p_compression) {
}
int DataBuffer::get_exponent_bits(CompressionLevel p_compression) {
#ifndef REAL_T_IS_DOUBLE
// Fallback to compression level 1 if real_t is float
if (p_compression == DataBuffer::COMPRESSION_LEVEL_0) {
WARN_PRINT_ONCE("Compression level 0 is not supported for a binary compiled with single precision float. Falling back to compression level 1");
p_compression = DataBuffer::COMPRESSION_LEVEL_1;
}
#endif
// https://en.wikipedia.org/wiki/IEEE_754#Basic_and_interchange_formats
switch (p_compression) {
case CompressionLevel::COMPRESSION_LEVEL_0:

View File

@ -1,16 +1,12 @@
#ifndef INPUT_BUFFER_H
#define INPUT_BUFFER_H
/*************************************************************************/
/* data_buffer.h */
/*************************************************************************/
/* This file is part of: */
/* PANDEMONIUM ENGINE */
/* https://github.com/Relintai/pandemonium_engine */
/* This file is part of: */
/* GODOT ENGINE */
/* https://godotengine.org */
/*************************************************************************/
/* Copyright (c) 2022-present Péter Magyar. */
/* Copyright (c) 2014-2022 Godot Engine contributors (cf. AUTHORS.md). */
/* Copyright (c) 2007-2022 Juan Linietsky, Ariel Manzur. */
/* 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 */
@ -36,10 +32,13 @@
@author AndreaCatania
*/
#include "core/object/class_db.h"
#include "core/class_db.h"
#include "bit_array.h"
#ifndef INPUT_BUFFER_H
#define INPUT_BUFFER_H
class DataBuffer : public Object {
GDCLASS(DataBuffer, Object);
@ -47,6 +46,7 @@ public:
enum DataType {
DATA_TYPE_BOOL,
DATA_TYPE_INT,
DATA_TYPE_UINT,
DATA_TYPE_REAL,
DATA_TYPE_POSITIVE_UNIT_REAL,
DATA_TYPE_UNIT_REAL,
@ -147,6 +147,10 @@ private:
bool is_reading = false;
BitArray buffer;
#if DEBUG_ENABLED
bool debug_enabled = true;
#endif
public:
static void _bind_methods();
@ -154,6 +158,9 @@ public:
DataBuffer(const DataBuffer &p_other);
DataBuffer(const BitArray &p_buffer);
void copy(const DataBuffer &p_other);
void copy(const BitArray &p_buffer);
const BitArray &get_buffer() const {
return buffer;
}
@ -204,6 +211,12 @@ public:
/// Parse the next data as int.
int64_t read_int(CompressionLevel p_compression_level);
/// Add the next data as uint
uint64_t add_uint(uint64_t p_input, CompressionLevel p_compression_level);
/// Parse the next data as uint.
uint64_t read_uint(CompressionLevel p_compression_level);
/// Add a real into the buffer. Depending on the compression level is possible
/// to store different range level.
/// The fractional part has a precision of ~0.3%
@ -238,7 +251,7 @@ public:
real_t read_unit_real(CompressionLevel p_compression_level);
/// Add a vector2 into the buffer.
/// Note: This kind of vector occupies more space than the normalized version.
/// Note: This kind of vector occupies more space than the normalized verison.
/// Consider use a normalized vector to save bandwidth if possible.
///
/// Returns the decompressed vector so both the client and the peers can use
@ -260,7 +273,7 @@ public:
Vector2 read_normalized_vector2(CompressionLevel p_compression_level);
/// Add a vector3 into the buffer.
/// Note: This kind of vector occupies more space than the normalized version.
/// Note: This kind of vector occupies more space than the normalized verison.
/// Consider use a normalized vector to save bandwidth if possible.
///
/// Returns the decompressed vector so both the client and the peers can use
@ -294,7 +307,9 @@ public:
void skip_bool();
void skip_int(CompressionLevel p_compression);
void skip_uint(CompressionLevel p_compression);
void skip_real(CompressionLevel p_compression);
void skip_positive_unit_real(CompressionLevel p_compression);
void skip_unit_real(CompressionLevel p_compression);
void skip_vector2(CompressionLevel p_compression);
void skip_normalized_vector2(CompressionLevel p_compression);
@ -305,7 +320,9 @@ public:
int get_bool_size() const;
int get_int_size(CompressionLevel p_compression) const;
int get_uint_size(CompressionLevel p_compression) const;
int get_real_size(CompressionLevel p_compression) const;
int get_positive_unit_real_size(CompressionLevel p_compression) const;
int get_unit_real_size(CompressionLevel p_compression) const;
int get_vector2_size(CompressionLevel p_compression) const;
int get_normalized_vector2_size(CompressionLevel p_compression) const;
@ -316,7 +333,9 @@ public:
int read_bool_size();
int read_int_size(CompressionLevel p_compression);
int read_uint_size(CompressionLevel p_compression);
int read_real_size(CompressionLevel p_compression);
int read_positive_unit_real_size(CompressionLevel p_compression);
int read_unit_real_size(CompressionLevel p_compression);
int read_vector2_size(CompressionLevel p_compression);
int read_normalized_vector2_size(CompressionLevel p_compression);

View File

@ -0,0 +1,30 @@
from os import listdir
from os.path import isfile, join, isdir, exists
def create_debugger_header():
f = open("__generated__debugger_ui.h", "w", encoding="utf-8")
f.write("#pragma once\n")
f.write("\n")
f.write("/// This is a generated file by `cpplize_debugger.py`, executed by `SCsub`.\n")
f.write("/// \n")
f.write("/// DO NOT EDIT this header.\n")
f.write("/// If you want to modify this python code, you can simply change `debugger.py`\n")
f.write("/// During the next compilation this header will be updated.\n")
f.write("/// \n")
f.write("/// HOWEVER! The file will not be copied into the `bin` folder unless you remove the\n")
f.write("/// existing `bin/debugger.py` first; this algorithm prevents destroying eventual\n")
f.write("/// changes made on that file.\n")
f.write("\n")
f.write("static const char __debugger_ui_code[] = R\"TheCodeRKS(")
size = 0
with open('./debugger_ui/debugger.py', encoding="utf-8") as deb_f:
for l in deb_f.readlines():
l_utf8 = l.encode('utf-8')
size += len(l_utf8)
f.write(l);
f.write(" )TheCodeRKS\";\n")
f.write("static unsigned int __debugger_ui_code_size = "+str(size)+";\n")
f.close()

View File

@ -0,0 +1,389 @@
#!/usr/bin/env python3
# PySimpleGUI call reference manual: https://www.pysimplegui.org/en/latest/call%20reference/
import PySimpleGUI as sg
import json
from os import listdir
from os.path import isfile, join, isdir, exists
# ------------------------------------------------------------------------------------------- Config
font_size = 9
# ---------------------------------------------------------------------------------------- Utilities
def load_json(path):
if exists(path):
with open(path) as f:
return json.load(f)
return {}
def compare_arrays(arr_a, arr_b):
if len(arr_a) != len(arr_b):
return False
for i in range(len(arr_a)):
if arr_a[i] != arr_b[i]:
return False
return True
def repeat_characters(char, n):
s = ""
for i in range(0, n):
s += char
return s
# Fetches the directories containing the frame info.
directories = [f for f in listdir("./") if (isdir(join("./", f)) and len(listdir(join("./", f))) > 0)]
# ----------------------------------------------------------------------------------------------- UI
def create_layout():
# Fetch the frame count from the dirs.
frame_count = -1
frames_iterations = {}
frames_description = {}
for dir in directories:
dir_path = join("./", dir)
file_names = [f for f in listdir(dir_path) if isfile(join(dir_path, f))]
for file_name in file_names:
file_extension_index = file_name.find("fd-")
if file_extension_index == 0:
# This file contains information about the frame.
# Extract the frame index
frame_iteration = file_name.count("@") + 1
if frame_iteration > 1:
frame_index = int(file_name[3:file_name.index("@")])
else:
frame_index = int(file_name[3:file_name.index(".json")])
if frame_index in frames_iterations:
frames_iterations[frame_index] = max(frame_iteration, frames_iterations[frame_index])
else:
frames_iterations[frame_index] = frame_iteration
frame_count = max(frame_count, frame_index)
file_path = join(dir_path, file_name)
file_json = load_json(file_path)
if "frame_summary" in file_json:
if frame_index in frames_description:
frames_description[frame_index] += " " + file_json["frame_summary"]
else:
frames_description[frame_index] = file_json["frame_summary"]
else:
frames_description[frame_index] = ""
frame_count += 1
# --- UI - Compose timeline ---
frame_list_values = []
for frame_index in range(frame_count):
frame_description = frames_description[frame_index]
frame_iterations = frames_iterations[frame_index]
for i in range(0, frame_iterations):
if i == 0:
frame_list_values.append("# " + str(frame_index) + " - " + frame_description)
else:
frame_list_values.append("# " + str(frame_index) + " @" + str(i + 1) + " - " + frame_description)
# Release this array, we don't need anylonger.
frames_description.clear()
frames_list = sg.Listbox(frame_list_values, key="FRAMES_LIST", size = [45, 30], enable_events=True, horizontal_scroll=True, select_mode=sg.LISTBOX_SELECT_MODE_BROWSE)
frames_list = sg.Frame("Frames", layout=[[frames_list]], vertical_alignment="top")
# --- UI - Compose frame detail ---
# Node list
nodes_list_listbox = sg.Listbox([], key="NODE_LIST", size = [45, 0], enable_events=True, horizontal_scroll=True, expand_y=True, expand_x=True, select_mode=sg.LISTBOX_SELECT_MODE_MULTIPLE)
nodes_list_listbox = sg.Frame("Nodes", layout=[[nodes_list_listbox]], vertical_alignment="top", expand_y=True, expand_x=True);
# Selected nodes title.
node_tile_txt = sg.Text("", key="FRAME_SUMMARY", font="Any, " + str(font_size - 1), justification="left", border_width=1, text_color="dark red")
# States table
table_status_header = ["name"]
table_status_widths = [30]
for dir in directories:
table_status_header.append("Begin ["+dir+"]")
table_status_widths.append(30)
for dir in directories:
table_status_header.append("End ("+dir+")")
table_status_widths.append(30)
table_status = sg.Table([], table_status_header, key="TABLE_STATUS", justification='left', auto_size_columns=False, col_widths=table_status_widths, vertical_scroll_only=False, num_rows=38)
table_status = sg.Frame("States", layout=[[table_status]], vertical_alignment="top")
# Messages table
tables_logs = []
for dir in directories:
tables_logs.append(sg.Frame("Log: " + dir + " Iteration: ", key=dir+"_FRAME_TABLE_LOG", layout=[[sg.Table([], [" #", "Log"], key=dir+"_TABLE_LOG", justification='left', auto_size_columns=False, col_widths=[4, 70], vertical_scroll_only=False, num_rows=25)]], vertical_alignment="top"))
logs = sg.Frame("Messages", layout=[tables_logs], vertical_alignment="top")
# --- UI - Main Window ---
layout = [
[
sg.Frame("", [[frames_list], [nodes_list_listbox]], vertical_alignment="top", expand_y=True),
sg.Frame("Frame detail", [[node_tile_txt], [table_status], [logs]], key="FRAME_FRAME_DETAIL", vertical_alignment="top")
],
[
sg.Button("Exit")
]
]
return layout
# ------------------------------------------------------------------------------------ Create window
window = sg.Window(title="Network Synchronizer Debugger.", layout=create_layout(), margins=(5, 5), font="Any, " + str(font_size), resizable=True)
# ----------------------------------------------------------------------------------- Event handling
frame_data = {}
nodes_list = []
selected_nodes = []
used_frame_iteration = {}
while True:
event, event_values = window.read()
# EVENT: Close the program.
if event == "Exit" or event == sg.WIN_CLOSED:
window.close()
break
# EVENT: Show frame
if event == "FRAMES_LIST":
window["NODE_LIST"].update([])
if event_values["FRAMES_LIST"] != []:
frame_description = event_values["FRAMES_LIST"][0]
if "@" in frame_description:
frame_iteration = int(frame_description[frame_description.index(" @") + 2:frame_description.index(" - ")])
selected_frame_index = int(frame_description[2:frame_description.index(" @")])
else:
frame_iteration = 1
selected_frame_index = int(frame_description[2:frame_description.index(" - ")])
print("Show frame: ", selected_frame_index, " Iteration: ", frame_iteration)
frame_data = {}
nodes_list = []
used_frame_iteration = {}
for dir in directories:
used_frame_iteration[dir] = frame_iteration
frame_file_path = join("./", dir, "fd-" + str(selected_frame_index) + repeat_characters("@", frame_iteration - 1) + ".json")
if not exists(frame_file_path):
print("The path: ", frame_file_path, " was not found. Falling back to no iteration path.")
used_frame_iteration[dir] = 1
frame_file_path = join("./", dir, "fd-" + str(selected_frame_index) + ".json")
print("Path: ", frame_file_path)
if exists(frame_file_path):
frame_data_json = load_json(frame_file_path)
frame_data[dir] = frame_data_json
for node_path in frame_data_json["begin_state"]:
if node_path not in nodes_list:
# Add this node to the nodelist
nodes_list.append(node_path)
for node_path in frame_data_json["end_state"]:
if node_path not in nodes_list:
# Add this node to the nodelist
nodes_list.append(node_path)
for node_path in frame_data_json["node_log"]:
if node_path not in nodes_list:
# Add this node to the nodelist
nodes_list.append(node_path)
# Update the node list.
window["NODE_LIST"].update(nodes_list)
if len(selected_nodes) == 0:
if len(nodes_list) > 0:
selected_nodes = [nodes_list[0]]
else:
selected_nodes = []
window["NODE_LIST"].set_value(selected_nodes)
event = "NODE_LIST"
event_values = {"NODE_LIST": selected_nodes}
# EVENT: Show node data
if event == "NODE_LIST":
window["FRAME_SUMMARY"].update("")
window["TABLE_STATUS"].update([])
for dir_name in directories:
window[dir_name + "_FRAME_TABLE_LOG"].update("Log: " + dir_name + "Frame: " + str(selected_frame_index) + " Iteration: " + str(used_frame_iteration[dir_name]))
window[dir_name + "_TABLE_LOG"].update([["", "[Nothing for this node]"]])
if event_values["NODE_LIST"] != []:
instances_count = len(directories)
row_size = 1 + (instances_count * 2)
# Compose the status table
states_table_values = []
states_row_colors = []
table_logs = {}
log_row_colors = {}
selected_nodes = event_values["NODE_LIST"]
for node_path in selected_nodes:
# First collects the var names
vars_names = ["***"]
for dir in directories:
if dir in frame_data:
if "begin_state" in frame_data[dir]:
if node_path in frame_data[dir]["begin_state"]:
for var_name in frame_data[dir]["begin_state"][node_path]:
if var_name not in vars_names:
vars_names.append(var_name)
if "end_state" in frame_data[dir]:
if node_path in frame_data[dir]["end_state"]:
for var_name in frame_data[dir]["end_state"][node_path]:
if var_name not in vars_names:
vars_names.append(var_name)
vars_names.append("---")
# Add those to the table.
for var_name in vars_names:
# Initializes the row.
row = [""] * row_size
row_index = len(states_table_values)
# Special rows
if var_name == "***":
# This is a special row to signal the start of a new node data
row[0] = node_path
states_table_values.append(row)
states_row_colors.append((row_index, "black"))
continue
elif var_name == "---":
# This is a special row to signal the end of the node data
states_table_values.append(row)
continue
row[0] = var_name.replace("*", "🔄")
# Set the row data.
for dir_i, dir_name in enumerate(directories):
if dir_name in frame_data:
if "begin_state" in frame_data[dir_name]:
if node_path in frame_data[dir_name]["begin_state"]:
if var_name in frame_data[dir_name]["begin_state"][node_path]:
#print(1, " + (", instances_count, " * 0) + ", dir_i)
row[1 + (instances_count * 0) + dir_i] = str(frame_data[dir_name]["begin_state"][node_path][var_name])
if "end_state" in frame_data[dir_name]:
if node_path in frame_data[dir_name]["end_state"]:
if var_name in frame_data[dir_name]["end_state"][node_path]:
#print(1, " + (", instances_count, " * 1) + ", dir_i)
row[1 + (instances_count * 1) + dir_i] = str(frame_data[dir_name]["end_state"][node_path][var_name])
# Check if different, so mark a worning.
for state_index in range(2):
for i in range(instances_count - 1):
if row[1 + (state_index*instances_count) + i + 0] != row[1 + (state_index*instances_count) + i + 1]:
row[1 + (state_index*instances_count) + i + 0] = "⚠️ " + row[1 + (state_index*instances_count) + i + 0]
row[1 + (state_index*instances_count) + i + 1] = "⚠️ " + row[1 + (state_index*instances_count) + i + 1]
states_row_colors.append((row_index, "dark salmon"))
break
states_table_values.append(row)
# Compose the log
for dir_name in directories:
if dir_name in frame_data:
if "node_log" in frame_data[dir_name]:
if node_path in frame_data[dir_name]["node_log"]:
table_logs[dir_name] = table_logs.get(dir_name, [])
log_row_colors[dir_name] = log_row_colors.get(dir_name, [])
table_logs[dir_name] += [["", node_path]]
log_row_colors[dir_name] += [(len(table_logs[dir_name]) - 1, "black")]
for val in frame_data[dir_name]["node_log"][node_path]:
# Append the log
table_logs[dir_name] += [["{:4d}".format(val["i"]), val["m"]]]
row_index = len(table_logs[dir_name]) - 1
# Check if this line should be colored
if val["m"].find("[WARNING]") == 0:
log_row_colors[dir_name] += [(row_index, "dark salmon")]
elif val["m"].find("[ERROR]") == 0:
log_row_colors[dir_name] += [(row_index, "red")]
elif val["m"].find("[WRITE]") == 0:
log_row_colors[dir_name] += [(row_index, "cadet blue")]
elif val["m"].find("[READ]") == 0:
log_row_colors[dir_name] += [(row_index, "medium aquamarine")]
table_logs[dir_name] += [["", ""]]
window["FRAME_FRAME_DETAIL"].update("Frame " + str(selected_frame_index) + " details")
window["TABLE_STATUS"].update(states_table_values, row_colors=states_row_colors)
frame_summary = ""
for dir_name in directories:
if dir_name in frame_data:
frame_summary += frame_data[dir_name]["frame_summary"]
if dir_name in table_logs:
window[dir_name + "_TABLE_LOG"].update(table_logs[dir_name], row_colors=log_row_colors[dir_name]);
# Check if write and read databuffer is the same.
for dir_name in directories:
if dir_name in frame_data and (dir_name == "nonet" or dir_name == "client"):
are_the_same = compare_arrays(frame_data[dir_name]["data_buffer_writes"], frame_data[dir_name]["data_buffer_reads"])
if not are_the_same:
if frame_summary != "":
frame_summary += "\n"
frame_summary += "[BUG] The DataBuffer written by `_collect_inputs` and read by `_controller_process` is different. Both should be exactly the same."
# Check if the server read correctly the received buffer.
if "server" in frame_data and "client" in frame_data:
are_the_same = compare_arrays(frame_data["server"]["data_buffer_reads"], frame_data["client"]["data_buffer_reads"])
if not are_the_same:
if frame_summary != "":
frame_summary += "\n"
frame_summary += "[BUG] The DataBuffer written by the client is different from the one read on the server."
# Check if the client sent the correct inputs to the server.
if "client" in frame_data:
for other_frame_index, is_similar in frame_data["client"]["are_inputs_different_results"].items():
other_frame_index = int(other_frame_index)
other_file_path = join("./", "client", "fd-" + str(other_frame_index) + ".json")
other_frame_json = load_json(other_file_path)
if "data_buffer_reads" in other_frame_json:
is_really_similar = compare_arrays(other_frame_json["data_buffer_reads"], frame_data["client"]["data_buffer_reads"])
if is_really_similar != is_similar:
if frame_summary != "":
frame_summary += "\n"
frame_summary += "[BUG] The function `_are_inputs_different` doesn't seems to work:\n"
frame_summary += " As the inputs read on the frame " + str(frame_data["client"]["frame"]) + " and " + str(other_frame_index) + " are " + ("SIMILAR" if is_really_similar else "DIFFERENT") +" but the net sync considered it "+ ("SIMILAR" if is_similar else "DIFFERENT")
window["FRAME_SUMMARY"].update(frame_summary)
# --------------------------------------------------------------------------------------------- Exit

View File

@ -0,0 +1,15 @@
# Network debugger UI
Requirements
- Install python
- Install pip
- Install PySimpleGUI: `pip install pysymplegui`
- Install Tkinter:
1. Install Tkinter app:
- Fedora: `sudo dnf install python3-tkinter`
- Ubuntu: `sudo apt-get install python3-tk`
1. Install python library: `pip install tk`
# Missing features
1. At the moment it's not supported multiple client debugging.

View File

@ -1,5 +1,5 @@
<?xml version="1.0" encoding="UTF-8" ?>
<class name="DataBuffer" inherits="Object" version="4.2">
<class name="DataBuffer" inherits="Object" version="3.5" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:noNamespaceSchemaLocation="../../../doc/class.xsd">
<brief_description>
</brief_description>
<description>
@ -48,6 +48,13 @@
<description>
</description>
</method>
<method name="add_uint">
<return type="int" />
<argument index="0" name="value" type="int" />
<argument index="1" name="compression_level" type="int" enum="DataBuffer.CompressionLevel" default="1" />
<description>
</description>
</method>
<method name="add_unit_real">
<return type="float" />
<argument index="0" name="value" type="float" />
@ -120,6 +127,12 @@
<description>
</description>
</method>
<method name="get_uint_size" qualifiers="const">
<return type="int" />
<argument index="0" name="compression_level" type="int" enum="DataBuffer.CompressionLevel" default="1" />
<description>
</description>
</method>
<method name="get_unit_real_size" qualifiers="const">
<return type="int" />
<argument index="0" name="compression_level" type="int" enum="DataBuffer.CompressionLevel" default="1" />
@ -184,6 +197,12 @@
<description>
</description>
</method>
<method name="read_positive_unit_real">
<return type="float" />
<argument index="0" name="compression_level" type="int" enum="DataBuffer.CompressionLevel" default="1" />
<description>
</description>
</method>
<method name="read_real">
<return type="float" />
<argument index="0" name="compression_level" type="int" enum="DataBuffer.CompressionLevel" default="1" />
@ -196,6 +215,18 @@
<description>
</description>
</method>
<method name="read_uint">
<return type="int" />
<argument index="0" name="compression_level" type="int" enum="DataBuffer.CompressionLevel" default="1" />
<description>
</description>
</method>
<method name="read_uint_size">
<return type="int" />
<argument index="0" name="compression_level" type="int" enum="DataBuffer.CompressionLevel" default="1" />
<description>
</description>
</method>
<method name="read_unit_real">
<return type="float" />
<argument index="0" name="compression_level" type="int" enum="DataBuffer.CompressionLevel" default="1" />
@ -276,6 +307,12 @@
<description>
</description>
</method>
<method name="skip_uint">
<return type="void" />
<argument index="0" name="compression_level" type="int" enum="DataBuffer.CompressionLevel" default="1" />
<description>
</description>
</method>
<method name="skip_unit_real">
<return type="void" />
<argument index="0" name="compression_level" type="int" enum="DataBuffer.CompressionLevel" default="1" />
@ -300,17 +337,23 @@
</constant>
<constant name="DATA_TYPE_INT" value="1" enum="DataType">
</constant>
<constant name="DATA_TYPE_REAL" value="2" enum="DataType">
<constant name="DATA_TYPE_UINT" value="2" enum="DataType">
</constant>
<constant name="DATA_TYPE_UNIT_REAL" value="4" enum="DataType">
<constant name="DATA_TYPE_REAL" value="3" enum="DataType">
</constant>
<constant name="DATA_TYPE_VECTOR2" value="5" enum="DataType">
<constant name="DATA_TYPE_POSITIVE_UNIT_REAL" value="4" enum="DataType">
</constant>
<constant name="DATA_TYPE_NORMALIZED_VECTOR2" value="6" enum="DataType">
<constant name="DATA_TYPE_UNIT_REAL" value="5" enum="DataType">
</constant>
<constant name="DATA_TYPE_VECTOR3" value="7" enum="DataType">
<constant name="DATA_TYPE_VECTOR2" value="6" enum="DataType">
</constant>
<constant name="DATA_TYPE_NORMALIZED_VECTOR3" value="8" enum="DataType">
<constant name="DATA_TYPE_NORMALIZED_VECTOR2" value="7" enum="DataType">
</constant>
<constant name="DATA_TYPE_VECTOR3" value="8" enum="DataType">
</constant>
<constant name="DATA_TYPE_NORMALIZED_VECTOR3" value="9" enum="DataType">
</constant>
<constant name="DATA_TYPE_VARIANT" value="10" enum="DataType">
</constant>
<constant name="COMPRESSION_LEVEL_0" value="0" enum="CompressionLevel">
</constant>

View File

@ -1,62 +0,0 @@
<?xml version="1.0" encoding="UTF-8" ?>
<class name="Interpolator" inherits="Object" version="4.2">
<brief_description>
</brief_description>
<description>
</description>
<tutorials>
</tutorials>
<methods>
<method name="epoch_insert">
<return type="void" />
<argument index="0" name="var_id" type="int" />
<argument index="1" name="value" type="Variant" />
<description>
</description>
</method>
<method name="get_last_pop_epoch" qualifiers="const">
<return type="int" />
<description>
</description>
</method>
<method name="pop_epoch">
<return type="Array" />
<argument index="0" name="epoch" type="int" />
<argument index="1" name="arg1" type="float" />
<description>
</description>
</method>
<method name="register_variable">
<return type="int" />
<argument index="0" name="default" type="Variant" />
<argument index="1" name="fallback" type="int" enum="Interpolator.Fallback" />
<description>
</description>
</method>
<method name="set_variable_custom_interpolator">
<return type="void" />
<argument index="0" name="var_id" type="int" />
<argument index="1" name="object" type="Object" />
<argument index="2" name="function_name" type="StringName" />
<description>
</description>
</method>
<method name="set_variable_default">
<return type="void" />
<argument index="0" name="var_id" type="int" />
<argument index="1" name="default" type="Variant" />
<description>
</description>
</method>
</methods>
<constants>
<constant name="FALLBACK_INTERPOLATE" value="0" enum="Fallback">
</constant>
<constant name="FALLBACK_DEFAULT" value="1" enum="Fallback">
</constant>
<constant name="FALLBACK_NEW_OR_NEAREST" value="3" enum="Fallback">
</constant>
<constant name="FALLBACK_OLD_OR_NEAREST" value="2" enum="Fallback">
</constant>
</constants>
</class>

View File

@ -1,5 +1,5 @@
<?xml version="1.0" encoding="UTF-8" ?>
<class name="NetworkedController" inherits="Node" version="4.2">
<class name="NetworkedController" inherits="Node" version="3.5" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:noNamespaceSchemaLocation="../../../doc/class.xsd">
<brief_description>
</brief_description>
<description>
@ -10,7 +10,9 @@
<method name="_apply_epoch" qualifiers="virtual">
<return type="void" />
<argument index="0" name="delta" type="float" />
<argument index="1" name="interpolated_data" type="Array" />
<argument index="1" name="interpolation_alpha" type="float" />
<argument index="2" name="past_buffer" type="DataBuffer" />
<argument index="3" name="future_buffer" type="DataBuffer" />
<description>
</description>
</method>
@ -47,19 +49,6 @@
<description>
</description>
</method>
<method name="_parse_epoch_data" qualifiers="virtual">
<return type="void" />
<argument index="0" name="interpolator" type="Interpolator" />
<argument index="1" name="buffer" type="DataBuffer" />
<description>
</description>
</method>
<method name="_setup_interpolator" qualifiers="virtual">
<return type="void" />
<argument index="0" name="interpolator" type="Interpolator" />
<description>
</description>
</method>
<method name="get_current_input_id" qualifiers="const">
<return type="int" />
<description>
@ -92,7 +81,7 @@
</method>
<method name="player_get_pretended_delta" qualifiers="const">
<return type="float" />
<argument index="0" name="physics_ticks_per_seconds" type="int" />
<argument index="0" name="iterations_per_seconds" type="int" />
<description>
</description>
</method>
@ -112,21 +101,17 @@
</method>
</methods>
<members>
<member name="doll_connection_stats_frame_span" type="int" setter="set_doll_connection_stats_frame_span" getter="get_doll_connection_stats_frame_span" default="30">
<member name="doll_connection_stats_frame_span" type="int" setter="set_doll_connection_stats_frame_span" getter="get_doll_connection_stats_frame_span" default="60">
</member>
<member name="doll_epoch_batch_sync_rate" type="float" setter="set_doll_epoch_batch_sync_rate" getter="get_doll_epoch_batch_sync_rate" default="0.25">
<member name="doll_interpolation_max_overshot" type="float" setter="set_doll_interpolation_max_overshot" getter="get_doll_interpolation_max_overshot" default="0.2">
</member>
<member name="doll_epoch_collect_rate" type="int" setter="set_doll_epoch_collect_rate" getter="get_doll_epoch_collect_rate" default="1">
<member name="doll_max_frames_delay" type="int" setter="set_doll_max_frames_delay" getter="get_doll_max_frames_delay" default="25">
</member>
<member name="doll_interpolation_max_speedup" type="float" setter="set_doll_interpolation_max_speedup" getter="get_doll_interpolation_max_speedup" default="0.2">
<member name="doll_min_frames_delay" type="int" setter="set_doll_min_frames_delay" getter="get_doll_min_frames_delay" default="0">
</member>
<member name="doll_max_delay" type="int" setter="set_doll_max_delay" getter="get_doll_max_delay" default="60">
<member name="doll_net_sensitivity" type="float" setter="set_doll_net_sensitivity" getter="get_doll_net_sensitivity" default="0.21">
</member>
<member name="doll_max_frames_delay" type="int" setter="set_doll_max_frames_delay" getter="get_doll_max_frames_delay" default="5">
</member>
<member name="doll_min_frames_delay" type="int" setter="set_doll_min_frames_delay" getter="get_doll_min_frames_delay" default="2">
</member>
<member name="doll_net_sensitivity" type="float" setter="set_doll_net_sensitivity" getter="get_doll_net_sensitivity" default="0.2">
<member name="doll_sync_rate" type="int" setter="set_doll_sync_rate" getter="get_doll_sync_rate" default="30">
</member>
<member name="input_storage_size" type="int" setter="set_player_input_storage_size" getter="get_player_input_storage_size" default="180">
</member>
@ -134,18 +119,24 @@
</member>
<member name="max_redundant_inputs" type="int" setter="set_max_redundant_inputs" getter="get_max_redundant_inputs" default="5">
</member>
<member name="min_frames_delay" type="int" setter="set_min_frames_delay" getter="get_min_frames_delay" default="2">
<member name="min_frames_delay" type="int" setter="set_min_frames_delay" getter="get_min_frames_delay" default="1">
</member>
<member name="net_sensitivity" type="float" setter="set_net_sensitivity" getter="get_net_sensitivity" default="0.1">
</member>
<member name="network_traced_frames" type="int" setter="set_network_traced_frames" getter="get_network_traced_frames" default="120">
</member>
<member name="server_controlled" type="bool" setter="set_server_controlled" getter="get_server_controlled" default="false">
</member>
<member name="tick_acceleration" type="float" setter="set_tick_acceleration" getter="get_tick_acceleration" default="2.0">
</member>
<member name="tick_speedup_notification_delay" type="float" setter="set_tick_speedup_notification_delay" getter="get_tick_speedup_notification_delay" default="0.33">
</member>
</members>
<signals>
<signal name="controller_reset">
<description>
</description>
</signal>
<signal name="doll_sync_paused">
<description>
</description>

View File

@ -1,13 +0,0 @@
<?xml version="1.0" encoding="UTF-8" ?>
<class name="SceneDiff" inherits="Object" version="4.2">
<brief_description>
</brief_description>
<description>
</description>
<tutorials>
</tutorials>
<methods>
</methods>
<constants>
</constants>
</class>

View File

@ -1,6 +1,7 @@
<?xml version="1.0" encoding="UTF-8" ?>
<class name="SceneSynchronizer" inherits="Node" version="4.2">
<class name="SceneSynchronizer" inherits="Node" version="3.5" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:noNamespaceSchemaLocation="../../../doc/class.xsd">
<brief_description>
The `SceneSynchronizer` is used to synchronize all the peers using server authoritative networking model.
</brief_description>
<description>
</description>
@ -45,6 +46,13 @@
<description>
</description>
</method>
<method name="find_action_id" qualifiers="const">
<return type="int" />
<argument index="0" name="node" type="Node" />
<argument index="1" name="event_name" type="String" />
<description>
</description>
</method>
<method name="force_state_notify">
<return type="void" />
<description>
@ -71,7 +79,7 @@
<method name="get_variable_id">
<return type="int" />
<argument index="0" name="node" type="Node" />
<argument index="1" name="variable" type="StringName" />
<argument index="1" name="variable" type="String" />
<description>
</description>
</method>
@ -122,6 +130,24 @@
<description>
</description>
</method>
<method name="register_action">
<return type="int" />
<argument index="0" name="node" type="Node" />
<argument index="1" name="action_func" type="String" />
<argument index="2" name="action_encoding_func" type="String" />
<argument index="3" name="can_client_trigger" type="bool" default="false" />
<argument index="4" name="wait_server_validation" type="bool" default="false" />
<argument index="5" name="server_action_validation_func" type="String" default="&quot;&quot;" />
<description>
Register an new action.
`node` The node that owns the event
`action_func` The function that is triggered when the event is executed.
`action_encoding_func` The function called to definte the validation encoding.
`can_client_trigger` If true this `Action` can be triggered on client.
`wait_server_validation` If true the event will be emitted locally only if the server validates it.
`server_action_validation_func` The validation function, must return a boolean.
</description>
</method>
<method name="register_node">
<return type="int" />
<argument index="0" name="node" type="Node" />
@ -131,15 +157,15 @@
<method name="register_process">
<return type="void" />
<argument index="0" name="node" type="Node" />
<argument index="1" name="function" type="StringName" />
<argument index="1" name="function" type="String" />
<description>
</description>
</method>
<method name="register_variable">
<return type="void" />
<argument index="0" name="node" type="Node" />
<argument index="1" name="variable" type="StringName" />
<argument index="2" name="on_change_notify" type="StringName" default="@&quot;&quot;" />
<argument index="1" name="variable" type="String" />
<argument index="2" name="on_change_notify" type="String" default="&quot;&quot;" />
<argument index="3" name="flags" type="int" enum="NetEventFlag" default="17" />
<description>
</description>
@ -172,7 +198,7 @@
<method name="set_skip_rewinding">
<return type="void" />
<argument index="0" name="node" type="Node" />
<argument index="1" name="variable" type="StringName" />
<argument index="1" name="variable" type="String" />
<argument index="2" name="skip_rewinding" type="bool" />
<description>
</description>
@ -204,13 +230,37 @@
<method name="track_variable_changes">
<return type="void" />
<argument index="0" name="node" type="Node" />
<argument index="1" name="variable" type="StringName" />
<argument index="1" name="variable" type="String" />
<argument index="2" name="object" type="Object" />
<argument index="3" name="method" type="StringName" />
<argument index="3" name="method" type="String" />
<argument index="4" name="flags" type="int" enum="NetEventFlag" default="17" />
<description>
</description>
</method>
<method name="trigger_action">
<return type="void" />
<argument index="0" name="node" type="Node" />
<argument index="1" name="action_id" type="int" />
<argument index="2" name="arguments" type="Array" default="[ ]" />
<argument index="3" name="recipients_peers" type="PoolIntArray" default="PoolIntArray( )" />
<description>
Trigger an action.
This action can be triggered by the client, only if it was registered with `can_client_trigger = true`.
Note: If you pass a recipient the action is notified only to that peer; if you leave it unset the action will be propagated to all the peers. Generally you never specify the `recipients`. In any case, only the server can use this feature.
</description>
</method>
<method name="trigger_action_by_name">
<return type="void" />
<argument index="0" name="node" type="Node" />
<argument index="1" name="event_name" type="String" />
<argument index="2" name="arguments" type="Array" default="[ ]" />
<argument index="3" name="recipients_peers" type="PoolIntArray" default="PoolIntArray( )" />
<description>
Trigger an action.
This action can be triggered by the client, only if it was registered with `can_client_trigger = true`.
Note: If you pass a recipient the action is notified only to that peer; if you leave it unset the action will be propagated to all the peers. Generally you never specify the `recipients`. In any case, only the server can use this feature.
</description>
</method>
<method name="unregister_node">
<return type="void" />
<argument index="0" name="node" type="Node" />
@ -220,28 +270,32 @@
<method name="unregister_process">
<return type="void" />
<argument index="0" name="node" type="Node" />
<argument index="1" name="function" type="StringName" />
<argument index="1" name="function" type="String" />
<description>
</description>
</method>
<method name="unregister_variable">
<return type="void" />
<argument index="0" name="node" type="Node" />
<argument index="1" name="variable" type="StringName" />
<argument index="1" name="variable" type="String" />
<description>
</description>
</method>
<method name="untrack_variable_changes">
<return type="void" />
<argument index="0" name="node" type="Node" />
<argument index="1" name="variable" type="StringName" />
<argument index="1" name="variable" type="String" />
<argument index="2" name="object" type="Object" />
<argument index="3" name="method" type="StringName" />
<argument index="3" name="method" type="String" />
<description>
</description>
</method>
</methods>
<members>
<member name="actions_redundancy" type="int" setter="set_actions_redundancy" getter="get_actions_redundancy" default="3">
</member>
<member name="actions_resend_time" type="float" setter="set_actions_resend_time" getter="get_actions_resend_time" default="0.0333333">
</member>
<member name="comparison_float_tolerance" type="float" setter="set_comparison_float_tolerance" getter="get_comparison_float_tolerance" default="0.001">
</member>
<member name="server_notify_state_interval" type="float" setter="set_server_notify_state_interval" getter="get_server_notify_state_interval" default="1.0">

View File

@ -0,0 +1,14 @@
#pragma once
/// Used to compile against Godot 3.x
/**
@author AndreaCatania
*/
#include "core/engine.h"
#include "scene/main/viewport.h"
#define FLOAT REAL
#define STRING_NAME STRING
#define Callable(a, b) a, b
#define ObjectID CompatObjectID
#define SNAME(a) a

View File

@ -0,0 +1,23 @@
#pragma once
/// Used to compile against Godot 3.x
/**
@author AndreaCatania
*/
#include "core/typedefs.h"
struct CompatObjectID {
uint64_t id;
CompatObjectID() :
id(0){};
CompatObjectID(uint64_t p_id) :
id(p_id){};
operator uint64_t() const { return id; }
bool is_valid() const { return id != 0; }
bool is_null() const { return id == 0; }
};

View File

@ -0,0 +1,94 @@
<?xml version="1.0" encoding="UTF-8" standalone="no"?>
<svg
inkscape:version="1.2.1 (9c6d41e410, 2022-07-14)"
sodipodi:docname="icon_networked_controller.svg"
id="svg8"
version="1.1"
width="16"
viewBox="0 0 16 16"
height="16"
xmlns:inkscape="http://www.inkscape.org/namespaces/inkscape"
xmlns:sodipodi="http://sodipodi.sourceforge.net/DTD/sodipodi-0.dtd"
xmlns="http://www.w3.org/2000/svg"
xmlns:svg="http://www.w3.org/2000/svg"
xmlns:rdf="http://www.w3.org/1999/02/22-rdf-syntax-ns#"
xmlns:cc="http://creativecommons.org/ns#"
xmlns:dc="http://purl.org/dc/elements/1.1/">
<metadata
id="metadata14">
<rdf:RDF>
<cc:Work
rdf:about="">
<dc:format>image/svg+xml</dc:format>
<dc:type
rdf:resource="http://purl.org/dc/dcmitype/StillImage" />
</cc:Work>
</rdf:RDF>
</metadata>
<defs
id="defs12" />
<sodipodi:namedview
inkscape:current-layer="svg8"
inkscape:window-maximized="1"
inkscape:window-y="0"
inkscape:window-x="0"
inkscape:cy="12.711873"
inkscape:cx="8.1047594"
inkscape:zoom="26.589315"
showgrid="false"
id="namedview10"
inkscape:window-height="1659"
inkscape:window-width="3072"
inkscape:pageshadow="2"
inkscape:pageopacity="0"
guidetolerance="10"
gridtolerance="10"
objecttolerance="10"
borderopacity="1"
bordercolor="#666666"
pagecolor="#ffffff"
inkscape:document-rotation="0"
showguides="true"
inkscape:guide-bbox="true"
inkscape:showpageshadow="2"
inkscape:pagecheckerboard="0"
inkscape:deskcolor="#d1d1d1" />
<g
id="g313"
transform="translate(0,-0.47092181)">
<circle
style="fill:#99ff55;stroke-width:0.916107"
id="path711"
cx="2.788183"
cy="8.4011793"
r="2.0042512" />
<circle
style="fill:#99ff55;stroke-width:0.720163"
id="path711-3"
cx="7.8340111"
cy="4.1984701"
r="1.5755664" />
<circle
style="fill:#99ff55;stroke-width:0.916107"
id="path711-6"
cx="13.211817"
cy="12.314689"
r="2.0042512" />
<rect
style="fill:#99ff55"
id="rect267"
width="5.6688361"
height="2.7708793"
x="8.2227888"
y="5.7568631"
transform="rotate(19.971473)" />
<rect
style="fill:#99ff55;stroke-width:0.466395"
id="rect267-7"
width="1.5560057"
height="2.1958716"
x="-0.56298721"
y="7.1034822"
transform="rotate(-40.166886)" />
</g>
</svg>

After

Width:  |  Height:  |  Size: 2.5 KiB

View File

@ -0,0 +1,400 @@
#include "input_network_encoder.h"
#include "scene_synchronizer.h"
void InputNetworkEncoder::_bind_methods() {
ClassDB::bind_method(D_METHOD("register_input", "name", "default_value", "type", "compression_level", "comparison_floating_point_precision"), &InputNetworkEncoder::register_input, DEFVAL(CMP_EPSILON));
ClassDB::bind_method(D_METHOD("find_input_id", "name"), &InputNetworkEncoder::find_input_id);
ClassDB::bind_method(D_METHOD("encode", "inputs", "buffer"), &InputNetworkEncoder::script_encode);
ClassDB::bind_method(D_METHOD("decode", "buffer"), &InputNetworkEncoder::script_decode);
ClassDB::bind_method(D_METHOD("get_defaults"), &InputNetworkEncoder::script_get_defaults);
ClassDB::bind_method(D_METHOD("are_different", "buffer_a", "buffer_b"), &InputNetworkEncoder::script_are_different);
ClassDB::bind_method(D_METHOD("count_size", "buffer"), &InputNetworkEncoder::script_count_size);
}
uint32_t InputNetworkEncoder::register_input(
const StringName &p_name,
const Variant &p_default_value,
DataBuffer::DataType p_type,
DataBuffer::CompressionLevel p_compression_level,
real_t p_comparison_floating_point_precision) {
switch (p_type) {
case DataBuffer::DATA_TYPE_BOOL:
ERR_FAIL_COND_V_MSG(p_default_value.get_type() != Variant::BOOL, UINT32_MAX, "The moveset initialization failed for" + p_name + " the specified data type is `BOOL` but the default parameter is " + itos(p_default_value.get_type()));
break;
case DataBuffer::DATA_TYPE_INT:
ERR_FAIL_COND_V_MSG(p_default_value.get_type() != Variant::INT, UINT32_MAX, "The moveset initialization failed for" + p_name + " the specified data type is `INT` but the default parameter is " + itos(p_default_value.get_type()));
break;
case DataBuffer::DATA_TYPE_UINT:
ERR_FAIL_COND_V_MSG(p_default_value.get_type() != Variant::INT, UINT32_MAX, "The moveset initialization failed for" + p_name + " the specified data type is `UINT` but the default parameter is " + itos(p_default_value.get_type()));
break;
case DataBuffer::DATA_TYPE_REAL:
ERR_FAIL_COND_V_MSG(p_default_value.get_type() != Variant::REAL, UINT32_MAX, "The moveset initialization failed for" + p_name + " the specified data type is `REAL` but the default parameter is " + itos(p_default_value.get_type()));
break;
case DataBuffer::DATA_TYPE_POSITIVE_UNIT_REAL:
ERR_FAIL_COND_V_MSG(p_default_value.get_type() != Variant::REAL, UINT32_MAX, "The moveset initialization failed for" + p_name + " the specified data type is `REAL` but the default parameter is " + itos(p_default_value.get_type()));
break;
case DataBuffer::DATA_TYPE_UNIT_REAL:
ERR_FAIL_COND_V_MSG(p_default_value.get_type() != Variant::REAL, UINT32_MAX, "The moveset initialization failed for" + p_name + " the specified data type is `REAL` but the default parameter is " + itos(p_default_value.get_type()));
break;
case DataBuffer::DATA_TYPE_VECTOR2:
ERR_FAIL_COND_V_MSG(p_default_value.get_type() != Variant::VECTOR2, UINT32_MAX, "The moveset initialization failed for" + p_name + " the specified data type is `Vector2` but the default parameter is " + itos(p_default_value.get_type()));
break;
case DataBuffer::DATA_TYPE_NORMALIZED_VECTOR2:
ERR_FAIL_COND_V_MSG(p_default_value.get_type() != Variant::VECTOR2, UINT32_MAX, "The moveset initialization failed for" + p_name + " the specified data type is `Vector2` but the default parameter is " + itos(p_default_value.get_type()));
break;
case DataBuffer::DATA_TYPE_VECTOR3:
ERR_FAIL_COND_V_MSG(p_default_value.get_type() != Variant::VECTOR3, UINT32_MAX, "The moveset initialization failed for" + p_name + " the specified data type is `Vector3` but the default parameter is " + itos(p_default_value.get_type()));
break;
case DataBuffer::DATA_TYPE_NORMALIZED_VECTOR3:
ERR_FAIL_COND_V_MSG(p_default_value.get_type() != Variant::VECTOR3, UINT32_MAX, "The moveset initialization failed for" + p_name + " the specified data type is `Vector3` but the default parameter is " + itos(p_default_value.get_type()));
break;
case DataBuffer::DATA_TYPE_VARIANT:
/* No need to check variant, anything is accepted at this point.*/
break;
};
const uint32_t index = input_info.size();
input_info.resize(input_info.size() + 1);
input_info[index].name = p_name;
input_info[index].default_value = p_default_value;
input_info[index].data_type = p_type;
input_info[index].compression_level = p_compression_level;
input_info[index].comparison_floating_point_precision = p_comparison_floating_point_precision;
return index;
}
uint32_t InputNetworkEncoder::find_input_id(const StringName &p_name) const {
for (uint32_t i = 0; i < input_info.size(); i += 1) {
if (input_info[i].name == p_name) {
return i;
}
}
return INDEX_NONE;
}
const LocalVector<NetworkedInputInfo> &InputNetworkEncoder::get_input_info() const {
return input_info;
}
void InputNetworkEncoder::encode(const LocalVector<Variant> &p_input, DataBuffer &r_buffer) const {
for (uint32_t i = 0; i < input_info.size(); i += 1) {
const NetworkedInputInfo &info = input_info[i];
#ifdef DEBUG_ENABLED
if (i < p_input.size() && info.default_value.get_type() != p_input[i].get_type() && p_input[i].get_type() != Variant::NIL) {
NET_DEBUG_ERR("During the input encoding the passed value `" + p_input[i] + "` has a different type to the expected one. Using the default value `" + info.default_value + "`.");
}
#endif
const bool is_default =
// If the input exist into the array.
i >= p_input.size() ||
// Use default if the variable type is different.
info.default_value.get_type() != p_input[i].get_type() ||
// Use default if the variable value is equal to default.
info.default_value == p_input[i];
if (info.default_value.get_type() != Variant::BOOL) {
r_buffer.add_bool(is_default);
if (!is_default) {
const Variant &pending_input = p_input[i];
switch (info.data_type) {
case DataBuffer::DATA_TYPE_BOOL:
CRASH_NOW_MSG("Boolean are handled differently. Thanks to the above IF this condition never occurs.");
break;
case DataBuffer::DATA_TYPE_UINT:
r_buffer.add_uint(pending_input.operator unsigned int(), info.compression_level);
break;
case DataBuffer::DATA_TYPE_INT:
r_buffer.add_int(pending_input.operator int(), info.compression_level);
break;
case DataBuffer::DATA_TYPE_REAL:
r_buffer.add_real(pending_input.operator real_t(), info.compression_level);
break;
case DataBuffer::DATA_TYPE_POSITIVE_UNIT_REAL:
r_buffer.add_positive_unit_real(pending_input.operator real_t(), info.compression_level);
break;
case DataBuffer::DATA_TYPE_UNIT_REAL:
r_buffer.add_unit_real(pending_input.operator real_t(), info.compression_level);
break;
case DataBuffer::DATA_TYPE_VECTOR2:
r_buffer.add_vector2(pending_input.operator Vector2(), info.compression_level);
break;
case DataBuffer::DATA_TYPE_NORMALIZED_VECTOR2:
r_buffer.add_normalized_vector2(pending_input.operator Vector2(), info.compression_level);
break;
case DataBuffer::DATA_TYPE_VECTOR3:
r_buffer.add_vector3(pending_input.operator Vector3(), info.compression_level);
break;
case DataBuffer::DATA_TYPE_NORMALIZED_VECTOR3:
r_buffer.add_normalized_vector3(pending_input.operator Vector3(), info.compression_level);
break;
case DataBuffer::DATA_TYPE_VARIANT:
r_buffer.add_variant(pending_input);
break;
};
}
} else {
// If the data is bool no need to set the default.
if (!is_default) {
r_buffer.add_bool(p_input[i].operator bool());
} else {
r_buffer.add_bool(info.default_value.operator bool());
}
}
}
}
void InputNetworkEncoder::decode(DataBuffer &p_buffer, LocalVector<Variant> &r_inputs) const {
if (r_inputs.size() < input_info.size()) {
r_inputs.resize(input_info.size());
}
for (uint32_t i = 0; i < input_info.size(); i += 1) {
const NetworkedInputInfo &info = input_info[i];
const bool is_bool = info.default_value.get_type() == Variant::BOOL;
bool is_default = false;
if (is_bool == false) {
is_default = p_buffer.read_bool();
}
if (is_default) {
r_inputs[i] = info.default_value;
} else {
switch (info.data_type) {
case DataBuffer::DATA_TYPE_BOOL:
r_inputs[i] = p_buffer.read_bool();
break;
case DataBuffer::DATA_TYPE_UINT:
r_inputs[i] = p_buffer.read_uint(info.compression_level);
break;
case DataBuffer::DATA_TYPE_INT:
r_inputs[i] = p_buffer.read_int(info.compression_level);
break;
case DataBuffer::DATA_TYPE_REAL:
r_inputs[i] = p_buffer.read_real(info.compression_level);
break;
case DataBuffer::DATA_TYPE_POSITIVE_UNIT_REAL:
r_inputs[i] = p_buffer.read_positive_unit_real(info.compression_level);
break;
case DataBuffer::DATA_TYPE_UNIT_REAL:
r_inputs[i] = p_buffer.read_unit_real(info.compression_level);
break;
case DataBuffer::DATA_TYPE_VECTOR2:
r_inputs[i] = p_buffer.read_vector2(info.compression_level);
break;
case DataBuffer::DATA_TYPE_NORMALIZED_VECTOR2:
r_inputs[i] = p_buffer.read_normalized_vector2(info.compression_level);
break;
case DataBuffer::DATA_TYPE_VECTOR3:
r_inputs[i] = p_buffer.read_vector3(info.compression_level);
break;
case DataBuffer::DATA_TYPE_NORMALIZED_VECTOR3:
r_inputs[i] = p_buffer.read_normalized_vector3(info.compression_level);
break;
case DataBuffer::DATA_TYPE_VARIANT:
r_inputs[i] = p_buffer.read_variant();
break;
};
}
}
}
void InputNetworkEncoder::reset_inputs_to_defaults(LocalVector<Variant> &r_input) const {
const uint32_t size = r_input.size() < input_info.size() ? r_input.size() : input_info.size();
for (uint32_t i = 0; i < size; i += 1) {
r_input[i] = input_info[i].default_value;
}
}
bool InputNetworkEncoder::are_different(DataBuffer &p_buffer_A, DataBuffer &p_buffer_B) const {
for (uint32_t i = 0; i < input_info.size(); i += 1) {
const NetworkedInputInfo &info = input_info[i];
const bool is_bool = info.default_value.get_type() == Variant::BOOL;
bool is_default_A = false;
bool is_default_B = false;
if (is_bool == false) {
is_default_A = p_buffer_A.read_bool();
is_default_B = p_buffer_B.read_bool();
}
bool are_equals = true;
if (is_default_A && is_default_B) {
are_equals = true;
} else {
switch (info.data_type) {
case DataBuffer::DATA_TYPE_BOOL:
are_equals = p_buffer_A.read_bool() == p_buffer_B.read_bool();
break;
case DataBuffer::DATA_TYPE_UINT:
are_equals = Math::is_equal_approx(p_buffer_A.read_uint(info.compression_level), p_buffer_B.read_uint(info.compression_level), info.comparison_floating_point_precision);
break;
case DataBuffer::DATA_TYPE_INT:
are_equals = Math::is_equal_approx(p_buffer_A.read_int(info.compression_level), p_buffer_B.read_int(info.compression_level), info.comparison_floating_point_precision);
break;
case DataBuffer::DATA_TYPE_REAL:
are_equals = Math::is_equal_approx(static_cast<real_t>(p_buffer_A.read_real(info.compression_level)), static_cast<real_t>(p_buffer_B.read_real(info.compression_level)), info.comparison_floating_point_precision);
break;
case DataBuffer::DATA_TYPE_POSITIVE_UNIT_REAL:
are_equals = Math::is_equal_approx(p_buffer_A.read_positive_unit_real(info.compression_level), p_buffer_B.read_positive_unit_real(info.compression_level), info.comparison_floating_point_precision);
break;
case DataBuffer::DATA_TYPE_UNIT_REAL:
are_equals = Math::is_equal_approx(p_buffer_A.read_unit_real(info.compression_level), p_buffer_B.read_unit_real(info.compression_level), info.comparison_floating_point_precision);
break;
case DataBuffer::DATA_TYPE_VECTOR2:
are_equals = SceneSynchronizer::compare(p_buffer_A.read_vector2(info.compression_level), p_buffer_B.read_vector2(info.compression_level), info.comparison_floating_point_precision);
break;
case DataBuffer::DATA_TYPE_NORMALIZED_VECTOR2:
are_equals = SceneSynchronizer::compare(p_buffer_A.read_normalized_vector2(info.compression_level), p_buffer_B.read_normalized_vector2(info.compression_level), info.comparison_floating_point_precision);
break;
case DataBuffer::DATA_TYPE_VECTOR3:
are_equals = SceneSynchronizer::compare(p_buffer_A.read_vector3(info.compression_level), p_buffer_B.read_vector3(info.compression_level), info.comparison_floating_point_precision);
break;
case DataBuffer::DATA_TYPE_NORMALIZED_VECTOR3:
are_equals = SceneSynchronizer::compare(p_buffer_A.read_normalized_vector3(info.compression_level), p_buffer_B.read_normalized_vector3(info.compression_level), info.comparison_floating_point_precision);
break;
case DataBuffer::DATA_TYPE_VARIANT:
are_equals = SceneSynchronizer::compare(p_buffer_A.read_variant(), p_buffer_B.read_variant(), info.comparison_floating_point_precision);
break;
};
}
if (!are_equals) {
return true;
}
}
return false;
}
uint32_t InputNetworkEncoder::count_size(DataBuffer &p_buffer) const {
int size = 0;
for (uint32_t i = 0; i < input_info.size(); i += 1) {
const NetworkedInputInfo &info = input_info[i];
const bool is_bool = info.default_value.get_type() == Variant::BOOL;
if (is_bool) {
// The bool data.
size += p_buffer.read_bool_size();
} else {
// The default marker
const bool is_default = p_buffer.read_bool();
size += p_buffer.get_bool_size();
if (is_default == false) {
// Non default data set the actual data, so we need to count
// the size.
switch (info.data_type) {
case DataBuffer::DATA_TYPE_BOOL:
CRASH_NOW_MSG("This can't ever happen, as the bool is already handled.");
break;
case DataBuffer::DATA_TYPE_UINT:
size += p_buffer.read_uint_size(info.compression_level);
break;
case DataBuffer::DATA_TYPE_INT:
size += p_buffer.read_int_size(info.compression_level);
break;
case DataBuffer::DATA_TYPE_REAL:
size += p_buffer.read_real_size(info.compression_level);
break;
case DataBuffer::DATA_TYPE_POSITIVE_UNIT_REAL:
size += p_buffer.read_positive_unit_real_size(info.compression_level);
break;
case DataBuffer::DATA_TYPE_UNIT_REAL:
size += p_buffer.read_unit_real_size(info.compression_level);
break;
case DataBuffer::DATA_TYPE_VECTOR2:
size += p_buffer.read_vector2_size(info.compression_level);
break;
case DataBuffer::DATA_TYPE_NORMALIZED_VECTOR2:
size += p_buffer.read_normalized_vector2_size(info.compression_level);
break;
case DataBuffer::DATA_TYPE_VECTOR3:
size += p_buffer.read_vector3_size(info.compression_level);
break;
case DataBuffer::DATA_TYPE_NORMALIZED_VECTOR3:
size += p_buffer.read_normalized_vector3_size(info.compression_level);
break;
case DataBuffer::DATA_TYPE_VARIANT:
size += p_buffer.read_variant_size();
break;
};
}
}
}
return size;
}
void InputNetworkEncoder::script_encode(const Array &p_inputs, Object *r_buffer) const {
ERR_FAIL_COND(r_buffer == nullptr);
DataBuffer *db = Object::cast_to<DataBuffer>(r_buffer);
ERR_FAIL_COND(db == nullptr);
LocalVector<Variant> inputs;
inputs.resize(p_inputs.size());
for (int i = 0; i < p_inputs.size(); i += 1) {
inputs[i] = p_inputs[i];
}
encode(inputs, *db);
}
Array InputNetworkEncoder::script_decode(Object *p_buffer) const {
ERR_FAIL_COND_V(p_buffer == nullptr, Array());
DataBuffer *db = Object::cast_to<DataBuffer>(p_buffer);
ERR_FAIL_COND_V(db == nullptr, Array());
LocalVector<Variant> inputs;
decode(*db, inputs);
Array out;
out.resize(inputs.size());
for (uint32_t i = 0; i < inputs.size(); i += 1) {
out[i] = inputs[i];
}
return out;
}
Array InputNetworkEncoder::script_get_defaults() const {
LocalVector<Variant> inputs;
inputs.resize(input_info.size());
reset_inputs_to_defaults(inputs);
Array out;
out.resize(inputs.size());
for (uint32_t i = 0; i < inputs.size(); i += 1) {
out[i] = inputs[i];
}
return out;
}
bool InputNetworkEncoder::script_are_different(Object *p_buffer_A, Object *p_buffer_B) const {
ERR_FAIL_COND_V(p_buffer_A == nullptr, true);
DataBuffer *db_A = Object::cast_to<DataBuffer>(p_buffer_A);
ERR_FAIL_COND_V(db_A == nullptr, false);
ERR_FAIL_COND_V(p_buffer_B == nullptr, true);
DataBuffer *db_B = Object::cast_to<DataBuffer>(p_buffer_B);
ERR_FAIL_COND_V(db_B == nullptr, true);
return are_different(*db_A, *db_B);
}
uint32_t InputNetworkEncoder::script_count_size(Object *p_buffer) const {
ERR_FAIL_COND_V(p_buffer == nullptr, 0);
DataBuffer *db = Object::cast_to<DataBuffer>(p_buffer);
ERR_FAIL_COND_V(db == nullptr, 0);
return count_size(*db);
}

View File

@ -0,0 +1,51 @@
#pragma once
#include "core/local_vector.h"
#include "core/resource.h"
#include "core/variant.h"
#include "data_buffer.h"
struct NetworkedInputInfo {
StringName name;
Variant default_value;
uint32_t index;
DataBuffer::DataType data_type;
DataBuffer::CompressionLevel compression_level;
real_t comparison_floating_point_precision;
};
class InputNetworkEncoder : public Resource {
GDCLASS(InputNetworkEncoder, Resource)
public:
constexpr static uint32_t INDEX_NONE = UINT32_MAX;
private:
LocalVector<NetworkedInputInfo> input_info;
protected:
static void _bind_methods();
public:
uint32_t register_input(
const StringName &p_name,
const Variant &p_default_value,
DataBuffer::DataType p_type,
DataBuffer::CompressionLevel p_compression_level,
real_t p_comparison_floating_point_precision = CMP_EPSILON);
uint32_t find_input_id(const StringName &p_name) const;
const LocalVector<NetworkedInputInfo> &get_input_info() const;
void encode(const LocalVector<Variant> &p_inputs, DataBuffer &r_buffer) const;
void decode(DataBuffer &p_buffer, LocalVector<Variant> &r_inputs) const;
void reset_inputs_to_defaults(LocalVector<Variant> &r_inputs) const;
bool are_different(DataBuffer &p_buffer_A, DataBuffer &p_buffer_B) const;
uint32_t count_size(DataBuffer &p_buffer) const;
void script_encode(const Array &p_inputs, Object *r_buffer) const;
Array script_decode(Object *p_buffer) const;
Array script_get_defaults() const;
bool script_are_different(Object *p_buffer_A, Object *p_buffer_B) const;
uint32_t script_count_size(Object *p_buffer) const;
};

View File

@ -1,416 +0,0 @@
/*************************************************************************/
/* interpolator.cpp */
/*************************************************************************/
/* This file is part of: */
/* PANDEMONIUM ENGINE */
/* https://github.com/Relintai/pandemonium_engine */
/*************************************************************************/
/* Copyright (c) 2022-present Péter Magyar. */
/* Copyright (c) 2014-2022 Godot Engine contributors (cf. AUTHORS.md). */
/* Copyright (c) 2007-2022 Juan Linietsky, Ariel Manzur. */
/* */
/* 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
*/
// TODO write unit tests to make sure all cases are covered.
#include "interpolator.h"
#include "core/string/ustring.h"
void Interpolator::_bind_methods() {
ClassDB::bind_method(D_METHOD("register_variable", "default", "fallback"), &Interpolator::register_variable);
ClassDB::bind_method(D_METHOD("set_variable_default", "var_id", "default"), &Interpolator::set_variable_default);
ClassDB::bind_method(D_METHOD("set_variable_custom_interpolator", "var_id", "object", "function_name"), &Interpolator::set_variable_custom_interpolator);
ClassDB::bind_method(D_METHOD("epoch_insert", "var_id", "value"), &Interpolator::epoch_insert);
ClassDB::bind_method(D_METHOD("pop_epoch", "epoch"), &Interpolator::pop_epoch);
ClassDB::bind_method(D_METHOD("get_last_pop_epoch"), &Interpolator::get_last_pop_epoch);
// TODO used to do the tests.
//ClassDB::bind_method(D_METHOD("terminate_init"), &Interpolator::terminate_init);
//ClassDB::bind_method(D_METHOD("begin_write", "epoch"), &Interpolator::begin_write);
//ClassDB::bind_method(D_METHOD("end_write"), &Interpolator::end_write);
BIND_ENUM_CONSTANT(FALLBACK_INTERPOLATE);
BIND_ENUM_CONSTANT(FALLBACK_DEFAULT);
BIND_ENUM_CONSTANT(FALLBACK_NEW_OR_NEAREST);
BIND_ENUM_CONSTANT(FALLBACK_OLD_OR_NEAREST);
}
void Interpolator::clear() {
epochs.clear();
buffer.clear();
write_position = UINT32_MAX;
}
void Interpolator::reset() {
variables.clear();
epochs.clear();
buffer.clear();
init_phase = true;
write_position = UINT32_MAX;
last_pop_epoch = 0;
}
int Interpolator::register_variable(const Variant &p_default, Fallback p_fallback) {
ERR_FAIL_COND_V_MSG(init_phase == false, -1, "You cannot add another variable at this point.");
const uint32_t id = variables.size();
variables.push_back(VariableInfo{ p_default, p_fallback, ObjectID(), StringName() });
return id;
}
void Interpolator::set_variable_default(int p_var_id, const Variant &p_default) {
ERR_FAIL_INDEX(p_var_id, int(variables.size()));
ERR_FAIL_COND(variables[p_var_id].default_value.get_type() != p_default.get_type());
variables[p_var_id].default_value = p_default;
}
void Interpolator::set_variable_custom_interpolator(int p_var_id, Object *p_object, const StringName &p_function_name) {
ERR_FAIL_COND_MSG(init_phase == false, "You cannot add another variable at this point.");
ERR_FAIL_INDEX_MSG(p_var_id, int(variables.size()), "The variable_id passed is unknown.");
variables[p_var_id].fallback = FALLBACK_CUSTOM_INTERPOLATOR;
variables[p_var_id].custom_interpolator_object = p_object->get_instance_id();
variables[p_var_id].custom_interpolator_function = p_function_name;
}
void Interpolator::terminate_init() {
init_phase = false;
}
uint32_t Interpolator::known_epochs_count() const {
return epochs.size();
}
void Interpolator::begin_write(uint32_t p_epoch) {
ERR_FAIL_COND_MSG(write_position != UINT32_MAX, "You can't call this function twice.");
ERR_FAIL_COND_MSG(init_phase, "You cannot write data while the buffer is not fully initialized, call `terminate_init`.");
// Make room for this epoch.
// Insert the epoch sorted in the buffer.
write_position = UINT32_MAX;
for (uint32_t i = 0; i < epochs.size(); i += 1) {
if (epochs[i] >= p_epoch) {
write_position = i;
break;
}
}
if (write_position < UINT32_MAX) {
if (epochs[write_position] == p_epoch) {
// This epoch already exists, nothing to do.
return;
} else {
// Make room.
epochs.push_back(UINT32_MAX);
buffer.push_back(Vector<Variant>());
// Sort the epochs.
for (int i = epochs.size() - 2; i >= int(write_position); i -= 1) {
epochs[uint32_t(i) + 1] = epochs[uint32_t(i)];
buffer[uint32_t(i) + 1] = buffer[uint32_t(i)];
}
// Init the new epoch.
epochs[write_position] = p_epoch;
buffer[write_position].clear();
buffer[write_position].resize(variables.size());
}
} else {
// No sort needed.
write_position = epochs.size();
epochs.push_back(p_epoch);
buffer.push_back(Vector<Variant>());
buffer[write_position].resize(variables.size());
}
// Set defaults.
Variant *ptr = buffer[write_position].ptrw();
for (uint32_t i = 0; i < variables.size(); i += 1) {
ptr[i] = variables[i].default_value;
}
}
void Interpolator::epoch_insert(int p_var_id, const Variant &p_value) {
ERR_FAIL_COND_MSG(write_position == UINT32_MAX, "Please call `begin_write` before.");
ERR_FAIL_INDEX_MSG(p_var_id, int(variables.size()), "The variable_id passed is unknown.");
const uint32_t var_id(p_var_id);
ERR_FAIL_COND_MSG(variables[var_id].default_value.get_type() != p_value.get_type(), "The variable: " + itos(p_var_id) + " expects the variable type: " + Variant::get_type_name(variables[var_id].default_value.get_type()) + ", and not: " + Variant::get_type_name(p_value.get_type()));
buffer[write_position].write[var_id] = p_value;
}
void Interpolator::end_write() {
ERR_FAIL_COND_MSG(write_position == UINT32_MAX, "You can't call this function before starting the epoch with `begin_write`.");
write_position = UINT32_MAX;
}
Vector<Variant> Interpolator::pop_epoch(uint32_t p_epoch, real_t p_fraction) {
ERR_FAIL_COND_V_MSG(init_phase, Vector<Variant>(), "You can't pop data if the interpolator is not fully initialized.");
ERR_FAIL_COND_V_MSG(write_position != UINT32_MAX, Vector<Variant>(), "You can't pop data while writing the epoch");
double epoch = double(p_epoch) + double(p_fraction);
// Search the epoch.
uint32_t position = UINT32_MAX;
for (uint32_t i = 0; i < epochs.size(); i += 1) {
if (static_cast<double>(epochs[i]) >= epoch) {
position = i;
break;
}
}
ObjectID cache_object_id = 0;
Object *cache_object = nullptr;
Vector<Variant> data;
if (unlikely(position == UINT32_MAX)) {
data.resize(variables.size());
Variant *ptr = data.ptrw();
if (buffer.size() == 0) {
// No data found, set all to default.
for (uint32_t i = 0; i < variables.size(); i += 1) {
ptr[i] = variables[i].default_value;
}
} else {
// No new data.
for (uint32_t i = 0; i < variables.size(); i += 1) {
switch (variables[i].fallback) {
case FALLBACK_DEFAULT:
ptr[i] = variables[i].default_value;
break;
case FALLBACK_INTERPOLATE: // No way to interpolate, so just send the nearest.
case FALLBACK_NEW_OR_NEAREST: // No new data, so send the nearest.
case FALLBACK_OLD_OR_NEAREST: // Just send the oldest, as desired.
ptr[i] = buffer[buffer.size() - 1][i];
break;
case FALLBACK_CUSTOM_INTERPOLATOR:
ptr[i] = variables[i].default_value;
if (cache_object_id != variables[i].custom_interpolator_object) {
ERR_CONTINUE_MSG(!variables[i].custom_interpolator_object, "The variable: " + itos(i) + " has a custom interpolator, but the function is invalid.");
Object *o = ObjectDB::get_instance(variables[i].custom_interpolator_object);
ERR_CONTINUE_MSG(o == nullptr, "The variable: " + itos(i) + " has a custom interpolator, but the function is invalid.");
cache_object_id = variables[i].custom_interpolator_object;
cache_object = o;
}
ptr[i] = cache_object->call(
variables[i].custom_interpolator_function,
epochs[buffer.size() - 1],
buffer[buffer.size() - 1][i],
-1,
variables[i].default_value,
0.0);
if (ptr[i].get_type() != variables[i].default_value.get_type()) {
ERR_PRINT("The variable: " + itos(i) + " custom interpolator [" + variables[i].custom_interpolator_function + "], returned a different variant type. Expected: " + Variant::get_type_name(variables[i].default_value.get_type()) + ", returned: " + Variant::get_type_name(ptr[i].get_type()));
ptr[i] = variables[i].default_value;
}
break;
}
}
}
} else if (unlikely(ABS(epochs[position] - epoch) <= CMP_EPSILON)) {
// Precise data.
data = buffer[position];
} else if (unlikely(position == 0)) {
// No old data.
data.resize(variables.size());
Variant *ptr = data.ptrw();
for (uint32_t i = 0; i < variables.size(); i += 1) {
switch (variables[i].fallback) {
case FALLBACK_DEFAULT:
ptr[i] = variables[i].default_value;
break;
case FALLBACK_INTERPOLATE: // No way to interpolate, so just send the nearest.
case FALLBACK_NEW_OR_NEAREST: // Just send the newer data as desired.
case FALLBACK_OLD_OR_NEAREST: // No old data, so send nearest.
ptr[i] = buffer[0][i];
break;
case FALLBACK_CUSTOM_INTERPOLATOR:
ptr[i] = variables[i].default_value;
if (cache_object_id != variables[i].custom_interpolator_object) {
ERR_CONTINUE_MSG(!variables[i].custom_interpolator_object, "The variable: " + itos(i) + " has a custom interpolator, but the function is invalid.");
Object *o = ObjectDB::get_instance(variables[i].custom_interpolator_object);
ERR_CONTINUE_MSG(o == nullptr, "The variable: " + itos(i) + " has a custom interpolator, but the function is invalid.");
cache_object_id = variables[i].custom_interpolator_object;
cache_object = o;
}
ptr[i] = cache_object->call(
variables[i].custom_interpolator_function,
-1,
variables[i].default_value,
epochs[0],
buffer[0][i],
1.0);
if (ptr[i].get_type() != variables[i].default_value.get_type()) {
ERR_PRINT("The variable: " + itos(i) + " custom interpolator [" + variables[i].custom_interpolator_function + "], returned a different variant type. Expected: " + Variant::get_type_name(variables[i].default_value.get_type()) + ", returned: " + Variant::get_type_name(ptr[i].get_type()));
ptr[i] = variables[i].default_value;
}
break;
}
}
} else {
// Enough data to do anything needed.
data.resize(variables.size());
Variant *ptr = data.ptrw();
for (uint32_t i = 0; i < variables.size(); i += 1) {
switch (variables[i].fallback) {
case FALLBACK_DEFAULT:
ptr[i] = variables[i].default_value;
break;
case FALLBACK_INTERPOLATE: {
const real_t delta = (epoch - double(epochs[position - 1])) / double(epochs[position] - epochs[position - 1]);
ptr[i] = interpolate(
buffer[position - 1][i],
buffer[position][i],
delta);
} break;
case FALLBACK_NEW_OR_NEAREST:
ptr[i] = buffer[position][i];
break;
case FALLBACK_OLD_OR_NEAREST:
ptr[i] = buffer[position - 1][i];
break;
case FALLBACK_CUSTOM_INTERPOLATOR: {
ptr[i] = variables[i].default_value;
if (cache_object_id != variables[i].custom_interpolator_object) {
ERR_CONTINUE_MSG(!variables[i].custom_interpolator_object, "The variable: " + itos(i) + " has a custom interpolator, but the function is invalid.");
Object *o = ObjectDB::get_instance(variables[i].custom_interpolator_object);
ERR_CONTINUE_MSG(o == nullptr, "The variable: " + itos(i) + " has a custom interpolator, but the function is invalid.");
cache_object_id = variables[i].custom_interpolator_object;
cache_object = o;
}
const real_t delta = (epoch - double(epochs[position - 1])) / double(epochs[position] - epochs[position - 1]);
ptr[i] = cache_object->call(
variables[i].custom_interpolator_function,
epochs[position - 1],
buffer[position - 1][i],
epochs[position],
buffer[position][i],
delta);
if (ptr[i].get_type() != variables[i].default_value.get_type()) {
ERR_PRINT("The variable: " + itos(i) + " custom interpolator [" + variables[i].custom_interpolator_function + "], returned a different variant type. Expected: " + Variant::get_type_name(variables[i].default_value.get_type()) + ", returned: " + Variant::get_type_name(ptr[i].get_type()));
ptr[i] = variables[i].default_value;
}
} break;
}
}
}
if (unlikely(position == UINT32_MAX)) {
if (buffer.size() > 1) {
// Remove all the elements but last. This happens when the p_epoch is
// bigger than the one already stored into the queue.
epochs[0] = epochs[buffer.size() - 1];
buffer[0] = buffer[buffer.size() - 1];
epochs.resize(1);
buffer.resize(1);
}
} else if (position >= 2) {
// TODO improve this by performing first the shifting then the resizing.
// Remove the old elements, but leave the one used to interpolate.
for (uint32_t i = 0; i < position - 1; i += 1) {
epochs.remove(0);
buffer.remove(0);
}
}
// TODO this is no more valid since I'm using the fractional part.
last_pop_epoch = MAX(p_epoch, last_pop_epoch);
return data;
}
uint32_t Interpolator::get_last_pop_epoch() const {
return last_pop_epoch;
}
uint32_t Interpolator::get_youngest_epoch() const {
if (epochs.size() <= 0) {
return UINT32_MAX;
}
return epochs[0];
}
uint32_t Interpolator::get_oldest_epoch() const {
if (epochs.size() <= 0) {
return UINT32_MAX;
}
return epochs[epochs.size() - 1];
}
uint32_t Interpolator::epochs_between_last_time_window() const {
if (epochs.size() <= 1) {
return 0;
}
return epochs[epochs.size() - 1] - epochs[epochs.size() - 2];
}
Variant Interpolator::interpolate(const Variant &p_v1, const Variant &p_v2, real_t p_delta) {
ERR_FAIL_COND_V(p_v1.get_type() != p_v2.get_type(), p_v1);
switch (p_v1.get_type()) {
case Variant::Type::INT:
return int(Math::round(Math::lerp(p_v1.operator real_t(), p_v2.operator real_t(), p_delta)));
case Variant::Type::REAL:
return Math::lerp(p_v1, p_v2, p_delta);
case Variant::Type::VECTOR2:
return p_v1.operator Vector2().linear_interpolate(p_v2.operator Vector2(), p_delta);
case Variant::Type::VECTOR2I:
return Vector2i(
int(Math::round(Math::lerp(p_v1.operator Vector2i()[0], p_v2.operator Vector2i()[0], p_delta))),
int(Math::round(Math::lerp(p_v1.operator Vector2i()[1], p_v2.operator Vector2i()[1], p_delta))));
case Variant::Type::TRANSFORM2D:
return p_v1.operator Transform2D().interpolate_with(p_v2.operator Transform2D(), p_delta);
case Variant::Type::VECTOR3:
return p_v1.operator Vector3().linear_interpolate(p_v2.operator Vector3(), p_delta);
case Variant::Type::VECTOR3I:
return Vector3i(
int(Math::round(Math::lerp(p_v1.operator Vector3i()[0], p_v2.operator Vector3i()[0], p_delta))),
int(Math::round(Math::lerp(p_v1.operator Vector3i()[1], p_v2.operator Vector3i()[1], p_delta))),
int(Math::round(Math::lerp(p_v1.operator Vector3i()[2], p_v2.operator Vector3i()[2], p_delta))));
case Variant::Type::QUATERNION:
return p_v1.operator Quaternion().slerp(p_v2.operator Quaternion(), p_delta);
case Variant::Type::BASIS:
return p_v1.operator Basis().slerp(p_v2.operator Basis(), p_delta);
case Variant::Type::TRANSFORM:
return p_v1.operator Transform().interpolate_with(p_v2.operator Transform(), p_delta);
default:
return p_delta > 0.5 ? p_v2 : p_v1;
}
}

View File

@ -1,105 +0,0 @@
#ifndef INTERPOLATOR_H
#define INTERPOLATOR_H
/*************************************************************************/
/* interpolator.h */
/*************************************************************************/
/* This file is part of: */
/* PANDEMONIUM ENGINE */
/* https://github.com/Relintai/pandemonium_engine */
/*************************************************************************/
/* Copyright (c) 2022-present Péter Magyar. */
/* Copyright (c) 2014-2022 Godot Engine contributors (cf. AUTHORS.md). */
/* Copyright (c) 2007-2022 Juan Linietsky, Ariel Manzur. */
/* */
/* 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
*/
#include "core/containers/vector.h"
#include "core/object/class_db.h"
class Interpolator : public Object {
GDCLASS(Interpolator, Object);
public:
enum Fallback {
FALLBACK_INTERPOLATE,
FALLBACK_DEFAULT,
FALLBACK_OLD_OR_NEAREST,
FALLBACK_NEW_OR_NEAREST,
FALLBACK_CUSTOM_INTERPOLATOR
};
private:
struct VariableInfo {
// TODO Do we need a name?
Variant default_value;
Fallback fallback;
ObjectID custom_interpolator_object;
StringName custom_interpolator_function;
};
LocalVector<VariableInfo> variables;
/// Epoch ids, sorted from youngest to oldest.
LocalVector<uint32_t> epochs;
/// Epoch data.
LocalVector<Vector<Variant>> buffer;
bool init_phase = true;
uint32_t write_position = UINT32_MAX;
uint32_t last_pop_epoch = 0;
static void _bind_methods();
public:
Interpolator() = default;
void clear();
void reset();
int register_variable(const Variant &p_default, Fallback p_fallback);
void set_variable_default(int p_var_id, const Variant &p_default);
void set_variable_custom_interpolator(int p_var_id, Object *p_object, const StringName &p_function_name);
void terminate_init();
/// Returns the epochs stored.
uint32_t known_epochs_count() const;
void begin_write(uint32_t p_epoch);
void epoch_insert(int p_var_id, const Variant &p_value);
void end_write();
Vector<Variant> pop_epoch(uint32_t p_epoch, real_t p_fraction);
uint32_t get_last_pop_epoch() const; // TODO do I need this? Remove if not.
uint32_t get_youngest_epoch() const;
uint32_t get_oldest_epoch() const;
/// Returns the epochs count between the two last received time window.
uint32_t epochs_between_last_time_window() const;
static Variant interpolate(const Variant &p_v1, const Variant &p_v2, real_t p_delta);
};
VARIANT_ENUM_CAST(Interpolator::Fallback);
#endif

View File

@ -0,0 +1,258 @@
#include "net_action.h"
#include "core/error_macros.h"
#include "core/os/os.h"
#include "scene/main/node.h"
#include "scene_synchronizer.h"
#include "scene_synchronizer_debugger.h"
void SenderNetAction::prepare_processor(
NetUtility::NodeData *p_nd,
NetActionId p_action_id,
const Array &p_vars) {
action_processor.action_id = p_action_id;
action_processor.nd = p_nd;
// Compress the vars so that locally we use the compressed version like remotely.
const NetActionInfo &info = p_nd->net_actions[p_action_id];
LocalVector<Variant> raw_vars;
raw_vars.resize(p_vars.size());
for (int i = 0; i < p_vars.size(); i += 1) {
raw_vars[i] = p_vars[i];
}
DataBuffer db;
db.begin_write(0);
info.network_encoder->encode(raw_vars, db);
db.begin_read();
info.network_encoder->decode(db, raw_vars);
action_processor.vars.resize(raw_vars.size());
for (int i = 0; i < p_vars.size(); i += 1) {
action_processor.vars[i] = raw_vars[i];
}
}
const NetActionInfo &SenderNetAction::get_action_info() const {
return action_processor.nd->net_actions[action_processor.action_id];
}
void SenderNetAction::client_set_executed_input_id(uint32_t p_input_id) {
peers_executed_input_id[1] = p_input_id;
}
uint32_t SenderNetAction::client_get_executed_input_id() const {
const uint32_t *input_id = peers_executed_input_id.getptr(1);
return input_id == nullptr ? UINT32_MAX : *input_id;
}
uint32_t SenderNetAction::peer_get_executed_input_id(int p_peer) const {
const uint32_t *input_id = peers_executed_input_id.getptr(p_peer);
return input_id == nullptr ? UINT32_MAX : *input_id;
}
void net_action::encode_net_action(
const LocalVector<SenderNetAction *> &p_actions,
int p_peer,
DataBuffer &r_data_buffer) {
for (uint32_t i = 0; i < p_actions.size(); i++) {
// ---------------------------------------------------------- Add a boolean to note a new action
const bool has_anotherone = true;
r_data_buffer.add_bool(has_anotherone);
// ----------------------------------------------------------------- Add the sender action token
r_data_buffer.add_uint(p_actions[i]->action_token, DataBuffer::COMPRESSION_LEVEL_1);
// ----------------------------------------------------------------------------- Add the node id
const bool uses_node_id = p_actions[i]->action_processor.nd->id != UINT32_MAX;
r_data_buffer.add_bool(uses_node_id);
if (uses_node_id) {
r_data_buffer.add_uint(p_actions[i]->action_processor.nd->id, DataBuffer::COMPRESSION_LEVEL_2);
} else {
r_data_buffer.add_variant(p_actions[i]->action_processor.nd->node->get_path());
}
// --------------------------------------------------------------------------- Add the action_id
const NetActionId action_id = p_actions[i]->action_processor.action_id;
r_data_buffer.add_uint(action_id, DataBuffer::COMPRESSION_LEVEL_2);
// -------------------------------------------------------------------------- Add executed frame
const uint32_t *executed_frame = p_actions[i]->peers_executed_input_id.getptr(p_peer);
const bool has_executed_frame = executed_frame != nullptr;
r_data_buffer.add_bool(has_executed_frame);
if (has_executed_frame) {
r_data_buffer.add_uint(*executed_frame, DataBuffer::COMPRESSION_LEVEL_1);
}
// --------------------------------------------------------------- Add the executed time changed
const bool sender_executed_time_changed =
p_actions[i]->sender_executed_time_changed &&
p_peer == p_actions[i]->sender_peer;
r_data_buffer.add_bool(sender_executed_time_changed);
if (sender_executed_time_changed) {
r_data_buffer.add_uint(p_actions[i]->triggerer_action_token, DataBuffer::COMPRESSION_LEVEL_1);
}
// --------------------------------------------------------------------------- Add the variables
LocalVector<Variant> inputs;
inputs.resize(p_actions[i]->action_processor.vars.size());
for (uint32_t u = 0; u < inputs.size(); u++) {
inputs[u] = p_actions[i]->action_processor.vars[u];
}
p_actions[i]->action_processor.nd->net_actions[action_id].network_encoder->encode(inputs, r_data_buffer);
}
const bool has_anotherone = false;
r_data_buffer.add_bool(has_anotherone);
}
void net_action::decode_net_action(
SceneSynchronizer *synchronizer,
DataBuffer &p_data_buffer,
int p_peer,
LocalVector<SenderNetAction> &r_actions) {
const int sender_peer = synchronizer->get_tree()->get_multiplayer()->get_rpc_sender_id();
LocalVector<Variant> variables;
while (p_data_buffer.get_bit_offset() < p_data_buffer.total_size()) {
// ---------------------------------------------------------- Fetch the boolean `has_anotherone`
const bool has_anotherone = p_data_buffer.read_bool();
if (!has_anotherone) {
break;
}
// --------------------------------------------------------------- Fetch the sender action token
const uint32_t action_token = p_data_buffer.read_uint(DataBuffer::COMPRESSION_LEVEL_1);
// --------------------------------------------------------------------------- Fetch the node id
const bool uses_node_id = p_data_buffer.read_bool();
// Fetch the node_data.
NetUtility::NodeData *node_data;
if (uses_node_id) {
const uint32_t node_data_id = p_data_buffer.read_uint(DataBuffer::COMPRESSION_LEVEL_2);
node_data = synchronizer->get_node_data(node_data_id);
if (node_data == nullptr) {
SceneSynchronizerDebugger::singleton()->debug_error(synchronizer, "The received action data contains a node which is not registered on this peer. NodeDataId: `" + itos(node_data_id) + "`");
continue;
}
} else {
Variant node_path = p_data_buffer.read_variant();
ERR_FAIL_COND_MSG(node_path.get_type() != Variant::NODE_PATH, "The received acts data is malformed, expected NodePath at this point.");
Node *node = synchronizer->get_node(node_path);
if (node == nullptr) {
SceneSynchronizerDebugger::singleton()->debug_error(synchronizer, String("The received action data contains a node path which is unknown: `") + node_path + "`");
continue;
}
node_data = synchronizer->find_node_data(node);
if (node_data == nullptr) {
SceneSynchronizerDebugger::singleton()->debug_error(synchronizer, String("The received action data contains a node which is not registered on this peer. NodePath: `") + node_path + "`");
continue;
}
}
// ------------------------------------------------------------------------- Fetch the action_id
const NetActionId action_id = p_data_buffer.read_uint(DataBuffer::COMPRESSION_LEVEL_2);
if (node_data->net_actions.size() <= action_id) {
SceneSynchronizerDebugger::singleton()->debug_error(synchronizer, "The received action data is malformed. This peer doesn't have the action_id (`" + itos(action_id) + "`) for the node `" + node_data->node->get_path() + "`");
continue;
}
// ------------------------------------------------------------------------ Fetch executed frame
uint32_t executed_frame = UINT32_MAX;
const bool has_executed_frame = p_data_buffer.read_bool();
if (has_executed_frame) {
executed_frame = p_data_buffer.read_uint(DataBuffer::COMPRESSION_LEVEL_1);
}
// ------------------------------------------------------------- Fetch the executed time changed
const bool sender_executed_time_changed = p_data_buffer.read_bool();
uint32_t triggerer_action_token = action_token;
if (sender_executed_time_changed) {
triggerer_action_token = p_data_buffer.read_uint(DataBuffer::COMPRESSION_LEVEL_1);
}
// ------------------------------------------------------------------------- Fetch the variables
variables.clear();
node_data->net_actions[action_id].network_encoder->decode(p_data_buffer, variables);
// This should never be triggered because the `has_anotherone` is meant to be false and stop the loop.
ERR_FAIL_COND_MSG(p_data_buffer.get_bit_offset() >= p_data_buffer.total_size(), "The received action data is malformed.");
Array arguments;
arguments.resize(variables.size());
for (uint32_t i = 0; i < variables.size(); i += 1) {
arguments[i] = variables[i];
}
const uint32_t index = r_actions.size();
r_actions.resize(index + 1);
r_actions[index].action_token = action_token;
r_actions[index].triggerer_action_token = triggerer_action_token;
r_actions[index].sender_executed_time_changed = sender_executed_time_changed;
r_actions[index].sender_peer = sender_peer;
r_actions[index].peers_executed_input_id[p_peer] = executed_frame;
r_actions[index].action_processor.nd = node_data;
r_actions[index].action_processor.action_id = action_id;
r_actions[index].action_processor.vars = arguments;
}
}
bool NetActionSenderInfo::process_received_action(uint32_t p_action_index) {
const uint64_t now = OS::get_singleton()->get_ticks_msec();
bool already_received = true;
if (last_received_action_id != UINT32_MAX) {
if (last_received_action_id < p_action_index) {
// Add all the in between ids as missing.
for (uint32_t missing_action_id = last_received_action_id + 1; missing_action_id < p_action_index; missing_action_id += 1) {
missing_actions.push_back({ missing_action_id, now });
}
last_received_action_id = p_action_index;
already_received = false;
} else if (last_received_action_id == p_action_index) {
// Already known, drop it.
already_received = true;
} else {
// Old act, check if it's a missing act.
const int64_t index = missing_actions.find({ p_action_index, 0 });
const bool known = index == -1;
if (known) {
already_received = true;
} else {
already_received = false;
missing_actions.remove_unordered(index);
}
}
} else {
last_received_action_id = p_action_index;
already_received = false;
}
return already_received;
}
void NetActionSenderInfo::check_missing_actions_and_clean_up(Node *p_owner) {
const uint64_t now = OS::get_singleton()->get_ticks_msec();
const uint64_t one_second = 1000;
for (int64_t i = int64_t(missing_actions.size()) - 1; i >= 0; i -= 1) {
if ((missing_actions[i].timestamp + one_second) <= now) {
// After more than 1 second the action is still missing.
SceneSynchronizerDebugger::singleton()->debug_warning(p_owner, "The action with ID: `" + itos(missing_actions[i].id) + "` was never received.");
// Remove it from missing actions, this will:
// 1. From now on this action will be discarded if received.
// 2. Reduce the `missing_actions` array size.
missing_actions.remove_unordered(i);
}
}
}

View File

@ -0,0 +1,92 @@
/*************************************************************************/
/* net_actions.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. */
/*************************************************************************/
#ifndef NET_ACTIONS_H
#define NET_ACTIONS_H
#include "core/reference.h"
#include "net_action_processor.h"
#include "net_utilities.h"
class SceneSynchronizer;
struct NetActionInfo;
struct SenderNetAction {
/// The token used to reference this Action.
uint32_t action_token = UINT32_MAX;
/// The token generated by the peer that generated the Action: Usually this is the same as `action_token`.
uint32_t triggerer_action_token = UINT32_MAX;
NetActionProcessor action_processor;
Vector<uint8_t> vars_buffer;
Vector<int> recipients;
int sender_peer = -1;
HashMap<int, uint32_t> peers_executed_input_id;
bool locally_executed = false;
bool sent_by_the_server = false;
uint32_t send_count = 0;
uint32_t send_timestamp = 0;
bool sender_executed_time_changed = false;
void prepare_processor(NetUtility::NodeData *p_nd, NetActionId p_event_id, const Array &p_vars);
const NetActionInfo &get_action_info() const;
void client_set_executed_input_id(uint32_t p_input_id);
uint32_t client_get_executed_input_id() const;
uint32_t peer_get_executed_input_id(int p_peer) const;
};
namespace net_action {
void encode_net_action(
const LocalVector<SenderNetAction *> &p_actions,
int p_source_peer,
DataBuffer &r_data_buffer);
void decode_net_action(
SceneSynchronizer *synchronizer,
DataBuffer &p_data_buffer,
int p_source_peer,
LocalVector<SenderNetAction> &r_actions);
}; //namespace net_action
struct NetActionSenderInfo {
uint32_t last_received_action_id = UINT32_MAX;
struct ActionAndTimestamp {
uint32_t id;
uint64_t timestamp;
bool operator==(const ActionAndTimestamp &p_other) const { return id == p_other.id; }
};
LocalVector<ActionAndTimestamp> missing_actions;
bool process_received_action(uint32_t p_action_id);
void check_missing_actions_and_clean_up(Node *p_owner);
};
#endif

View File

@ -0,0 +1,9 @@
#include "net_action_info.h"
bool NetActionInfo::operator==(const NetActionInfo &p_other) const {
return act_func == p_other.act_func;
}
bool NetActionInfo::operator<(const NetActionInfo &p_other) const {
return id < p_other.id;
}

View File

@ -0,0 +1,24 @@
#pragma once
#include "core/string_name.h"
#include "input_network_encoder.h"
#include "net_action_processor.h"
struct NetActionInfo {
NetActionId id = UINT32_MAX;
/// The event function
StringName act_func;
/// The event function encoding
StringName act_encoding_func;
/// If true the client can trigger this action.
bool can_client_trigger;
/// If true the client who triggered the event will wait the server validation to execute the event.
bool wait_server_validation;
/// The function to validate the event: Only executed on the server.
StringName server_action_validation_func;
/// The network_encoder used to encode decode the environment data.
Ref<InputNetworkEncoder> network_encoder;
bool operator==(const NetActionInfo &p_other) const;
bool operator<(const NetActionInfo &p_other) const;
};

View File

@ -0,0 +1,34 @@
#include "net_action_processor.h"
#include "input_network_encoder.h"
#include "net_action_info.h"
#include "net_utilities.h"
#include "scene/main/node.h"
void NetActionProcessor::execute() {
const NetActionInfo &info = nd->net_actions[action_id];
nd->node->callv(info.act_func, vars);
}
bool NetActionProcessor::server_validate() const {
const NetActionInfo &info = nd->net_actions[action_id];
if (info.server_action_validation_func == StringName()) {
// Always valid when the func is not set!
return true;
}
const Variant is_valid = nd->node->callv(info.server_action_validation_func, vars);
ERR_FAIL_COND_V_MSG(is_valid.get_type() != Variant::BOOL, false, "[FATAL] The function `" + nd->node->get_path() + "::" + info.server_action_validation_func + "` MUST return a bool.");
return is_valid.operator bool();
}
NetActionProcessor::operator String() const {
const NetActionInfo &info = nd->net_actions[action_id];
String v = Variant(vars);
// Strip `[]` from the Array string.
v = v.substr(1, v.size() - 3);
return String(nd->node->get_path()) + "::" + info.act_func + "(" + v + ")";
}

View File

@ -0,0 +1,40 @@
#pragma once
#include "core/array.h"
#include "core/ustring.h"
namespace NetUtility {
struct NodeData;
};
typedef uint32_t NetActionId;
struct NetActionProcessor {
NetUtility::NodeData *nd;
NetActionId action_id;
Array vars;
NetActionProcessor() = default;
NetActionProcessor(
NetUtility::NodeData *p_nd,
NetActionId p_action_id,
const Array &p_vars) :
nd(p_nd),
action_id(p_action_id),
vars(p_vars) {}
void execute();
bool server_validate() const;
operator String() const;
};
struct TokenizedNetActionProcessor {
uint32_t action_token;
NetActionProcessor processor;
bool operator==(const TokenizedNetActionProcessor &p_other) const { return action_token == p_other.action_token; }
TokenizedNetActionProcessor() = default;
TokenizedNetActionProcessor(uint32_t p_at, NetActionProcessor p_p) :
action_token(p_at), processor(p_p) {}
};

View File

@ -1,13 +1,12 @@
/*************************************************************************/
/* net_utilities.cpp */
/*************************************************************************/
/* This file is part of: */
/* PANDEMONIUM ENGINE */
/* https://github.com/Relintai/pandemonium_engine */
/* This file is part of: */
/* GODOT ENGINE */
/* https://godotengine.org */
/*************************************************************************/
/* Copyright (c) 2022-present Péter Magyar. */
/* Copyright (c) 2014-2022 Godot Engine contributors (cf. AUTHORS.md). */
/* Copyright (c) 2007-2022 Juan Linietsky, Ariel Manzur. */
/* 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 */
@ -34,9 +33,10 @@
*/
#include "net_utilities.h"
#include "scene/main/node.h"
#include "godot_backward_utility_cpp.h"
bool NetUtility::ChangeListener::operator==(const ChangeListener &p_other) const {
return object_id == p_other.object_id && method == p_other.method;
}
@ -61,7 +61,7 @@ bool NetUtility::VarData::operator<(const VarData &p_other) const {
return id < p_other.id;
}
void NetUtility::NodeData::process(const real_t p_delta) const {
void NetUtility::NodeData::process(const double p_delta) const {
const Variant var_delta = p_delta;
const Variant *fake_array_vars = &var_delta;

View File

@ -1,16 +1,12 @@
#ifndef NET_UTILITIES_H
#define NET_UTILITIES_H
/*************************************************************************/
/* net_utilities.h */
/*************************************************************************/
/* This file is part of: */
/* PANDEMONIUM ENGINE */
/* https://github.com/Relintai/pandemonium_engine */
/* This file is part of: */
/* GODOT ENGINE */
/* https://godotengine.org */
/*************************************************************************/
/* Copyright (c) 2022-present Péter Magyar. */
/* Copyright (c) 2014-2022 Godot Engine contributors (cf. AUTHORS.md). */
/* Copyright (c) 2007-2022 Juan Linietsky, Ariel Manzur. */
/* 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 */
@ -36,14 +32,25 @@
@author AndreaCatania
*/
#include "core/containers/local_vector.h"
#ifndef NET_UTILITIES_H
#define NET_UTILITIES_H
#include "core/local_vector.h"
#include "core/math/math_funcs.h"
#include "core/variant/variant.h"
#include "core/project_settings.h"
#include "core/variant.h"
#include "net_action_info.h"
#include "net_action_processor.h"
#include "godot_backward_utility_header.h"
#define ObjectID CompatObjectID
#ifdef DEBUG_ENABLED
#define NET_DEBUG_PRINT(msg) \
#define NET_DEBUG_PRINT(msg) \
if (ProjectSettings::get_singleton()->get_setting("NetworkSynchronizer/log_debug_warnings_and_messages")) \
print_line(String("[Net] ") + msg)
#define NET_DEBUG_WARN(msg) \
#define NET_DEBUG_WARN(msg) \
if (ProjectSettings::get_singleton()->get_setting("NetworkSynchronizer/log_debug_warnings_and_messages")) \
WARN_PRINT(String("[Net] ") + msg)
#define NET_DEBUG_ERR(msg) \
ERR_PRINT(String("[Net] ") + msg)
@ -56,6 +63,28 @@
typedef uint32_t NetNodeId;
typedef uint32_t NetVarId;
#ifdef TRACY_ENABLE
#include "godot_tracy/profiler.h"
#define PROFILE \
ZoneScoped;
#define PROFILE_NODE \
ZoneScoped; \
CharString c = String(get_path()).utf8(); \
if (c.size() >= std::numeric_limits<uint16_t>::max()) { \
c.resize(std::numeric_limits<uint16_t>::max() - 1); \
} \
ZoneText(c.ptr(), c.size());
#else
#define PROFILE
#define PROFILE_NODE
#endif
/// Flags used to control when an event is executed.
enum NetEventFlag {
@ -108,11 +137,12 @@ public:
/// Maximum value.
T max() const;
/// Minimum value.
T min(uint32_t p_consider_last) const;
/// Minumum value.
T min(uint32_t p_consider_last = UINT32_MAX) const;
/// Median value.
T average() const;
T average_rounded() const;
T get_deviation(T p_mean) const;
@ -209,6 +239,29 @@ T StatisticalRingBuffer<T>::average() const {
#endif
}
template <class T>
T StatisticalRingBuffer<T>::average_rounded() 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 = Math::round(double(a) / double(data.size()));
T b = Math::round(double(avg_sum) / double(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 Math::round(double(avg_sum) / double(data.size()));
#endif
}
template <class T>
T StatisticalRingBuffer<T>::get_deviation(T p_mean) const {
if (data.size() <= 0) {
@ -247,7 +300,7 @@ struct NodeChangeListener {
bool operator==(const NodeChangeListener &p_other) const;
};
/// Change listener that represents a pair of Object and Method.
/// 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.
@ -291,7 +344,7 @@ struct NodeData {
ObjectID instance_id = ObjectID();
NodeData *controlled_by = nullptr;
/// When `false`, this node is not sync. It's useful to locally pause sync
/// When `false`, this node is not sync. It's usefult to locally pause sync
/// of specific nodes.
bool sync_enabled = true;
@ -305,12 +358,14 @@ struct NodeData {
LocalVector<VarData> vars;
LocalVector<StringName> functions;
LocalVector<NetActionInfo> net_actions;
// This is valid to use only inside the process function.
Node *node = nullptr;
NodeData() = default;
void process(const real_t p_delta) const;
void process(const double p_delta) const;
};
struct PeerData {
@ -330,6 +385,8 @@ struct Snapshot {
/// The variable array order also matter.
Vector<Vector<Var>> node_vars;
Vector<TokenizedNetActionProcessor> actions;
operator String() const;
};
@ -340,4 +397,6 @@ struct PostponedRecover {
} // namespace NetUtility
#undef ObjectID
#endif

File diff suppressed because it is too large Load Diff

View File

@ -1,16 +1,12 @@
#ifndef NETWORKED_CONTROLLER_H
#define NETWORKED_CONTROLLER_H
/*************************************************************************/
/* networked_controller.h */
/*************************************************************************/
/* This file is part of: */
/* PANDEMONIUM ENGINE */
/* https://github.com/Relintai/pandemonium_engine */
/* This file is part of: */
/* GODOT ENGINE */
/* https://godotengine.org */
/*************************************************************************/
/* Copyright (c) 2022-present Péter Magyar. */
/* Copyright (c) 2014-2022 Godot Engine contributors (cf. AUTHORS.md). */
/* Copyright (c) 2007-2022 Juan Linietsky, Ariel Manzur. */
/* 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 */
@ -39,10 +35,14 @@
#include "scene/main/node.h"
#include "data_buffer.h"
#include "interpolator.h"
#include "net_utilities.h"
#include <deque>
#ifndef NETWORKED_CONTROLLER_H
#define NETWORKED_CONTROLLER_H
#include "godot_backward_utility_header.h"
class SceneSynchronizer;
struct Controller;
struct ServerController;
@ -56,7 +56,7 @@ struct NoNetController;
///
/// The `NetworkedController` will sync inputs, based on those will perform
/// operations.
/// The result of these operations, are guaranteed to be the same across the
/// The result of these operations, are guaranteed to be the same accross the
/// peers, if we stay under the assumption that the initial state is the same.
///
/// Is possible to use the `SceneSynchronizer` to keep the state in sync with the
@ -78,11 +78,24 @@ public:
CONTROLLER_TYPE_NULL,
CONTROLLER_TYPE_NONETWORK,
CONTROLLER_TYPE_PLAYER,
CONTROLLER_TYPE_AUTONOMOUS_SERVER,
CONTROLLER_TYPE_SERVER,
CONTROLLER_TYPE_DOLL
};
private:
/// When `true`, this controller is controlled by the server: All the clients
/// see it as a `Doll`.
/// This property is really useful to implement bots (Character controlled by
/// the AI).
///
/// NOTICE: Generally you specify this property on the editor, in addition
/// it's possible to change this at runtime: this will cause the server to
/// notify all the clients; so the switch is not immediate. This feature can be
/// used to switch the Character possession between the AI (Server) and
/// PlayerController (Client) without the need to re-instantiate the Character.
bool server_controlled = false;
/// The input storage size is used to cap the amount of inputs collected by
/// the `PlayerController`.
///
@ -104,11 +117,11 @@ private:
/// Amount of time an inputs is re-sent to each peer.
/// Resenging inputs is necessary because the packets may be lost since as
/// they are sent in an unreliable way.
int max_redundant_inputs = 5;
int max_redundant_inputs = 6;
/// Time in seconds between each `tick_speedup` that the server sends to the
/// client.
real_t tick_speedup_notification_delay = 0.33;
/// client. In ms.
int tick_speedup_notification_delay = 600;
/// The connection quality is established by watching the time passed
/// between each input is received.
@ -120,22 +133,6 @@ private:
/// - Small values make the mechanism too sensible.
int network_traced_frames = 120;
/// Sensitivity to network oscillations. The value is in seconds and can be
/// used to establish the connection quality.
///
/// For each input, the time needed for its arrival is traced; the standard
/// deviation of these is divided by `net_sensitivity`: the result is
/// the connection quality.
///
/// The more the time needed for each batch to arrive is different the
/// bigger this value is: when this value approaches to
/// `net_sensitivity` (or even surpasses it) the bad the connection is.
///
/// The result is the `connection_poorness` that goes from 0 to 1 and is
/// used to decide the `optimal_frame_delay` that is interpolated between
/// `min_frames_delay` and `max_frames_delay`.
real_t net_sensitivity = 0.1;
/// The `ServerController` will try to keep a margin of error, so that
/// network oscillations doesn't leave the `ServerController` without
/// inputs.
@ -143,26 +140,14 @@ private:
/// This margin of error is called `optimal_frame_delay` and it changes
/// depending on the connection health:
/// it can go from `min_frames_delay` to `max_frames_delay`.
int min_frames_delay = 2;
int max_frames_delay = 6;
int min_frames_delay = 0;
int max_frames_delay = 7;
/// Rate at which the tick speed changes, so the `optimal_frame_delay` is
/// matched.
real_t tick_acceleration = 2.0;
/// Amount of additional frames produced per second.
double tick_acceleration = 5.0;
/// Collect rate (in frames) used by the server to establish when to collect
/// the state for a particular peer.
/// It's possible to scale down this rate, for a particular peer,
/// using the function: set_doll_collect_rate_factor(peer, factor);
/// Current default is 10Hz.
///
/// The collected state is not immediately sent to the clients, rather it's
/// delayed so to be sent in batch. The states marked as important are
/// always collected.
int doll_epoch_collect_rate = 1;
/// The time rate at which a new batch is sent.
real_t doll_epoch_batch_sync_rate = 0.25;
/// The doll epoch send rate: in Hz (frames per seconds).
uint32_t doll_sync_rate = 30;
/// The doll interpolator will try to keep a margin of error, so that network
/// oscillations doesn't make the dolls freeze.
@ -170,11 +155,22 @@ private:
/// This margin of error is called `optimal_frame_delay` and it changes
/// depending on the connection health:
/// it can go from `doll_min_frames_delay` to `doll_max_frames_delay`.
int doll_min_frames_delay = 2;
int doll_max_frames_delay = 5;
int doll_min_frames_delay = 0;
int doll_max_frames_delay = 25;
/// Max speedup / slowdown the doll can apply to recover its epoch buffer size.
real_t doll_interpolation_max_speedup = 0.2;
/// Sensitivity to network oscillations. The value is in seconds and can be
/// used to establish the connection quality.
///
/// The net sync checks the amount of time for each packet to arrive.
/// The different it is, the more unreliable the connection is, so virtual latency
/// is used to smooth the interpolation.
///
/// `doll_net_sensitivity` is an amount in seconds used to determine the maximum delta difference
/// between the packets.
real_t doll_net_sensitivity = 0.21;
/// Max doll interpolation overshot. Unit: normalized percentage.
real_t doll_interpolation_max_overshot = 0.2;
/// The connection quality is established by watching the time passed
/// between each batch arrival.
@ -185,31 +181,8 @@ private:
/// - Big values make the mechanism too slow.
/// - Small values make the mechanism too sensible.
/// The correct value should be give considering the
/// `doll_epoch_batch_sync_rate`.
int doll_connection_stats_frame_span = 30;
/// Sensitivity to network oscillations. The value is in seconds and can be
/// used to establish the connection quality.
///
/// For each batch, the time needed for its arrival is traced; the standard
/// deviation of these is divided by `doll_net_sensitivity`: the result is
/// the connection quality.
///
/// The more the time needed for each batch to arrive is different the
/// bigger this value is: when this value approaches to
/// `doll_net_sensitivity` (or even surpasses it) the bad the connection is.
///
/// The result is the `connection_poorness` that goes from 0 to 1 and is
/// used to decide the `optimal_frames_delay` that is interpolated between
/// `doll_min_frames_delay` and `doll_max_frames_delay`.
real_t doll_net_sensitivity = 0.2;
/// Just after a bad connection phase, may happen that all the sent epochs
/// are received at once. To make sure the actor is always the more up-to-date
/// possible, the number of epochs the actor has to fetch is clamped.
/// When `doll_max_delay` is surpassed the doll is teleported forward the
/// timeline so to be the more update possible.
uint64_t doll_max_delay = 60;
/// `doll_sync_rate`.
int doll_connection_stats_frame_span = 60;
ControllerType controller_type = CONTROLLER_TYPE_NULL;
Controller *controller = nullptr;
@ -225,6 +198,10 @@ public:
public:
NetworkedController();
~NetworkedController();
void set_server_controlled(bool p_server_controlled);
bool get_server_controlled() const;
void set_player_input_storage_size(int p_size);
int get_player_input_storage_size() const;
@ -232,8 +209,8 @@ public:
void set_max_redundant_inputs(int p_max);
int get_max_redundant_inputs() const;
void set_tick_speedup_notification_delay(real_t p_delay);
real_t get_tick_speedup_notification_delay() const;
void set_tick_speedup_notification_delay(int p_delay_in_ms);
int get_tick_speedup_notification_delay() const;
void set_network_traced_frames(int p_size);
int get_network_traced_frames() const;
@ -244,17 +221,14 @@ public:
void set_max_frames_delay(int p_val);
int get_max_frames_delay() const;
void set_net_sensitivity(real_t p_val);
real_t get_net_sensitivity() const;
void set_tick_acceleration(real_t p_acceleration);
real_t get_tick_acceleration() const;
void set_tick_acceleration(double p_acceleration);
double get_tick_acceleration() const;
void set_doll_epoch_collect_rate(int p_rate);
int get_doll_epoch_collect_rate() const;
void set_doll_epoch_batch_sync_rate(real_t p_rate);
real_t get_doll_epoch_batch_sync_rate() const;
void set_doll_sync_rate(uint32_t p_rate);
uint32_t get_doll_sync_rate() const;
void set_doll_min_frames_delay(int p_min);
int get_doll_min_frames_delay() const;
@ -262,17 +236,17 @@ public:
void set_doll_max_frames_delay(int p_max);
int get_doll_max_frames_delay() const;
void set_doll_interpolation_max_speedup(real_t p_speedup);
real_t get_doll_interpolation_max_speedup() const;
void set_doll_net_sensitivity(real_t p_sensitivity);
real_t get_doll_net_sensitivity() const;
void set_doll_interpolation_max_overshot(real_t p_speedup);
real_t get_doll_interpolation_max_overshot() const;
void set_doll_connection_stats_frame_span(int p_span);
int get_doll_connection_stats_frame_span() const;
void set_doll_net_sensitivity(real_t p_sensitivity);
real_t get_doll_net_sensitivity() const;
void set_doll_max_delay(uint32_t p_max_delay);
uint32_t get_doll_max_delay() const;
void set_doll_virtual_delay_max_bias(uint32_t p_max_delay);
uint32_t get_doll_virtual_delay_max_bias() const;
uint32_t get_current_input_id() const;
@ -285,7 +259,7 @@ public:
}
/// Returns the pretended delta used by the player.
real_t player_get_pretended_delta(uint32_t p_physics_ticks_per_seconds) const;
real_t player_get_pretended_delta() const;
void mark_epoch_as_important();
@ -293,6 +267,14 @@ public:
void set_doll_peer_active(int p_peer_id, bool p_active);
void pause_notify_dolls();
virtual void validate_script_implementation();
virtual void native_collect_inputs(double p_delta, DataBuffer &r_buffer);
virtual void native_controller_process(double p_delta, DataBuffer &p_buffer);
virtual bool native_are_inputs_different(DataBuffer &p_buffer_A, DataBuffer &p_buffer_B);
virtual uint32_t native_count_input_size(DataBuffer &p_buffer);
virtual void native_collect_epoch_data(DataBuffer &r_buffer);
virtual void native_apply_epoch(double p_delta, real_t p_interpolation_alpha, DataBuffer &p_past_buffer, DataBuffer &p_future_buffer);
bool process_instant(int p_i, real_t p_delta);
/// Returns the server controller or nullptr if this is not a server.
@ -308,6 +290,7 @@ public:
NoNetController *get_nonet_controller();
const NoNetController *get_nonet_controller() const;
bool is_networking_initialized() const;
bool is_server_controller() const;
bool is_player_controller() const;
bool is_doll_controller() const;
@ -321,24 +304,26 @@ public:
bool has_scene_synchronizer() const;
/* On server rpc functions. */
void _rpc_server_send_inputs(const PoolVector<uint8_t> &p_data);
void _rpc_server_send_inputs(const Vector<uint8_t> &p_data);
/* On client rpc functions. */
void _rpc_send_tick_additional_speed(const PoolVector<uint8_t> &p_data);
void _rpc_set_server_controlled(bool p_server_controlled);
void _rpc_notify_fps_acceleration(const Vector<uint8_t> &p_data);
/* On puppet rpc functions. */
void _rpc_doll_notify_sync_pause(uint32_t p_epoch);
void _rpc_doll_send_epoch_batch(const PoolVector<uint8_t> &p_data);
void _rpc_doll_send_epoch_batch(const Vector<uint8_t> &p_data);
void process(real_t p_delta);
void process(double p_delta);
void player_set_has_new_input(bool p_has);
bool player_has_new_input() const;
void __on_sync_paused();
private:
virtual void _notification(int p_what);
protected:
void _notification(int p_what);
void notify_controller_reset();
};
struct FrameSnapshot {
@ -346,6 +331,8 @@ struct FrameSnapshot {
BitArray inputs_buffer;
uint32_t buffer_size_bit;
uint32_t similarity;
/// Local timestamp.
uint32_t received_timestamp;
bool operator==(const FrameSnapshot &p_other) const {
return p_other.id == id;
@ -376,57 +363,56 @@ struct ServerController : public Controller {
int peer = 0;
bool active = true;
real_t update_rate_factor = 1.0;
int collect_timer = 0; // In frames
int collect_threshold = 0; // In frames
LocalVector<PoolVector<uint8_t>> epoch_batch;
uint32_t batch_size = 0;
real_t doll_sync_rate_factor = 1.0;
real_t doll_sync_timer = 0.0;
real_t doll_sync_time_threshold = 0.0;
};
uint32_t current_input_buffer_id = UINT32_MAX;
uint32_t ghost_input_count = 0;
uint32_t last_sent_state_input_id = 0;
real_t client_tick_additional_speed = 0.0;
real_t additional_speed_notif_timer = 0.0;
uint32_t additional_fps_notif_timer = 0;
std::deque<FrameSnapshot> snapshots;
bool streaming_paused = false;
bool enabled = true;
uint32_t input_arrival_time = UINT32_MAX;
uint32_t previous_frame_received_timestamp = UINT32_MAX;
NetUtility::StatisticalRingBuffer<uint32_t> network_watcher;
NetUtility::StatisticalRingBuffer<int> consecutive_input_watcher;
/// Used to sync the dolls.
LocalVector<Peer> peers;
DataBuffer epoch_state_data_cache;
uint32_t epoch = 0;
bool is_epoch_important = false;
real_t batch_sync_timer = 0.0;
ServerController(
NetworkedController *p_node,
int p_traced_frames);
void process(real_t p_delta);
void process(double p_delta);
uint32_t last_known_input() const;
virtual uint32_t get_current_input_id() const;
virtual uint32_t get_current_input_id() const override;
void set_enabled(bool p_enable);
virtual void clear_peers();
virtual void activate_peer(int p_peer);
virtual void deactivate_peer(int p_peer);
virtual void clear_peers() override;
virtual void activate_peer(int p_peer) override;
virtual void deactivate_peer(int p_peer) override;
void receive_inputs(const PoolVector<uint8_t> &p_data);
int get_inputs_count() const;
virtual void receive_inputs(const Vector<uint8_t> &p_data);
virtual int get_inputs_count() const;
/// Fetch the next inputs, returns true if the input is new.
bool fetch_next_input();
virtual bool fetch_next_input(real_t p_delta);
void set_frame_input(const FrameSnapshot &p_frame_snapshot);
void notify_send_state();
void doll_sync(real_t p_delta);
/// This function updates the `tick_additional_speed` so that the `frames_inputs`
/// This function updates the `tick_additional_fps` so that the `frames_inputs`
/// size is enough to reduce the missing packets to 0.
///
/// When the internet connection is bad, the packets need more time to arrive.
@ -437,33 +423,44 @@ struct ServerController : public Controller {
/// the server is artificial and no more dependent on the internet. For this
/// reason the server tells the client to slowdown so to keep the `frames_inputs`
/// size moderate to the needs.
void calculates_player_tick_rate(real_t p_delta);
void adjust_player_tick_rate(real_t p_delta);
virtual void adjust_player_tick_rate(double p_delta);
uint32_t find_peer(int p_peer) const;
};
struct AutonomousServerController : public ServerController {
AutonomousServerController(
NetworkedController *p_node);
virtual void receive_inputs(const Vector<uint8_t> &p_data) override;
virtual int get_inputs_count() const override;
virtual bool fetch_next_input(real_t p_delta) override;
virtual void adjust_player_tick_rate(double p_delta) override;
};
struct PlayerController : public Controller {
uint32_t current_input_id;
uint32_t input_buffers_counter;
real_t time_bank;
real_t tick_additional_speed;
double time_bank;
double acceleration_fps_speed = 0.0;
double acceleration_fps_timer = 1.0;
bool streaming_paused = false;
double pretended_delta = 1.0;
std::deque<FrameSnapshot> frames_snapshot;
LocalVector<uint8_t> cached_packet_data;
PlayerController(NetworkedController *p_node);
void process(real_t p_delta);
int calculates_sub_ticks(real_t p_delta, real_t p_iteration_per_seconds);
void process(double p_delta);
/// Returns the amount of frames to process for this frame.
int calculates_sub_ticks(const double p_delta, const double p_iteration_per_seconds);
int notify_input_checked(uint32_t p_input_id);
uint32_t last_known_input() const;
uint32_t get_stored_input_id(int p_i) const;
virtual uint32_t get_current_input_id() const;
virtual uint32_t get_current_input_id() const override;
bool process_instant(int p_i, real_t p_delta);
real_t get_pretended_delta(real_t p_iteration_per_second) const;
void store_input_buffer(uint32_t p_id);
@ -485,30 +482,35 @@ struct PlayerController : public Controller {
/// with the server execution (see `soft_reset_to_server_state`) and the possibility
/// for the server to stop the data streaming.
struct DollController : public Controller {
Interpolator interpolator;
real_t additional_speed = 0.0;
uint32_t current_epoch = UINT32_MAX;
real_t advancing_epoch = 0.0;
uint32_t missing_epochs = 0;
real_t interpolation_alpha = 0.0;
real_t interpolation_time_window = 0.0;
uint32_t current_epoch = 0;
uint32_t past_epoch = 0;
DataBuffer past_epoch_buffer;
uint32_t future_epoch = 0;
DataBuffer future_epoch_buffer;
// Any received epoch prior to this one is discarded.
uint32_t paused_epoch = 0;
// Used to track the time taken for the next batch to arrive.
uint32_t batch_receiver_timer = UINT32_MAX;
uint32_t epoch_received_timestamp = UINT32_MAX;
real_t next_epoch_expected_in = 0.0;
/// Used to track how network is performing.
NetUtility::StatisticalRingBuffer<uint32_t> network_watcher;
NetUtility::StatisticalRingBuffer<real_t> network_watcher;
DollController(NetworkedController *p_node);
virtual void ready();
void process(real_t p_delta);
virtual void ready() override;
void process(double p_delta);
// TODO consider make this non virtual
virtual uint32_t get_current_input_id() const;
virtual uint32_t get_current_input_id() const override;
void receive_batch(const PoolVector<uint8_t> &p_data);
uint32_t receive_epoch(const PoolVector<uint8_t> &p_data);
void receive_epoch(const Vector<uint8_t> &p_data);
uint32_t next_epoch();
void pause(uint32_t p_epoch);
};
@ -520,8 +522,8 @@ struct NoNetController : public Controller {
NoNetController(NetworkedController *p_node);
void process(real_t p_delta);
virtual uint32_t get_current_input_id() const;
void process(double p_delta);
virtual uint32_t get_current_input_id() const override;
};
#endif

View File

@ -1,13 +1,12 @@
/*************************************************************************/
/* register_types.cpp */
/*************************************************************************/
/* This file is part of: */
/* PANDEMONIUM ENGINE */
/* https://github.com/Relintai/pandemonium_engine */
/* This file is part of: */
/* GODOT ENGINE */
/* https://godotengine.org */
/*************************************************************************/
/* Copyright (c) 2022-present Péter Magyar. */
/* Copyright (c) 2014-2022 Godot Engine contributors (cf. AUTHORS.md). */
/* Copyright (c) 2007-2022 Juan Linietsky, Ariel Manzur. */
/* 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 */
@ -35,25 +34,33 @@
#include "register_types.h"
#include "core/config/project_settings.h"
#include "core/engine.h"
#include "data_buffer.h"
#include "interpolator.h"
#include "networked_controller.h"
#include "scene_diff.h"
#include "scene_synchronizer.h"
#include "scene_synchronizer_debugger.h"
#include "input_network_encoder.h"
void register_network_synchronizer_types(ModuleRegistrationLevel p_level) {
if (p_level == MODULE_REGISTRATION_LEVEL_SCENE) {
ClassDB::register_class<DataBuffer>();
ClassDB::register_class<SceneDiff>();
ClassDB::register_class<Interpolator>();
ClassDB::register_class<NetworkedController>();
ClassDB::register_class<SceneSynchronizer>();
void register_network_synchronizer_types() {
ClassDB::register_class<DataBuffer>();
ClassDB::register_class<SceneDiff>();
ClassDB::register_class<NetworkedController>();
ClassDB::register_class<SceneSynchronizer>();
ClassDB::register_class<SceneSynchronizerDebugger>();
ClassDB::register_class<InputNetworkEncoder>();
GLOBAL_DEF("NetworkSynchronizer/debug_server_speedup", false);
GLOBAL_DEF("NetworkSynchronizer/debug_doll_speedup", false);
}
memnew(SceneSynchronizerDebugger);
Engine::get_singleton()->add_singleton(Engine::Singleton("SceneSynchronizerDebugger", SceneSynchronizerDebugger::singleton()));
GLOBAL_DEF("NetworkSynchronizer/debug_server_speedup", false);
GLOBAL_DEF("NetworkSynchronizer/debug_doll_speedup", false);
GLOBAL_DEF("NetworkSynchronizer/log_debug_warnings_and_messages", true);
GLOBAL_DEF("NetworkSynchronizer/debugger/dump_enabled", false);
GLOBAL_DEF("NetworkSynchronizer/debugger/dump_classes", Array());
GLOBAL_DEF("NetworkSynchronizer/debugger/log_debug_fps_warnings", true);
}
void unregister_network_synchronizer_types(ModuleRegistrationLevel p_level) {
void unregister_network_synchronizer_types() {
memdelete(SceneSynchronizerDebugger::singleton());
}

View File

@ -1,16 +1,12 @@
#ifndef NETWORK_SYNCHRONIZER_REGISTER_TYPES_H
#define NETWORK_SYNCHRONIZER_REGISTER_TYPES_H
/*************************************************************************/
/* register_types.h */
/*************************************************************************/
/* This file is part of: */
/* PANDEMONIUM ENGINE */
/* https://github.com/Relintai/pandemonium_engine */
/* This file is part of: */
/* GODOT ENGINE */
/* https://godotengine.org */
/*************************************************************************/
/* Copyright (c) 2022-present Péter Magyar. */
/* Copyright (c) 2014-2022 Godot Engine contributors (cf. AUTHORS.md). */
/* Copyright (c) 2007-2022 Juan Linietsky, Ariel Manzur. */
/* 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 */
@ -36,9 +32,5 @@
@author AndreaCatania
*/
#include "modules/register_module_types.h"
void register_network_synchronizer_types(ModuleRegistrationLevel p_level);
void unregister_network_synchronizer_types(ModuleRegistrationLevel p_level);
#endif
void register_network_synchronizer_types();
void unregister_network_synchronizer_types();

View File

@ -1,13 +1,12 @@
/*************************************************************************/
/* scene_diff.cpp */
/*************************************************************************/
/* This file is part of: */
/* PANDEMONIUM ENGINE */
/* https://github.com/Relintai/pandemonium_engine */
/* This file is part of: */
/* GODOT ENGINE */
/* https://godotengine.org */
/*************************************************************************/
/* Copyright (c) 2022-present Péter Magyar. */
/* Copyright (c) 2014-2022 Godot Engine contributors (cf. AUTHORS.md). */
/* Copyright (c) 2007-2022 Juan Linietsky, Ariel Manzur. */
/* 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 */

View File

@ -1,16 +1,12 @@
#ifndef SCENE_DIFF_H
#define SCENE_DIFF_H
/*************************************************************************/
/* scene_diff.h */
/*************************************************************************/
/* This file is part of: */
/* PANDEMONIUM ENGINE */
/* https://github.com/Relintai/pandemonium_engine */
/* This file is part of: */
/* GODOT ENGINE */
/* https://godotengine.org */
/*************************************************************************/
/* Copyright (c) 2022-present Péter Magyar. */
/* Copyright (c) 2014-2022 Godot Engine contributors (cf. AUTHORS.md). */
/* Copyright (c) 2007-2022 Juan Linietsky, Ariel Manzur. */
/* 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 */
@ -36,8 +32,10 @@
@author AndreaCatania
*/
#include "core/containers/local_vector.h"
#include "core/object/class_db.h"
#ifndef SCENE_DIFF_H
#define SCENE_DIFF_H
#include "core/class_db.h"
#include "net_utilities.h"
class SceneSynchronizer;

File diff suppressed because it is too large Load Diff

View File

@ -1,16 +1,12 @@
#ifndef SCENE_SYNCHRONIZER_H
#define SCENE_SYNCHRONIZER_H
/*************************************************************************/
/* scene_synchronizer.h */
/*************************************************************************/
/* This file is part of: */
/* PANDEMONIUM ENGINE */
/* https://github.com/Relintai/pandemonium_engine */
/* This file is part of: */
/* GODOT ENGINE */
/* https://godotengine.org */
/*************************************************************************/
/* Copyright (c) 2022-present Péter Magyar. */
/* Copyright (c) 2014-2022 Godot Engine contributors (cf. AUTHORS.md). */
/* Copyright (c) 2007-2022 Juan Linietsky, Ariel Manzur. */
/* 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 */
@ -38,13 +34,20 @@
#include "scene/main/node.h"
#include "core/containers/local_vector.h"
#include "core/containers/oa_hash_map.h"
#include "core/local_vector.h"
#include "core/oa_hash_map.h"
#include "net_action.h"
#include "net_utilities.h"
#include <deque>
#ifndef SCENE_SYNCHRONIZER_H
#define SCENE_SYNCHRONIZER_H
#include "godot_backward_utility_header.h"
class Synchronizer;
class NetworkedController;
class PlayerController;
/// # SceneSynchronizer
///
@ -61,20 +64,20 @@ class NetworkedController;
/// The clients receives the server snapshot, so it compares with the local
/// snapshot and if it's necessary perform the recovery.
///
/// ## Variable tracking
/// ## Variable traking
///
/// The `SceneSynchronizer` is able to track any node variable. It's possible to specify
/// the variables to track using the function `register_variable`.
///
/// ## NetworkedController
/// The `NetworkedController` is able to acquire the `Player` input and perform
/// The `NetworkedController` is able to aquire the `Player` input and perform
/// operation in sync with other peers. When a discrepancy is found by the
/// `SceneSynchronizer`, it will drive the `NetworkedController` so to recover that
/// missalignment.
///
///
/// ## Processing function
/// Some objects, that are not directly controlled by a `Player`, may need to be
/// Some objects, that are not direclty controlled by a `Player`, may need to be
/// in sync between peers; since those are not controlled by a `Player` is
/// not necessary use the `NetworkedController`.
///
@ -127,6 +130,11 @@ public:
private:
real_t server_notify_state_interval = 1.0;
real_t comparison_float_tolerance = 0.001;
/// The amount of time the same act is sent to a client before being considered delivered.
/// This is part of the UDP reliability mechanism.
int actions_redundancy = 3;
/// Send the act again in (seconds):
real_t actions_resend_time = 1.0 / 30.0;
SynchronizerType synchronizer_type = SYNCHRONIZER_TYPE_NULL;
Synchronizer *synchronizer = nullptr;
@ -150,7 +158,7 @@ private:
// Controller nodes.
LocalVector<NetUtility::NodeData *> node_data_controllers;
// Just used to detect when the peer change. TODO Remove this and use a signal instead.
// Just used to detect when the peer change. TODO Remove this and use a singnal instead.
void *peer_ptr = nullptr;
int event_flag;
@ -159,7 +167,7 @@ private:
public:
static void _bind_methods();
virtual void _notification(int p_what);
void _notification(int p_what);
public:
SceneSynchronizer();
@ -174,6 +182,14 @@ public:
void set_comparison_float_tolerance(real_t p_tolerance);
real_t get_comparison_float_tolerance() const;
void set_actions_redundancy(int p_redundancy);
int get_actions_redundancy() const;
void set_actions_resend_time(real_t p_time);
real_t get_actions_resend_time() const;
bool is_variable_registered(Node *p_node, const StringName &p_variable) const;
/// Register a new node and returns its `NodeData`.
NetUtility::NodeData *register_node(Node *p_node);
uint32_t register_node_gdscript(Node *p_node);
@ -195,6 +211,36 @@ public:
void stop_node_sync(const Node *p_node);
bool is_node_sync(const Node *p_node) const;
/// Register an new action.
///
/// @param p_node The node that owns the event
/// @param p_action_func The function that is triggered when the event is executed.
/// @param p_action_encoding_func The function called to definte the validation encoding.
/// @param p_can_client_trigger If true this `Action` can be triggered on client.
/// @param p_wait_server_validation If true the event will be emitted locally only if the server validates it.
/// @param p_server_action_validation_func The validation function, must return a boolean.
NetActionId register_action(
Node *p_node,
const StringName &p_action_func,
const StringName &p_action_encoding_func,
bool p_can_client_trigger = false,
bool p_wait_server_validation = false,
const StringName &p_server_action_validation_func = StringName());
NetActionId find_action_id(Node *p_node, const StringName &p_action_func) const;
void trigger_action_by_name(
Node *p_node,
const StringName &p_action_func,
const Array &p_arguments = Array(),
const Vector<int> &p_recipients = Vector<int>());
void trigger_action(
Node *p_node,
NetActionId p_id,
const Array &p_arguments = Array(),
const Vector<int> &p_recipients = Vector<int>());
/// Returns the variable ID relative to the `Node`.
/// This may return `UINT32_MAX` in various cases:
/// - The node is not registered.
@ -212,7 +258,7 @@ public:
/// Add a dependency to a controller, so that the rewinding mechanism can
/// make sure to rewind that node when the controller is rewinded.
/// You can remove and add dependency at any time. This operation
/// don't need to be performed on server.
/// don't need to be perfomed on server.
void controller_add_dependency(Node *p_controller, Node *p_node);
void controller_remove_dependency(Node *p_controller, Node *p_node);
int controller_get_dependency_count(Node *p_controller) const;
@ -250,10 +296,13 @@ public:
void reset_synchronizer_mode();
void clear();
void notify_controller_control_mode_changed(NetworkedController *controller);
void _rpc_send_state(const Variant &p_snapshot);
void _rpc_notify_need_full_snapshot();
void _rpc_set_network_enabled(bool p_enabled);
void _rpc_notify_peer_status(bool p_enabled);
void _rpc_send_actions(const Vector<uint8_t> &p_data);
void update_peers();
void clear_peers();
@ -262,7 +311,7 @@ public:
void change_event_add(NetUtility::NodeData *p_node_data, NetVarId p_var_id, const Variant &p_old);
void change_events_flush();
private:
public: // -------------------------------------------------------------------------------- INTERNAL
void expand_organized_node_data_vector(uint32_t p_size);
/// This function is slow, but allow to take the node data even if the
@ -301,6 +350,8 @@ private:
/// Set the node data net id.
void set_node_data_id(NetUtility::NodeData *p_node_data, NetNodeId p_id);
NetworkedController *fetch_controller_by_peer(int peer);
public:
/// Returns true when the vectors are the same. Uses comparison_float_tolerance member.
bool compare(const Vector2 &p_first, const Vector2 &p_second) const;
@ -342,18 +393,33 @@ public:
virtual void on_variable_added(NetUtility::NodeData *p_node_data, const StringName &p_var_name) {}
virtual void on_variable_changed(NetUtility::NodeData *p_node_data, NetVarId p_var_id, const Variant &p_old_value, int p_flag) {}
virtual void on_controller_reset(NetUtility::NodeData *p_node_data) {}
virtual void on_action_triggered(
NetUtility::NodeData *p_node_data,
NetActionId p_id,
const Array &p_arguments,
const Vector<int> &p_recipients) {}
virtual void on_actions_received(
int sender_peer,
const LocalVector<SenderNetAction> &p_actions) {}
};
class NoNetSynchronizer : public Synchronizer {
friend class SceneSynchronizer;
bool enabled = true;
uint32_t frame_count = 0;
LocalVector<NetActionProcessor> pending_actions;
public:
NoNetSynchronizer(SceneSynchronizer *p_node);
virtual void clear();
virtual void process();
virtual void clear() override;
virtual void process() override;
virtual void on_action_triggered(
NetUtility::NodeData *p_node_data,
NetActionId p_id,
const Array &p_arguments,
const Vector<int> &p_recipients) override;
void set_enabled(bool p_enabled);
bool is_enabled() const;
@ -366,26 +432,58 @@ class ServerSynchronizer : public Synchronizer {
struct Change {
bool not_known_before = false;
RBSet<StringName> uknown_vars;
RBSet<StringName> vars;
Set<StringName> uknown_vars;
Set<StringName> vars;
};
enum SnapshotGenerationMode {
/// The shanpshot will include The NodeId or NodePath and allthe changed variables.
SNAPSHOT_GENERATION_MODE_NORMAL,
/// The snapshot will include The NodePath only in case it was unknown before.
SNAPSHOT_GENERATION_MODE_NODE_PATH_ONLY,
/// The snapshot will include The NodePath only.
SNAPSHOT_GENERATION_MODE_FORCE_NODE_PATH_ONLY,
/// The snapshot will contains everything no matter what.
SNAPSHOT_GENERATION_MODE_FORCE_FULL,
};
/// The changes; the order matters because the index is the NetNodeId.
LocalVector<Change> changes;
OAHashMap<int, NetActionSenderInfo> senders_info;
OAHashMap<int, uint32_t> peers_next_action_trigger_input_id;
uint32_t server_actions_count = 0;
LocalVector<SenderNetAction> server_actions;
public:
ServerSynchronizer(SceneSynchronizer *p_node);
virtual void clear();
virtual void process();
virtual void on_node_added(NetUtility::NodeData *p_node_data);
virtual void on_variable_added(NetUtility::NodeData *p_node_data, const StringName &p_var_name);
virtual void on_variable_changed(NetUtility::NodeData *p_node_data, NetVarId p_var_id, const Variant &p_old_value, int p_flag);
virtual void clear() override;
virtual void process() override;
virtual void on_node_added(NetUtility::NodeData *p_node_data) override;
virtual void on_node_removed(NetUtility::NodeData *p_node_data) override;
virtual void on_variable_added(NetUtility::NodeData *p_node_data, const StringName &p_var_name) override;
virtual void on_variable_changed(NetUtility::NodeData *p_node_data, NetVarId p_var_id, const Variant &p_old_value, int p_flag) override;
virtual void on_action_triggered(
NetUtility::NodeData *p_node_data,
NetActionId p_id,
const Array &p_arguments,
const Vector<int> &p_recipients) override;
virtual void on_actions_received(
int sender_peer,
const LocalVector<SenderNetAction> &p_actions) override;
void process_snapshot_notificator(real_t p_delta);
Vector<Variant> global_nodes_generate_snapshot(bool p_force_full_snapshot) const;
void controller_generate_snapshot(const NetUtility::NodeData *p_node_data, bool p_force_full_snapshot, Vector<Variant> &r_snapshot_result) const;
void generate_snapshot_node_data(const NetUtility::NodeData *p_node_data, bool p_force_full_snapshot, Vector<Variant> &r_result) const;
void generate_snapshot_node_data(const NetUtility::NodeData *p_node_data, SnapshotGenerationMode p_mode, Vector<Variant> &r_result) const;
void execute_actions();
void send_actions_to_clients();
void clean_pending_actions();
void check_missing_actions();
};
class ClientSynchronizer : public Synchronizer {
@ -417,25 +515,39 @@ class ClientSynchronizer : public Synchronizer {
}
};
RBSet<EndSyncEvent> sync_end_events;
Set<EndSyncEvent> sync_end_events;
uint32_t locally_triggered_actions_count = 0;
uint32_t actions_input_id = 0;
LocalVector<SenderNetAction> pending_actions;
NetActionSenderInfo server_sender_info;
public:
ClientSynchronizer(SceneSynchronizer *p_node);
virtual void clear();
virtual void clear() override;
virtual void process();
virtual void on_node_added(NetUtility::NodeData *p_node_data);
virtual void on_node_removed(NetUtility::NodeData *p_node_data);
virtual void on_variable_changed(NetUtility::NodeData *p_node_data, NetVarId p_var_id, const Variant &p_old_value, int p_flag);
virtual void on_controller_reset(NetUtility::NodeData *p_node_data);
virtual void process() override;
virtual void on_node_added(NetUtility::NodeData *p_node_data) override;
virtual void on_node_removed(NetUtility::NodeData *p_node_data) override;
virtual void on_variable_changed(NetUtility::NodeData *p_node_data, NetVarId p_var_id, const Variant &p_old_value, int p_flag) override;
virtual void on_controller_reset(NetUtility::NodeData *p_node_data) override;
virtual void on_action_triggered(
NetUtility::NodeData *p_node_data,
NetActionId p_id,
const Array &p_arguments,
const Vector<int> &p_recipients) override;
virtual void on_actions_received(
int sender_peer,
const LocalVector<SenderNetAction> &p_actions) override;
void receive_snapshot(Variant p_snapshot);
bool parse_sync_data(
Variant p_snapshot,
void *p_user_pointer,
void (*p_node_parse)(void *p_user_pointer, NetUtility::NodeData *p_node_data),
void (*p_controller_parse)(void *p_user_pointer, NetUtility::NodeData *p_node_data, uint32_t p_input_id),
void (*p_input_id_parse)(void *p_user_pointer, uint32_t p_input_id),
void (*p_controller_parse)(void *p_user_pointer, NetUtility::NodeData *p_node_data),
void (*p_variable_parse)(void *p_user_pointer, NetUtility::NodeData *p_node_data, NetVarId p_var_id, const Variant &p_value));
void set_enabled(bool p_enabled);
@ -449,6 +561,29 @@ private:
std::deque<NetUtility::Snapshot> &r_snapshot_storage);
void process_controllers_recovery(real_t p_delta);
void __pcr__fetch_recovery_info(
const uint32_t p_input_id,
bool &r_need_recover,
bool &r_recover_controller,
LocalVector<NetUtility::NodeData *> &r_nodes_to_recover,
LocalVector<NetUtility::PostponedRecover> &r_postponed_recover);
void __pcr__sync_pre_rewind(
const LocalVector<NetUtility::NodeData *> &p_nodes_to_recover);
void __pcr__rewind(
real_t p_delta,
const uint32_t p_checkable_input_id,
NetworkedController *p_controller,
PlayerController *p_player_controller,
const bool p_recover_controller,
const LocalVector<NetUtility::NodeData *> &p_nodes_to_recover);
void __pcr__sync_no_rewind(
const LocalVector<NetUtility::PostponedRecover> &p_postponed_recover);
void apply_last_received_server_snapshot();
void process_paused_controller_recovery(real_t p_delta);
bool parse_snapshot(Variant p_snapshot);
bool compare_vars(
@ -458,6 +593,10 @@ private:
Vector<NetUtility::Var> &r_postponed_recover);
void notify_server_full_snapshot_is_needed();
void send_actions_to_server();
void clean_pending_actions();
void check_missing_actions();
};
VARIANT_ENUM_CAST(NetEventFlag)

View File

@ -0,0 +1,708 @@
/*************************************************************************/
/* scene_synchronizer_debugger.cpp */
/*************************************************************************/
/* 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. */
/*************************************************************************/
#include "scene_synchronizer_debugger.h"
#ifdef DEBUG_ENABLED
#include "__generated__debugger_ui.h"
#include "core/io/json.h"
#include "core/os/dir_access.h"
#include "core/os/file_access.h"
#include "core/os/os.h"
#include "data_buffer.h"
#include "net_utilities.h"
#include "scene/main/viewport.h"
#include "scene_synchronizer.h"
#endif
SceneSynchronizerDebugger *SceneSynchronizerDebugger::the_singleton = nullptr;
SceneSynchronizerDebugger *SceneSynchronizerDebugger::singleton() {
return the_singleton;
}
void SceneSynchronizerDebugger::_bind_methods() {
ClassDB::bind_method(D_METHOD("on_node_added"), &SceneSynchronizerDebugger::on_node_added);
ClassDB::bind_method(D_METHOD("on_node_removed"), &SceneSynchronizerDebugger::on_node_removed);
ClassDB::bind_method(D_METHOD("add_node_message", "node", "message"), &SceneSynchronizerDebugger::add_node_message);
ClassDB::bind_method(D_METHOD("add_node_message_by_path", "node_path", "message"), &SceneSynchronizerDebugger::add_node_message_by_path);
ClassDB::bind_method(D_METHOD("debug_print", "node", "message", "silent"), &SceneSynchronizerDebugger::debug_print);
ClassDB::bind_method(D_METHOD("debug_warning", "node", "message", "silent"), &SceneSynchronizerDebugger::debug_warning);
ClassDB::bind_method(D_METHOD("debug_error", "node", "message", "silent"), &SceneSynchronizerDebugger::debug_error);
}
SceneSynchronizerDebugger::SceneSynchronizerDebugger() :
Node() {
if (the_singleton == nullptr) {
the_singleton = this;
}
#ifdef DEBUG_ENABLED
// Code here
#endif
}
SceneSynchronizerDebugger::~SceneSynchronizerDebugger() {
if (the_singleton == this) {
the_singleton = nullptr;
}
#ifdef DEBUG_ENABLED
tracked_nodes.reset();
classes_property_lists.clear();
frame_dump__begin_state.clear();
frame_dump__end_state.clear();
frame_dump__node_log.clear();
frame_dump__data_buffer_writes.clear();
frame_dump__data_buffer_reads.clear();
frame_dump__are_inputs_different_results.clear();
#endif
}
void SceneSynchronizerDebugger::set_dump_enabled(bool p_dump_enabled) {
#ifdef DEBUG_ENABLED
dump_enabled = p_dump_enabled;
#endif
}
bool SceneSynchronizerDebugger::get_dump_enabled() const {
#ifdef DEBUG_ENABLED
return dump_enabled;
#else
return false;
#endif
}
void SceneSynchronizerDebugger::register_class_for_node_to_dump(Node *p_node) {
#ifdef DEBUG_ENABLED
register_class_to_dump(p_node->get_class_name());
track_node(p_node, false);
#endif
}
void SceneSynchronizerDebugger::register_class_to_dump(const StringName &p_class) {
#ifdef DEBUG_ENABLED
ERR_FAIL_COND(p_class == StringName());
if (dump_classes.find(p_class) == -1) {
dump_classes.push_back(p_class);
}
#endif
}
void SceneSynchronizerDebugger::unregister_class_to_dump(const StringName &p_class) {
#ifdef DEBUG_ENABLED
const int64_t index = dump_classes.find(p_class);
if (index >= 0) {
dump_classes.remove_unordered(index);
}
#endif
}
void SceneSynchronizerDebugger::setup_debugger(const String &p_dump_name, int p_peer, SceneTree *p_scene_tree) {
#ifdef DEBUG_ENABLED
if (setup_done == false) {
setup_done = true;
// Setup `dump_enabled`
if (!dump_enabled) {
dump_enabled = GLOBAL_GET("NetworkSynchronizer/debugger/dump_enabled");
}
// Setup `dump_classes`.
{
Array classes = GLOBAL_GET("NetworkSynchronizer/debugger/dump_classes");
for (uint32_t i = 0; i < uint32_t(classes.size()); i += 1) {
if (classes[i].get_type() == Variant::STRING) {
register_class_to_dump(classes[i]);
}
}
}
}
// Setup directories.
main_dump_directory_path = OS::get_singleton()->get_executable_path().get_base_dir() + "/net-sync-debugs/dump";
dump_name = p_dump_name;
prepare_dumping(p_peer, p_scene_tree);
setup_debugger_python_ui();
#endif
}
void SceneSynchronizerDebugger::prepare_dumping(int p_peer, SceneTree *p_scene_tree) {
#ifdef DEBUG_ENABLED
if (!dump_enabled) {
// Dumping is disabled, nothing to do.
return;
}
// Prepare the dir.
{
String path = main_dump_directory_path + "/" + dump_name;
DirAccess *dir = DirAccess::create_for_path(path);
Error e;
e = dir->make_dir_recursive(path);
if (e != OK) {
memdelete(dir);
ERR_FAIL_COND(e != OK);
}
e = dir->change_dir(path);
if (e != OK) {
memdelete(dir);
ERR_FAIL_COND(e != OK);
}
// Empty the directory making sure we are ready to write.
dir->erase_contents_recursive();
//if (dir->list_dir_begin() == OK) {
// for (String name = dir->get_next(); name != String(); name = dir->get_next()) {
// if (name.begins_with("fd-" + itos(p_peer) + "-")) {
// dir->remove(name);
// }
// }
// dir->list_dir_end();
//}
// Dispose the pointer.
memdelete(dir);
}
// Store generic info about this dump.
{
Error e;
FileAccess *file = FileAccess::open(main_dump_directory_path + "/" + "dump-info-" + dump_name + /*"-" + itos(p_peer) +*/ ".json", FileAccess::WRITE, &e);
ERR_FAIL_COND(e != OK);
OS::Date date = OS::get_singleton()->get_date();
OS::Time time = OS::get_singleton()->get_time();
Dictionary d;
d["dump-name"] = dump_name;
d["peer"] = p_peer;
d["date"] = itos(date.day) + "/" + itos(date.month) + "/" + itos(date.year);
d["time"] = itos(time.hour) + "::" + itos(time.min);
file->flush();
file->store_string(JSON::print(d));
file->close();
memdelete(file);
}
if (scene_tree) {
scene_tree->disconnect("node_added", this, "on_node_added");
scene_tree->disconnect("node_removed", this, "on_node_removed");
}
tracked_nodes.clear();
classes_property_lists.clear();
scene_tree = p_scene_tree;
if (scene_tree) {
scene_tree->connect("node_added", this, "on_node_added");
scene_tree->connect("node_removed", this, "on_node_removed");
// Start by tracking the existing node.
track_node(scene_tree->get_root(), true);
}
#endif
}
void SceneSynchronizerDebugger::setup_debugger_python_ui() {
#ifdef DEBUG_ENABLED
// Verify if file exists.
const String path = main_dump_directory_path + "/debugger.py";
if (FileAccess::exists(path)) {
// Nothing to do.
return;
}
// Copy the python UI into the directory.
FileAccess *f = FileAccess::open(path, FileAccess::WRITE);
ERR_FAIL_COND_MSG(!f, "Can't create the `" + path + "` file.");
f->store_buffer((uint8_t *)__debugger_ui_code, __debugger_ui_code_size);
f->close();
memdelete(f);
#endif
}
void SceneSynchronizerDebugger::track_node(Node *p_node, bool p_recursive) {
#ifdef DEBUG_ENABLED
if (tracked_nodes.find(p_node) == -1) {
const bool is_tracked = dump_classes.find(p_node->get_class_name()) != -1;
if (is_tracked) {
// Verify if the property list already exists.
if (!classes_property_lists.has(p_node->get_class_name())) {
// Property list not yet cached, fetch it now.
classes_property_lists.insert(p_node->get_class_name(), List<PropertyInfo>());
List<PropertyInfo> *properties = classes_property_lists.lookup_ptr(p_node->get_class_name());
p_node->get_property_list(properties);
}
List<PropertyInfo> *properties = classes_property_lists.lookup_ptr(p_node->get_class_name());
// Can't happen, as it was just created.
CRASH_COND(properties == nullptr);
// Assign the property list pointer for fast access.
tracked_nodes.push_back(TrackedNode(p_node, properties));
}
}
if (p_recursive) {
for (int i = 0; i < p_node->get_child_count(); i += 1) {
Node *child = p_node->get_child(i);
track_node(child, true);
}
}
#endif
}
void SceneSynchronizerDebugger::on_node_added(Node *p_node) {
#ifdef DEBUG_ENABLED
track_node(p_node, false);
#endif
}
void SceneSynchronizerDebugger::on_node_removed(Node *p_node) {
#ifdef DEBUG_ENABLED
const int64_t index = tracked_nodes.find(p_node);
if (index != -1) {
tracked_nodes.remove_unordered(index);
}
#endif
}
void SceneSynchronizerDebugger::write_dump(int p_peer, uint32_t p_frame_index) {
#ifdef DEBUG_ENABLED
if (!dump_enabled) {
return;
}
if (p_frame_index == UINT32_MAX) {
// Nothing to write.
return;
}
FileAccess *file = nullptr;
{
String file_path = "";
int iteration = 0;
String iteration_mark = "";
do {
file_path = main_dump_directory_path + "/" + dump_name + "/fd-" /*+ itos(p_peer) + "-"*/ + itos(p_frame_index) + iteration_mark + ".json";
iteration_mark += "@";
iteration += 1;
} while (FileAccess::exists(file_path) && iteration < 100);
Error e;
file = FileAccess::open(file_path, FileAccess::WRITE, &e);
ERR_FAIL_COND(e != OK);
}
String frame_summary;
if (frame_dump__has_warnings) {
frame_summary += "* ";
} else if (frame_dump__has_errors) {
frame_summary += "! ";
}
if ((frame_dump__frame_events & FrameEvent::CLIENT_DESYNC_DETECTED) > 0) {
frame_summary += "Client desync; ";
} else if ((frame_dump__frame_events & FrameEvent::CLIENT_DESYNC_DETECTED_SOFT) > 0) {
frame_summary += "Client desync; No controller rewind; ";
}
Dictionary d;
d["frame"] = itos(p_frame_index);
d["peer"] = itos(p_peer);
d["frame_summary"] = frame_summary;
d["begin_state"] = frame_dump__begin_state;
d["end_state"] = frame_dump__end_state;
d["node_log"] = frame_dump__node_log;
d["data_buffer_writes"] = frame_dump__data_buffer_writes;
d["data_buffer_reads"] = frame_dump__data_buffer_reads;
d["are_inputs_different_results"] = frame_dump__are_inputs_different_results;
file->store_string(JSON::print(d));
file->close();
memdelete(file);
#endif
}
void SceneSynchronizerDebugger::start_new_frame() {
#ifdef DEBUG_ENABLED
frame_dump__node_log.clear();
frame_dump__frame_events = FrameEvent::EMPTY;
frame_dump__has_warnings = false;
frame_dump__has_errors = false;
frame_dump__data_buffer_writes.clear();
frame_dump__data_buffer_reads.clear();
frame_dump__are_inputs_different_results.clear();
log_counter = 0;
#endif
}
#ifdef DEBUG_ENABLED
String type_to_string(Variant::Type p_type) {
switch (p_type) {
case Variant::NIL:
return "NIL";
case Variant::BOOL:
return "BOOL";
case Variant::INT:
return "INT";
case Variant::REAL:
return "REAL";
case Variant::STRING:
return "STRING";
case Variant::VECTOR2:
return "VECTOR2";
case Variant::RECT2:
return "RECT2";
case Variant::VECTOR3:
return "VECTOR3";
case Variant::TRANSFORM2D:
return "TRANSFORM2D";
case Variant::PLANE:
return "PLANE";
case Variant::QUAT:
return "QUAT";
case Variant::AABB:
return "AABB";
case Variant::BASIS:
return "BASIS";
case Variant::TRANSFORM:
return "TRANSFORM";
case Variant::COLOR:
return "COLOR";
case Variant::NODE_PATH:
return "NODE_PATH";
case Variant::_RID:
return "_RID";
case Variant::OBJECT:
return "OBJECT";
case Variant::DICTIONARY:
return "DICTIONARY";
case Variant::ARRAY:
return "ARRAY";
case Variant::POOL_BYTE_ARRAY:
return "POOL_BYTE_ARRAY";
case Variant::POOL_INT_ARRAY:
return "POOL_INT_ARRAY";
case Variant::POOL_REAL_ARRAY:
return "POOL_REAL_ARRAY";
case Variant::POOL_STRING_ARRAY:
return "POOL_STRING_ARRAY";
case Variant::POOL_VECTOR2_ARRAY:
return "POOL_VECTOR2_ARRAY";
case Variant::POOL_VECTOR3_ARRAY:
return "POOL_VECTOR3_ARRAY";
case Variant::POOL_COLOR_ARRAY:
return "POOL_COLOR_ARRAY";
case Variant::VARIANT_MAX:
return "VARIANT_MAX";
}
return "";
}
String data_type_to_string(uint32_t p_type) {
switch (p_type) {
case DataBuffer::DATA_TYPE_BOOL:
return "Bool";
case DataBuffer::DATA_TYPE_INT:
return "Int";
case DataBuffer::DATA_TYPE_UINT:
return "Uint";
case DataBuffer::DATA_TYPE_REAL:
return "Real";
case DataBuffer::DATA_TYPE_POSITIVE_UNIT_REAL:
return "Positive Unit Real";
case DataBuffer::DATA_TYPE_UNIT_REAL:
return "Unit Real";
case DataBuffer::DATA_TYPE_VECTOR2:
return "Vector2";
case DataBuffer::DATA_TYPE_NORMALIZED_VECTOR2:
return "Normalized Vector2";
case DataBuffer::DATA_TYPE_VECTOR3:
return "Vector3";
case DataBuffer::DATA_TYPE_NORMALIZED_VECTOR3:
return "Normalized Vector3";
case DataBuffer::DATA_TYPE_VARIANT:
return "Variant";
}
return "UNDEFINED";
}
String compression_level_to_string(uint32_t p_type) {
switch (p_type) {
case DataBuffer::COMPRESSION_LEVEL_0:
return "Compression Level 0";
case DataBuffer::COMPRESSION_LEVEL_1:
return "Compression Level 1";
case DataBuffer::COMPRESSION_LEVEL_2:
return "Compression Level 2";
case DataBuffer::COMPRESSION_LEVEL_3:
return "Compression Level 3";
}
return "Compression Level UNDEFINED";
}
#endif
void SceneSynchronizerDebugger::scene_sync_process_start(const SceneSynchronizer *p_scene_sync) {
#ifdef DEBUG_ENABLED
if (!dump_enabled) {
return;
}
dump_tracked_objects(p_scene_sync, frame_dump__begin_state);
#endif
}
void SceneSynchronizerDebugger::scene_sync_process_end(const SceneSynchronizer *p_scene_sync) {
#ifdef DEBUG_ENABLED
if (!dump_enabled) {
return;
}
dump_tracked_objects(p_scene_sync, frame_dump__end_state);
#endif
}
void SceneSynchronizerDebugger::databuffer_operation_begin_record(Node *p_node, DataBufferDumpMode p_mode) {
#ifdef DEBUG_ENABLED
if (!dump_enabled) {
return;
}
frame_dump__data_buffer_name = p_node->get_path();
frame_dump_data_buffer_dump_mode = p_mode;
if (frame_dump_data_buffer_dump_mode == DataBufferDumpMode::WRITE) {
add_node_message_by_path(frame_dump__data_buffer_name, "[WRITE] DataBuffer start write");
} else {
add_node_message_by_path(frame_dump__data_buffer_name, "[READ] DataBuffer start read");
}
#endif
}
void SceneSynchronizerDebugger::databuffer_operation_end_record() {
#ifdef DEBUG_ENABLED
if (!dump_enabled) {
return;
}
if (frame_dump_data_buffer_dump_mode == DataBufferDumpMode::WRITE) {
add_node_message_by_path(frame_dump__data_buffer_name, "[WRITE] end");
} else {
add_node_message_by_path(frame_dump__data_buffer_name, "[READ] end");
}
frame_dump_data_buffer_dump_mode = DataBufferDumpMode::NONE;
frame_dump__data_buffer_name = "";
#endif
}
void SceneSynchronizerDebugger::databuffer_write(uint32_t p_data_type, uint32_t p_compression_level, Variant p_variable) {
#ifdef DEBUG_ENABLED
if (!dump_enabled) {
return;
}
if (frame_dump_data_buffer_dump_mode != DataBufferDumpMode::WRITE) {
return;
}
List<const void *> stack;
String val_string = p_variable.stringify(stack);
frame_dump__data_buffer_writes.push_back(val_string);
const String operation = "[WRITE] [" + compression_level_to_string(p_compression_level) + "] [" + data_type_to_string(p_data_type) + "] " + val_string;
add_node_message_by_path(frame_dump__data_buffer_name, operation);
#endif
}
void SceneSynchronizerDebugger::databuffer_read(uint32_t p_data_type, uint32_t p_compression_level, Variant p_variable) {
#ifdef DEBUG_ENABLED
if (!dump_enabled) {
return;
}
if (frame_dump_data_buffer_dump_mode != DataBufferDumpMode::READ) {
return;
}
List<const void *> stack;
String val_string = p_variable.stringify(stack);
frame_dump__data_buffer_reads.push_back(val_string);
const String operation = "[READ] [" + compression_level_to_string(p_compression_level) + "] [" + data_type_to_string(p_data_type) + "] " + val_string;
add_node_message_by_path(frame_dump__data_buffer_name, operation);
#endif
}
void SceneSynchronizerDebugger::notify_input_sent_to_server(Node *p_node, uint32_t p_frame_index, uint32_t p_input_index) {
#ifdef DEBUG_ENABLED
debug_print(p_node, "The client sent to server the input `" + itos(p_input_index) + "` for frame:`" + itos(p_frame_index) + "`.", true);
#endif
}
void SceneSynchronizerDebugger::notify_are_inputs_different_result(
Node *p_node,
uint32_t p_other_frame_index,
bool p_is_similar) {
#ifdef DEBUG_ENABLED
if (p_is_similar) {
debug_print(p_node, "This frame input is SIMILAR to `" + itos(p_other_frame_index) + "`", true);
} else {
debug_print(p_node, "This frame input is DIFFERENT to `" + itos(p_other_frame_index) + "`", true);
}
frame_dump__are_inputs_different_results[p_other_frame_index] = p_is_similar;
#endif
}
void SceneSynchronizerDebugger::add_node_message(Node *p_node, const String &p_message) {
#ifdef DEBUG_ENABLED
if (p_node) {
add_node_message_by_path(p_node->get_path(), p_message);
} else {
add_node_message_by_path("GLOBAL", p_message);
}
#endif
}
void SceneSynchronizerDebugger::add_node_message_by_path(const String &p_node_path, const String &p_message) {
#ifdef DEBUG_ENABLED
if (!dump_enabled) {
return;
}
if (!frame_dump__node_log.has(p_node_path)) {
frame_dump__node_log[p_node_path] = Array();
}
Variant *v = frame_dump__node_log.getptr(p_node_path);
Array a = *v;
Dictionary m;
m["i"] = log_counter;
m["m"] = p_message;
a.append(m);
log_counter += 1;
#endif
}
void SceneSynchronizerDebugger::debug_print(Node *p_node, const String &p_message, bool p_silent) {
#ifdef DEBUG_ENABLED
if (!p_silent) {
NET_DEBUG_PRINT(p_message);
}
add_node_message(p_node, "[INFO] " + p_message);
#endif
}
void SceneSynchronizerDebugger::debug_warning(Node *p_node, const String &p_message, bool p_silent) {
#ifdef DEBUG_ENABLED
if (!p_silent) {
NET_DEBUG_WARN(p_message);
}
add_node_message(p_node, "[WARNING] " + p_message);
frame_dump__has_warnings = true;
#endif
}
void SceneSynchronizerDebugger::debug_error(Node *p_node, const String &p_message, bool p_silent) {
#ifdef DEBUG_ENABLED
if (!p_silent) {
NET_DEBUG_ERR(p_message);
}
add_node_message(p_node, "[ERROR] " + p_message);
frame_dump__has_errors = true;
#endif
}
void SceneSynchronizerDebugger::notify_event(FrameEvent p_event) {
#ifdef DEBUG_ENABLED
if (!dump_enabled) {
return;
}
frame_dump__frame_events = FrameEvent(frame_dump__frame_events | p_event);
#endif
}
void SceneSynchronizerDebugger::dump_tracked_objects(const SceneSynchronizer *p_scene_sync, Dictionary &p_dump) {
#ifdef DEBUG_ENABLED
p_dump.clear();
List<const void *> stack;
for (uint32_t i = 0; i < tracked_nodes.size(); i += 1) {
Dictionary node_dump;
String node_path = tracked_nodes[i].node->get_path();
node_dump["node_path"] = node_path;
for (List<PropertyInfo>::Element *e = tracked_nodes[i].properties->front(); e; e = e->next()) {
String prefix;
if (p_scene_sync->is_variable_registered(tracked_nodes[i].node, e->get().name)) {
prefix = "* ";
}
node_dump[prefix + e->get().name + "::" + type_to_string(e->get().type)] = tracked_nodes[i].node->get(e->get().name).stringify(stack);
}
p_dump[node_path] = node_dump;
}
#endif
}

View File

@ -0,0 +1,171 @@
/*************************************************************************/
/* scene_synchronizer_debugger.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. */
/*************************************************************************/
#pragma once
#include "scene/main/node.h"
#ifdef DEBUG_ENABLED
#include "core/oa_hash_map.h"
#endif
class SceneSynchronizer;
class SceneTree;
class SceneSynchronizerDebugger : public Node {
GDCLASS(SceneSynchronizerDebugger, Node);
static SceneSynchronizerDebugger *the_singleton;
public:
struct TrackedNode {
Node *node;
List<PropertyInfo> *properties;
TrackedNode() = default;
TrackedNode(Node *p_node) :
node(p_node) {}
TrackedNode(Node *p_node, List<PropertyInfo> *p_properties) :
node(p_node), properties(p_properties) {}
bool operator==(const TrackedNode &p_t) const { return p_t.node == node; }
};
enum DataBufferDumpMode {
NONE,
WRITE,
READ
};
enum FrameEvent : uint32_t {
EMPTY = 0,
CLIENT_DESYNC_DETECTED = 1 << 0,
CLIENT_DESYNC_DETECTED_SOFT = 1 << 0,
};
public:
static SceneSynchronizerDebugger *singleton();
static void _bind_methods();
private:
#ifdef DEBUG_ENABLED
bool dump_enabled = false;
LocalVector<StringName> dump_classes;
bool setup_done = false;
uint32_t log_counter = 0;
SceneTree *scene_tree = nullptr;
String main_dump_directory_path;
String dump_name;
LocalVector<TrackedNode> tracked_nodes;
// HaskMap between class name and property list: to avoid fetching the property list per object each frame.
OAHashMap<StringName, List<PropertyInfo>> classes_property_lists;
// Dictionary of dictionary containing nodes info.
Dictionary frame_dump__begin_state;
// Dictionary of dictionary containing nodes info.
Dictionary frame_dump__end_state;
// The dictionary containing the data buffer operations performed by the controllers.
Dictionary frame_dump__node_log;
// The controller name for which the data buffer operations is in progress.
String frame_dump__data_buffer_name;
// A really small description about what happens on this frame.
FrameEvent frame_dump__frame_events = FrameEvent::EMPTY;
// This Array contains all the inputs (stringified) written on the `DataBuffer` from the
// `_controller_process` function
Array frame_dump__data_buffer_writes;
// This Array contains all the inputs (stringified) read on the `DataBuffer` from the
// `_controller_process` function
Array frame_dump__data_buffer_reads;
// This Dictionary contains the comparison (`_are_inputs_different`) fetched by this frame, and
// the result.
Dictionary frame_dump__are_inputs_different_results;
DataBufferDumpMode frame_dump_data_buffer_dump_mode = NONE;
bool frame_dump__has_warnings = false;
bool frame_dump__has_errors = false;
#endif
public:
SceneSynchronizerDebugger();
~SceneSynchronizerDebugger();
void set_dump_enabled(bool p_dump_enabled);
bool get_dump_enabled() const;
void register_class_for_node_to_dump(Node *p_node);
void register_class_to_dump(const StringName &p_class);
void unregister_class_to_dump(const StringName &p_class);
void setup_debugger(const String &p_dump_name, int p_peer, SceneTree *p_scene_tree);
private:
void prepare_dumping(int p_peer, SceneTree *p_scene_tree);
void setup_debugger_python_ui();
void track_node(Node *p_node, bool p_recursive);
void on_node_added(Node *p_node);
void on_node_removed(Node *p_node);
public:
void write_dump(int p_peer, uint32_t p_frame_index);
void start_new_frame();
void scene_sync_process_start(const SceneSynchronizer *p_scene_sync);
void scene_sync_process_end(const SceneSynchronizer *p_scene_sync);
void databuffer_operation_begin_record(Node *p_node, DataBufferDumpMode p_mode);
void databuffer_operation_end_record();
void databuffer_write(uint32_t p_data_type, uint32_t p_compression_level, Variant p_variable);
void databuffer_read(uint32_t p_data_type, uint32_t p_compression_level, Variant p_variable);
void notify_input_sent_to_server(Node *p_node, uint32_t p_frame_index, uint32_t p_input_index);
void notify_are_inputs_different_result(Node *p_node, uint32_t p_other_frame_index, bool p_is_similar);
void add_node_message(Node *p_node, const String &p_message);
void add_node_message_by_path(const String &p_node_path, const String &p_message);
void debug_print(Node *p_node, const String &p_message, bool p_silent = false);
void debug_warning(Node *p_node, const String &p_message, bool p_silent = false);
void debug_error(Node *p_node, const String &p_message, bool p_silent = false);
void notify_event(FrameEvent p_event);
private:
void dump_tracked_objects(const SceneSynchronizer *p_scene_sync, Dictionary &p_dump);
};

View File

@ -1,121 +0,0 @@
/*************************************************************************/
/* test_bit_array.h */
/*************************************************************************/
/* This file is part of: */
/* PANDEMONIUM ENGINE */
/* https://github.com/Relintai/pandemonium_engine */
/*************************************************************************/
/* Copyright (c) 2022-present Péter Magyar. */
/* Copyright (c) 2014-2022 Godot Engine contributors (cf. AUTHORS.md). */
/* Copyright (c) 2007-2022 Juan Linietsky, Ariel Manzur. */
/* */
/* 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. */
/*************************************************************************/
#ifndef TEST_BIT_ARRAY_H
#define TEST_BIT_ARRAY_H
#include "modules/network_synchronizer/bit_array.h"
#include "tests/test_macros.h"
namespace TestBitArray {
TEST_CASE("[Modules][BitArray] Read and write") {
BitArray array;
int offset = 0;
int bits = {};
uint64_t value = {};
SUBCASE("[Modules][BitArray] One bit") {
bits = 1;
SUBCASE("[Modules][BitArray] One") {
value = 0b1;
}
SUBCASE("[Modules][BitArray] Zero") {
value = 0b0;
}
}
SUBCASE("[Modules][BitArray] 16 mixed bits") {
bits = 16;
value = 0b1010101010101010;
}
SUBCASE("[Modules][BitArray] One and 4 zeroes") {
bits = 5;
value = 0b10000;
}
SUBCASE("[Modules][BitArray] 64 bits") {
bits = 64;
SUBCASE("[Modules][BitArray] One") {
value = UINT64_MAX;
}
SUBCASE("[Modules][BitArray] Zero") {
value = 0;
}
}
SUBCASE("[Modules][BitArray] One bit with offset") {
bits = 1;
offset = 64;
array.resize_in_bits(offset);
SUBCASE("[Modules][BitArray] One") {
array.store_bits(0, UINT64_MAX, 64);
value = 0b0;
}
SUBCASE("[Modules][BitArray] Zero") {
array.store_bits(0, 0, 64);
value = 0b1;
}
}
array.resize_in_bits(offset + bits);
array.store_bits(offset, value, bits);
CHECK_MESSAGE(array.read_bits(offset, bits) == value, "Should read the same value");
}
TEST_CASE("[Modules][BitArray] Constructing from Vector") {
Vector<uint8_t> data;
data.push_back(-1);
data.push_back(0);
data.push_back(1);
const BitArray array(data);
CHECK_MESSAGE(array.size_in_bits() == data.size() * 8.0, "Number of bits must be equal to size of original data");
CHECK_MESSAGE(array.size_in_bytes() == data.size(), "Number of bytes must be equal to size of original data");
for (int i = 0; i < data.size(); ++i) {
CHECK_MESSAGE(array.read_bits(i * 8, 8) == data[i], "Readed bits should be equal to the original");
}
}
TEST_CASE("[Modules][BitArray] Pre-allocation and zeroing") {
constexpr uint64_t value = UINT64_MAX;
constexpr int bits = sizeof(value);
BitArray array(bits);
CHECK_MESSAGE(array.size_in_bits() == bits, "Number of bits must be equal to allocated");
array.store_bits(0, value, bits);
array.zero();
CHECK_MESSAGE(array.read_bits(0, bits) == 0, "Should read zero");
}
} // namespace TestBitArray
#endif // TEST_BIT_ARRAY_H

View File

@ -1,551 +0,0 @@
/*************************************************************************/
/* test_data_buffer.h */
/*************************************************************************/
/* This file is part of: */
/* PANDEMONIUM ENGINE */
/* https://github.com/Relintai/pandemonium_engine */
/*************************************************************************/
/* Copyright (c) 2022-present Péter Magyar. */
/* Copyright (c) 2014-2022 Godot Engine contributors (cf. AUTHORS.md). */
/* Copyright (c) 2007-2022 Juan Linietsky, Ariel Manzur. */
/* */
/* 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. */
/*************************************************************************/
#ifndef TEST_DATA_BUFFER_H
#define TEST_DATA_BUFFER_H
#include "modules/network_synchronizer/data_buffer.h"
#include "modules/network_synchronizer/scene_synchronizer.h"
#include "tests/test_macros.h"
namespace TestDataBuffer {
inline Vector<double> real_values(DataBuffer::CompressionLevel p_compression_level) {
Vector<double> values;
values.append(Math_PI);
values.append(0.0);
values.append(-3.04);
values.append(3.04);
values.append(0.5);
values.append(-0.5);
values.append(1);
values.append(-1);
values.append(0.9);
values.append(-0.9);
values.append(3.9);
values.append(-3.9);
values.append(8);
switch (p_compression_level) {
case DataBuffer::COMPRESSION_LEVEL_3: {
values.append(-15'360);
values.append(15'360);
} break;
case DataBuffer::COMPRESSION_LEVEL_2: {
// https://en.wikipedia.org/wiki/Half-precision_floating-point_format#Half_precision_examples
values.append(-65'504);
values.append(65'504);
values.append(Math::pow(2.0, -14) / 1024);
values.append(Math::pow(2.0, -14) * 1023 / 1024);
values.append(Math::pow(2.0, -1) * (1 + 1023.0 / 1024));
values.append((1 + 1.0 / 1024));
} break;
case DataBuffer::COMPRESSION_LEVEL_1: {
// https://en.wikipedia.org/wiki/Single-precision_floating-point_format#Single-precision_examples
values.append(FLT_MIN);
values.append(-FLT_MAX);
values.append(FLT_MAX);
values.append(Math::pow(2.0, -149));
values.append(Math::pow(2.0, -126) * (1 - Math::pow(2.0, -23)));
values.append(1 - Math::pow(2.0, -24));
values.append(1 + Math::pow(2.0, -23));
} break;
case DataBuffer::COMPRESSION_LEVEL_0: {
// https://en.wikipedia.org/wiki/Double-precision_floating-point_format#Double-precision_examples
values.append(DBL_MIN);
values.append(DBL_MAX);
values.append(-DBL_MAX);
values.append(1.0000000000000002);
values.append(4.9406564584124654 * Math::pow(10.0, -324));
values.append(2.2250738585072009 * Math::pow(10.0, -308));
} break;
}
return values;
}
TEST_CASE("[Modules][DataBuffer] Bool") {
bool value = {};
SUBCASE("[Modules][DataBuffer] false") {
value = false;
}
SUBCASE("[Modules][DataBuffer] true") {
value = true;
}
DataBuffer buffer;
buffer.begin_write(0);
CHECK_MESSAGE(buffer.add_bool(value) == value, "Should return the same value");
buffer.begin_read();
CHECK_MESSAGE(buffer.read_bool() == value, "Should read the same value");
}
TEST_CASE("[Modules][DataBuffer] Int") {
DataBuffer::CompressionLevel compression_level = {};
int64_t value = {};
DataBuffer buffer;
SUBCASE("[Modules][DataBuffer] Compression level 3") {
compression_level = DataBuffer::COMPRESSION_LEVEL_3;
SUBCASE("[Modules][DataBuffer] Positive") {
value = 127;
}
SUBCASE("[Modules][DataBuffer] Zero") {
value = 0;
}
SUBCASE("[Modules][DataBuffer] Negative") {
value = -128;
}
}
SUBCASE("[Modules][DataBuffer] Compression level 2") {
compression_level = DataBuffer::COMPRESSION_LEVEL_2;
SUBCASE("[Modules][DataBuffer] Positive") {
value = 32767;
}
SUBCASE("[Modules][DataBuffer] Zero") {
value = 0;
}
SUBCASE("[Modules][DataBuffer] Negative") {
value = -32768;
}
}
SUBCASE("[Modules][DataBuffer] Compression level 1") {
compression_level = DataBuffer::COMPRESSION_LEVEL_1;
SUBCASE("[Modules][DataBuffer] Positive") {
value = 2147483647;
}
SUBCASE("[Modules][DataBuffer] Zero") {
value = 0;
}
SUBCASE("[Modules][DataBuffer] Negative") {
value = -2147483648LL;
}
}
SUBCASE("[Modules][DataBuffer] Compression level 0") {
compression_level = DataBuffer::COMPRESSION_LEVEL_0;
SUBCASE("[Modules][DataBuffer] Positive") {
value = 2147483647;
}
SUBCASE("[Modules][DataBuffer] Zero") {
value = 0;
}
SUBCASE("[Modules][DataBuffer] Negative") {
value = -9223372036854775807LL;
}
}
buffer.begin_write(0);
CHECK_MESSAGE(buffer.add_int(value, compression_level) == value, "Should return the same value");
buffer.begin_read();
CHECK_MESSAGE(buffer.read_int(compression_level) == value, "Should read the same value");
}
TEST_CASE("[Modules][DataBuffer] Real") {
DataBuffer::CompressionLevel compression_level = {};
SUBCASE("[Modules][DataBuffer] Compression level 3 (Minifloat)") {
compression_level = DataBuffer::COMPRESSION_LEVEL_3;
}
SUBCASE("[Modules][DataBuffer] Compression level 2 (Half perception)") {
compression_level = DataBuffer::COMPRESSION_LEVEL_2;
}
SUBCASE("[Modules][DataBuffer] Compression level 1 (Single perception)") {
compression_level = DataBuffer::COMPRESSION_LEVEL_1;
}
SUBCASE("[Modules][DataBuffer] Compression level 0 (Double perception)") {
compression_level = DataBuffer::COMPRESSION_LEVEL_0;
}
DataBuffer buffer;
const Vector<double> values = real_values(compression_level);
const double epsilon = Math::pow(2.0, DataBuffer::get_mantissa_bits(compression_level) - 1);
for (int i = 0; i < values.size(); ++i) {
buffer.begin_write(0);
const double value = values[i];
CHECK_MESSAGE(buffer.add_real(value, compression_level) == doctest::Approx(value).epsilon(epsilon), "Should return the same value");
buffer.begin_read();
CHECK_MESSAGE(buffer.read_real(compression_level) == doctest::Approx(value).epsilon(epsilon), "Should read the same value");
}
}
TEST_CASE("[Modules][DataBuffer] Positive unit real") {
DataBuffer::CompressionLevel compression_level = {};
double epsilon = {};
SUBCASE("[Modules][DataBuffer] Compression level 3") {
compression_level = DataBuffer::COMPRESSION_LEVEL_3;
epsilon = 0.033335;
}
SUBCASE("[Modules][DataBuffer] Compression level 2") {
compression_level = DataBuffer::COMPRESSION_LEVEL_2;
epsilon = 0.007935;
}
SUBCASE("[Modules][DataBuffer] Compression level 1") {
compression_level = DataBuffer::COMPRESSION_LEVEL_1;
epsilon = 0.00196;
}
SUBCASE("[Modules][DataBuffer] Compression level 0") {
compression_level = DataBuffer::COMPRESSION_LEVEL_0;
epsilon = 0.00049;
}
DataBuffer buffer;
const Vector<double> values = real_values(compression_level);
for (int i = 0; i < values.size(); ++i) {
const double value = values[i];
if (value < 0) {
// Skip negative values
continue;
}
double value_integral;
const double value_unit = modf(values[i], &value_integral);
buffer.begin_write(0);
CHECK_MESSAGE(buffer.add_positive_unit_real(value_unit, compression_level) == doctest::Approx(value_unit).epsilon(epsilon), "Should return the same value");
buffer.begin_read();
CHECK_MESSAGE(buffer.read_positive_unit_real(compression_level) == doctest::Approx(value_unit).epsilon(epsilon), "Should read the same value");
}
}
TEST_CASE("[Modules][DataBuffer] Unit real") {
DataBuffer::CompressionLevel compression_level = {};
double epsilon = {};
SUBCASE("[Modules][DataBuffer] Compression level 3") {
compression_level = DataBuffer::COMPRESSION_LEVEL_3;
epsilon = 0.033335;
}
SUBCASE("[Modules][DataBuffer] Compression level 2") {
compression_level = DataBuffer::COMPRESSION_LEVEL_2;
epsilon = 0.007935;
}
SUBCASE("[Modules][DataBuffer] Compression level 1") {
compression_level = DataBuffer::COMPRESSION_LEVEL_1;
epsilon = 0.00196;
}
SUBCASE("[Modules][DataBuffer] Compression level 0") {
compression_level = DataBuffer::COMPRESSION_LEVEL_0;
epsilon = 0.00049;
}
DataBuffer buffer;
const Vector<double> values = real_values(compression_level);
for (int i = 0; i < values.size(); ++i) {
double value_integral;
const double value_unit = modf(values[i], &value_integral);
buffer.begin_write(0);
CHECK_MESSAGE(buffer.add_unit_real(value_unit, compression_level) == doctest::Approx(value_unit).epsilon(epsilon), "Should return the same value");
buffer.begin_read();
CHECK_MESSAGE(buffer.read_unit_real(compression_level) == doctest::Approx(value_unit).epsilon(epsilon), "Should read the same value");
}
}
TEST_CASE("[Modules][DataBuffer] Vector2") {
DataBuffer::CompressionLevel compression_level = {};
SUBCASE("[Modules][DataBuffer] Compression level 3") {
compression_level = DataBuffer::COMPRESSION_LEVEL_3;
}
SUBCASE("[Modules][DataBuffer] Compression level 2") {
compression_level = DataBuffer::COMPRESSION_LEVEL_2;
}
SUBCASE("[Modules][DataBuffer] Compression level 1") {
compression_level = DataBuffer::COMPRESSION_LEVEL_1;
}
SUBCASE("[Modules][DataBuffer] Compression level 0") {
compression_level = DataBuffer::COMPRESSION_LEVEL_0;
}
DataBuffer buffer;
const double epsilon = Math::pow(2.0, DataBuffer::get_mantissa_bits(compression_level) - 1);
const Vector<double> values = real_values(compression_level);
for (int i = 0; i < values.size(); ++i) {
#ifdef REAL_T_IS_DOUBLE
const Vector2 value = Vector2(values[i], values[i]);
#else
const real_t clamped_value = CLAMP(values[i], -FLT_MIN, FLT_MAX);
const Vector2 value = Vector2(clamped_value, clamped_value);
#endif
buffer.begin_write(0);
const Vector2 added_value = buffer.add_vector2(value, compression_level);
CHECK_MESSAGE(added_value.x == doctest::Approx(value.x).epsilon(epsilon), "Added Vector2 should have the same x axis");
CHECK_MESSAGE(added_value.y == doctest::Approx(value.y).epsilon(epsilon), "Added Vector2 should have the same y axis");
buffer.begin_read();
const Vector2 read_value = buffer.read_vector2(compression_level);
CHECK_MESSAGE(read_value.x == doctest::Approx(value.x).epsilon(epsilon), "Read Vector2 should have the same x axis");
CHECK_MESSAGE(read_value.y == doctest::Approx(value.y).epsilon(epsilon), "Read Vector2 should have the same y axis");
}
}
TEST_CASE("[Modules][DataBuffer] Vector3") {
DataBuffer::CompressionLevel compression_level = {};
SUBCASE("[Modules][DataBuffer] Compression level 3") {
compression_level = DataBuffer::COMPRESSION_LEVEL_3;
}
SUBCASE("[Modules][DataBuffer] Compression level 2") {
compression_level = DataBuffer::COMPRESSION_LEVEL_2;
}
SUBCASE("[Modules][DataBuffer] Compression level 1") {
compression_level = DataBuffer::COMPRESSION_LEVEL_1;
}
SUBCASE("[Modules][DataBuffer] Compression level 0") {
compression_level = DataBuffer::COMPRESSION_LEVEL_0;
}
DataBuffer buffer;
const Vector<double> values = real_values(compression_level);
const double epsilon = Math::pow(2.0, DataBuffer::get_mantissa_bits(compression_level) - 1);
for (int i = 0; i < values.size(); ++i) {
#ifdef REAL_T_IS_DOUBLE
const Vector3 value = Vector3(values[i], values[i], values[i]);
#else
const real_t clamped_value = CLAMP(values[i], -FLT_MIN, FLT_MAX);
const Vector3 value = Vector3(clamped_value, clamped_value, clamped_value);
#endif
buffer.begin_write(0);
const Vector3 added_value = buffer.add_vector3(value, compression_level);
CHECK_MESSAGE(added_value.x == doctest::Approx(value.x).epsilon(epsilon), "Added Vector3 should have the same x axis");
CHECK_MESSAGE(added_value.y == doctest::Approx(value.y).epsilon(epsilon), "Added Vector3 should have the same y axis");
CHECK_MESSAGE(added_value.z == doctest::Approx(value.z).epsilon(epsilon), "Added Vector3 should have the same z axis");
buffer.begin_read();
const Vector3 read_value = buffer.read_vector3(compression_level);
CHECK_MESSAGE(read_value.x == doctest::Approx(value.x).epsilon(epsilon), "Read Vector3 should have the same x axis");
CHECK_MESSAGE(read_value.y == doctest::Approx(value.y).epsilon(epsilon), "Read Vector3 should have the same y axis");
CHECK_MESSAGE(read_value.z == doctest::Approx(value.z).epsilon(epsilon), "Read Vector3 should have the same z axis");
}
}
TEST_CASE("[Modules][DataBuffer] Normalized Vector3") {
DataBuffer::CompressionLevel compression_level = {};
double epsilon = {};
SUBCASE("[Modules][DataBuffer] Compression level 3") {
compression_level = DataBuffer::COMPRESSION_LEVEL_3;
epsilon = 0.033335;
}
SUBCASE("[Modules][DataBuffer] Compression level 2") {
compression_level = DataBuffer::COMPRESSION_LEVEL_2;
epsilon = 0.007935;
}
SUBCASE("[Modules][DataBuffer] Compression level 1") {
compression_level = DataBuffer::COMPRESSION_LEVEL_1;
epsilon = 0.00196;
}
SUBCASE("[Modules][DataBuffer] Compression level 0") {
compression_level = DataBuffer::COMPRESSION_LEVEL_0;
epsilon = 0.00049;
}
DataBuffer buffer;
const Vector<double> values = real_values(compression_level);
for (int i = 0; i < values.size(); ++i) {
Vector3 value = Vector3(values[i], values[i], values[i]).normalized();
if (!value.is_normalized()) {
// Normalization fails for some numbers, probably a bug!
continue;
}
buffer.begin_write(0);
const Vector3 added_value = buffer.add_normalized_vector3(value, compression_level);
CHECK_MESSAGE(added_value.x == doctest::Approx(value.x).epsilon(epsilon), "Added Vector3 should have the same x axis");
CHECK_MESSAGE(added_value.y == doctest::Approx(value.y).epsilon(epsilon), "Added Vector3 should have the same y axis");
CHECK_MESSAGE(added_value.z == doctest::Approx(value.z).epsilon(epsilon), "Added Vector3 should have the same z axis");
buffer.begin_read();
const Vector3 read_value = buffer.read_normalized_vector3(compression_level);
CHECK_MESSAGE(read_value.x == doctest::Approx(value.x).epsilon(epsilon), "Read Vector3 should have the same x axis");
CHECK_MESSAGE(read_value.y == doctest::Approx(value.y).epsilon(epsilon), "Read Vector3 should have the same y axis");
CHECK_MESSAGE(read_value.z == doctest::Approx(value.z).epsilon(epsilon), "Read Vector3 should have the same z axis");
}
}
TEST_CASE("[Modules][DataBuffer] Variant") {
Variant value = {};
SUBCASE("[Modules][DataBuffer] Invalid value") {
value = {};
}
SUBCASE("[Modules][DataBuffer] String") {
value = "VariantString";
}
SUBCASE("[Modules][DataBuffer] Vector") {
value = sarray("VariantString1", "VariantString2", "VariantString3");
}
SUBCASE("[Modules][DataBuffer] Dictionary") {
Dictionary dictionary;
dictionary[1] = "Value";
dictionary["Key"] = -1;
value = dictionary;
}
SUBCASE("[Modules][DataBuffer] Array") {
Array array;
array.append("VariantString");
array.append(0);
array.append(-1.2);
value = array;
}
DataBuffer buffer;
buffer.begin_write(0);
CHECK_MESSAGE(SceneSynchronizer::compare(buffer.add_variant(value), value, DBL_EPSILON), "Should return the same value");
buffer.begin_read();
CHECK_MESSAGE(SceneSynchronizer::compare(buffer.read_variant(), value, DBL_EPSILON), "Should read the same value");
}
TEST_CASE("[Modules][DataBuffer] Seek") {
DataBuffer buffer;
buffer.begin_write(0);
buffer.add_bool(true);
buffer.add_bool(false);
buffer.begin_read();
ERR_PRINT_OFF
buffer.seek(-1);
CHECK_MESSAGE(buffer.get_bit_offset() == 0, "Bit offset should fail for negative values");
ERR_PRINT_ON
buffer.seek(1);
CHECK_MESSAGE(buffer.get_bit_offset() == 1, "Bit offset should be 1 after seek to 1");
CHECK_MESSAGE(buffer.read_bool() == false, "Should read false at position 1");
buffer.seek(0);
CHECK_MESSAGE(buffer.get_bit_offset() == 0, "Bit offset should be 0 after seek to 0");
CHECK_MESSAGE(buffer.read_bool() == true, "Should read true at position 0");
}
TEST_CASE("[Modules][DataBuffer] Metadata") {
bool value = {};
bool metadata = {};
SUBCASE("[Modules][DataBuffer] True") {
metadata = true;
value = false;
}
SUBCASE("[Modules][DataBuffer] False") {
metadata = false;
value = true;
}
const int metadata_size = DataBuffer::get_bit_taken(DataBuffer::DATA_TYPE_BOOL, DataBuffer::COMPRESSION_LEVEL_0);
DataBuffer buffer;
buffer.begin_write(metadata_size);
buffer.add_bool(metadata);
buffer.add_bool(value);
buffer.begin_read();
CHECK_MESSAGE(buffer.read_bool() == metadata, "Should return correct metadata");
CHECK_MESSAGE(buffer.read_bool() == value, "Should return correct value after metadata");
CHECK_MESSAGE(buffer.get_metadata_size() == metadata_size, "Metadata size should be equal to expected");
CHECK_MESSAGE(buffer.size() == DataBuffer::get_bit_taken(DataBuffer::DATA_TYPE_BOOL, DataBuffer::COMPRESSION_LEVEL_0), "Size should be equal to expected");
CHECK_MESSAGE(buffer.total_size() == DataBuffer::get_bit_taken(DataBuffer::DATA_TYPE_BOOL, DataBuffer::COMPRESSION_LEVEL_0) + metadata_size, "Total size should be equal to expected");
}
TEST_CASE("[Modules][DataBuffer] Zero") {
constexpr DataBuffer::CompressionLevel compression = DataBuffer::COMPRESSION_LEVEL_0;
DataBuffer buffer;
buffer.begin_write(0);
buffer.add_int(-1, compression);
buffer.zero();
buffer.begin_read();
CHECK_MESSAGE(buffer.read_int(compression) == 0, "Should return 0");
}
TEST_CASE("[Modules][DataBuffer] Shrinking") {
DataBuffer buffer;
buffer.begin_write(0);
for (int i = 0; i < 2; ++i) {
buffer.add_real(3.14, DataBuffer::COMPRESSION_LEVEL_0);
}
const int original_size = buffer.total_size();
ERR_PRINT_OFF;
buffer.shrink_to(0, original_size + 1);
ERR_PRINT_ON;
CHECK_MESSAGE(buffer.total_size() == original_size, "Shrinking to a larger size should fail.");
ERR_PRINT_OFF;
buffer.shrink_to(0, -1);
ERR_PRINT_ON;
CHECK_MESSAGE(buffer.total_size() == original_size, "Shrinking with a negative bits size should fail.");
buffer.shrink_to(0, original_size - 8);
CHECK_MESSAGE(buffer.total_size() == original_size - 8, "Shrinking by 1 byte should succeed.");
CHECK_MESSAGE(buffer.get_buffer().size_in_bits() == original_size, "Buffer size after shrinking by 1 byte should be the same.");
buffer.dry();
CHECK_MESSAGE(buffer.get_buffer().size_in_bits() == original_size - 8, "Buffer size after dry should changed to the smallest posiible.");
}
TEST_CASE("[Modules][DataBuffer] Skip") {
const bool value = true;
DataBuffer buffer;
buffer.add_bool(!value);
buffer.add_bool(value);
buffer.begin_read();
buffer.seek(DataBuffer::get_bit_taken(DataBuffer::DATA_TYPE_BOOL, DataBuffer::COMPRESSION_LEVEL_0));
CHECK_MESSAGE(buffer.read_bool() == value, "Should read the same value");
}
} // namespace TestDataBuffer
#endif // TEST_DATA_BUFFER_H

View File

@ -1,132 +0,0 @@
/*************************************************************************/
/* test_interpolator.h */
/*************************************************************************/
/* This file is part of: */
/* PANDEMONIUM ENGINE */
/* https://github.com/Relintai/pandemonium_engine */
/*************************************************************************/
/* Copyright (c) 2022-present Péter Magyar. */
/* Copyright (c) 2014-2022 Godot Engine contributors (cf. AUTHORS.md). */
/* Copyright (c) 2007-2022 Juan Linietsky, Ariel Manzur. */
/* */
/* 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. */
/*************************************************************************/
#ifndef TEST_INTERPOLATOR_H
#define TEST_INTERPOLATOR_H
#include "modules/network_synchronizer/interpolator.h"
#include "tests/test_macros.h"
namespace TestInterpolator {
template <class T>
T generate_value(real_t value) {
if constexpr (std::is_same_v<T, Vector2> || std::is_same_v<T, Vector2i>) {
return T(value, value);
} else if constexpr (std::is_same_v<T, Vector3> || std::is_same_v<T, Vector3i>) {
return T(value, value, value);
} else {
return static_cast<T>(value);
}
}
// TODO: Add other types
TEST_CASE_TEMPLATE("[Modules][Interpolator] Interpolation", T, int, real_t, Vector2, Vector2i, Vector3) {
LocalVector<real_t> fractions;
fractions.reserve(7);
fractions.push_back(0.0);
fractions.push_back(1.0);
fractions.push_back(0.5);
fractions.push_back(0.001);
fractions.push_back(0.999);
fractions.push_back(0.25);
fractions.push_back(0.75);
RBMap<real_t, real_t> values;
values.insert(0.0, 1.0);
values.insert(-1.0, 1.0);
values.insert(0.0, -1.0);
values.insert(10, 15);
Interpolator interpolator;
for (const RBMap<real_t, real_t>::Element *E = values.front(); E; E = E->next()) {
for (uint32_t j = 0; j < fractions.size(); ++j) {
// Skip custom interpolator for now
for (int k = Interpolator::FALLBACK_INTERPOLATE; k < Interpolator::FALLBACK_CUSTOM_INTERPOLATOR; ++k) {
const T first_value = generate_value<T>(E->key());
const T second_value = generate_value<T>(E->value());
interpolator.reset();
const int variable_id = interpolator.register_variable(T(), static_cast<Interpolator::Fallback>(k));
interpolator.terminate_init();
interpolator.begin_write(0);
interpolator.epoch_insert(variable_id, first_value);
interpolator.end_write();
interpolator.begin_write(1);
interpolator.epoch_insert(variable_id, second_value);
interpolator.end_write();
CAPTURE(k);
CAPTURE(fractions[j]);
CAPTURE(first_value);
CAPTURE(second_value);
const T result = interpolator.pop_epoch(0, fractions[j])[0];
switch (k) {
case Interpolator::FALLBACK_INTERPOLATE: {
CHECK(result == Interpolator::interpolate(first_value, second_value, fractions[j]).operator T());
} break;
case Interpolator::FALLBACK_DEFAULT: {
if (fractions[j] == 0.0) {
CHECK(result == first_value);
} else if (fractions[j] == 1.0) {
CHECK(result == second_value);
} else {
CHECK(result == T());
}
} break;
case Interpolator::FALLBACK_OLD_OR_NEAREST: {
if (fractions[j] == 0.0) {
CHECK(result == first_value);
} else if (fractions[j] == 1.0) {
CHECK(result == second_value);
} else {
CHECK(result == first_value);
}
} break;
case Interpolator::FALLBACK_NEW_OR_NEAREST: {
if (fractions[j] == 0.0) {
CHECK(result == first_value);
} else if (fractions[j] == 1.0) {
CHECK(result == second_value);
} else {
CHECK(result == second_value);
}
} break;
}
}
}
}
}
} // namespace TestInterpolator
#endif // TEST_INTERPOLATOR_H