// // Created by martin on 25.03.22. // #ifndef ANIMTESTBED_ANIMGRAPHDATA_H #define ANIMTESTBED_ANIMGRAPHDATA_H #include #include #include #include #include #include #include #include "SyncTrack.h" #include "ozz/animation/runtime/animation.h" #include "ozz/animation/runtime/skeleton.h" #include "ozz/base/containers/vector.h" // // Data types // struct AnimGraph; struct AnimNode; struct AnimData { ozz::vector m_local_matrices; }; struct AnimDataRef { AnimData* ptr = nullptr; }; struct AnimDataAllocator { struct AnimDataList { AnimData* m_anim_data = nullptr; AnimDataList* next = nullptr; }; std::list m_anim_data_list; size_t m_num_allocations = 0; ~AnimDataAllocator() { while (!m_anim_data_list.empty()) { AnimData* front = m_anim_data_list.front(); #ifdef ANIM_DATA_ALLOCATOR_DEBUG std::cout << "about to delete with size " << front->m_anim_data->m_local_matrices.size() << front->m_anim_data << std::endl; #endif delete front; m_anim_data_list.pop_front(); } } AnimData* allocate(ozz::animation::Skeleton* skeleton) { if (m_anim_data_list.empty()) { AnimData* result = new AnimData(); result->m_local_matrices.resize(skeleton->num_soa_joints()); #ifdef ANIM_DATA_ALLOCATOR_DEBUG std::cout << "Allocated with size " << result->m_local_matrices.size() << " " << result << std::endl; #endif m_num_allocations++; return result; } AnimData* result = m_anim_data_list.front(); m_anim_data_list.pop_front(); #ifdef ANIM_DATA_ALLOCATOR_DEBUG std::cout << "Reusing buffer with size " << result->m_local_matrices.size() << " " << result << std::endl; #endif return result; } void free(AnimData* anim_data) { #ifdef ANIM_DATA_ALLOCATOR_DEBUG std::cout << "Storing buffer with size " << anim_data->m_local_matrices.size() << " " << anim_data << std::endl; #endif m_anim_data_list.push_front(anim_data); } size_t size() { return m_anim_data_list.size(); } }; struct AnimGraphContext { AnimGraph* m_graph = nullptr; ozz::animation::Skeleton* m_skeleton = nullptr; typedef std::map AnimationFileMap; AnimationFileMap m_animation_map; void freeAnimations() { AnimationFileMap::iterator animation_map_iter = m_animation_map.begin(); while (animation_map_iter != m_animation_map.end()) { delete animation_map_iter->second; animation_map_iter++; } } }; union Vec3 { struct { float x; float y; float z; }; float v[3] = {0}; }; union Quat { struct { float x; float y; float z; float w; }; float v[4] = {0}; }; enum class SocketType { SocketTypeUndefined = 0, SocketTypeBool, SocketTypeAnimation, SocketTypeInt, SocketTypeFloat, SocketTypeVec3, SocketTypeQuat, SocketTypeString, SocketTypeLast }; constexpr size_t cSocketStringValueMaxLength = 256; static const char* SocketTypeNames[] = {"", "Bool", "Animation", "Int", "Float", "Vec3", "Quat", "String"}; enum SocketFlags { SocketFlagNone = 0, SocketFlagAffectsTime = 1 }; struct Socket { std::string m_name; SocketType m_type = SocketType::SocketTypeUndefined; union SocketValue { bool flag; int int_value; float float_value; Vec3 vec3; Quat quat; }; SocketValue m_value = {0}; std::string m_value_string; union SocketReference { void* ptr; void** ptr_ptr; }; SocketReference m_reference = {0}; SocketFlags m_flags = SocketFlagNone; size_t m_type_size = 0; template void SetValue(const T value) { if constexpr (std::is_same::value) { m_value.flag = value; } if constexpr (std::is_same::value) { m_value.int_value = value; } if constexpr (std::is_same::value) { m_value.float_value = value; } if constexpr (std::is_same::value) { m_value.vec3 = value; } if constexpr (std::is_same::value) { m_value.quat = value; } if constexpr (std::is_same::value) { m_value_string = value; } } template T GetValue() const { if constexpr (std::is_same::value) { return m_value.flag; } if constexpr (std::is_same::value) { return m_value.int_value; } if constexpr (std::is_same::value) { return m_value.float_value; } if constexpr (std::is_same::value) { return m_value.vec3; } if constexpr (std::is_same::value) { return m_value.quat; } if constexpr (std::is_same::value) { return m_value_string; } return T(); } }; template SocketType GetSocketType() { if constexpr (std::is_same::value) { return SocketType::SocketTypeBool; } if constexpr (std::is_same::value) { return SocketType::SocketTypeAnimation; } if constexpr (std::is_same::value) { return SocketType::SocketTypeInt; } if constexpr (std::is_same::value) { return SocketType::SocketTypeFloat; } if constexpr (std::is_same::value) { return SocketType::SocketTypeVec3; } if constexpr (std::is_same::value) { return SocketType::SocketTypeQuat; } if constexpr (std::is_same::value) { return SocketType::SocketTypeString; } assert(false && "This should not be reachable"); abort(); return SocketType::SocketTypeUndefined; } struct AnimGraphConnection { AnimNode* m_source_node = nullptr; Socket m_source_socket; AnimNode* m_target_node = nullptr; Socket m_target_socket; bool m_crosses_hierarchy = false; }; struct NodeDescriptorBase { std::vector m_inputs; std::vector m_outputs; std::vector m_properties; template bool RegisterInput( const char* name, T** value_ptr_ptr, SocketFlags flags = SocketFlags::SocketFlagNone) { return RegisterSocket(name, value_ptr_ptr, m_inputs, flags); } template bool RegisterOutput( const char* name, T** value_ptr_ptr, SocketFlags flags = SocketFlags::SocketFlagNone) { return RegisterSocket(name, value_ptr_ptr, m_outputs, flags); } template bool RegisterProperty( const char* name, T* value_ptr, SocketFlags flags = SocketFlags::SocketFlagNone) { for (int i = 0; i < m_properties.size(); i++) { if (m_properties[i].m_name == name) { return false; } } Socket socket; socket.m_name = name; socket.m_type = GetSocketType(); socket.m_reference.ptr = static_cast(value_ptr); socket.m_flags = flags; socket.m_type_size = sizeof(T); m_properties.push_back(socket); return true; } template T* GetInput(const char* name) { Socket* socket = FindSocket(name, m_inputs); assert(GetSocketType() == socket->m_type); return *socket->m_reference.ptr_ptr; } template T GetInputValue(const char* name) const { const Socket* socket = FindSocket(name, m_inputs); assert(GetSocketType() == socket->m_type); return socket->GetValue(); } template void SetInput(const char* name, T* value_ptr) { Socket* socket = FindSocket(name, m_inputs); assert(GetSocketType() == socket->m_type); *socket->m_reference.ptr_ptr = value_ptr; } template void SetInputValue(const char* name, T value) { Socket* socket = FindSocket(name, m_inputs); assert(GetSocketType() == socket->m_type); socket->SetValue(value); } void SetInputUnchecked(const char* name, void* value_ptr) { Socket* socket = FindSocket(name, m_inputs); *socket->m_reference.ptr_ptr = value_ptr; } Socket* GetInputSocket(const char* name) { return FindSocket(name, m_inputs); } int GetInputIndex(const char* name) { return FindSocketIndex(name, m_inputs); } template void SetOutput(const char* name, T* value_ptr) { Socket* socket = FindSocket(name, m_outputs); assert(GetSocketType() == socket->m_type); *socket->m_reference.ptr_ptr = value_ptr; } void SetOutputUnchecked(const char* name, void* value_ptr) { Socket* socket = FindSocket(name, m_outputs); *socket->m_reference.ptr_ptr = value_ptr; } Socket* GetOutputSocket(const char* name) { return FindSocket(name, m_outputs); } int GetOutputIndex(const char* name) { return FindSocketIndex(name, m_outputs); } /** Sets value of an AnimNode Socket. * * @note Should only be used when the NodeDescriptor is associated with an AnimNode instance. * * @tparam T can be any AnimGraph data type. * @param Socket name * @param value */ template void SetProperty(const char* name, const T& value) { Socket* socket = FindSocket(name, m_properties); assert(GetSocketType() == socket->m_type); *static_cast(socket->m_reference.ptr) = value; } /** Sets value of an AnimNodeResource Socket. * * @note Should only be used when the NodeDescriptor is associated with an AnimNodeResource instance. For AnimNode instances use Socket::SetProperty(). * * @tparam T can be any AnimGraph data type. * @param Socket name * @param value */ template void SetPropertyValue(const char* name, const T& value) { Socket* socket = FindSocket(name, m_properties); assert(GetSocketType() == socket->m_type); socket->SetValue(value); } template const T& GetProperty(const char* name) { Socket* socket = FindSocket(name, m_properties); assert(GetSocketType() == socket->m_type); return *static_cast(socket->m_reference.ptr); } template T GetPropertyValue(const char* name) { Socket* socket = FindSocket(name, m_properties); assert(GetSocketType() == socket->m_type); return socket->GetValue(); } virtual void UpdateFlags(){}; protected: Socket* FindSocket(const char* name, std::vector& sockets) { for (int i = 0, n = sockets.size(); i < n; i++) { if (sockets[i].m_name == name) { return &sockets[i]; } } return nullptr; } const Socket* FindSocket(const char* name, const std::vector& sockets) const { for (int i = 0, n = sockets.size(); i < n; i++) { if (sockets[i].m_name == name) { return &sockets[i]; } } return nullptr; } int FindSocketIndex(const char* name, std::vector& sockets) { for (int i = 0, n = sockets.size(); i < n; i++) { if (sockets[i].m_name == name) { return i; } } return -1; } template bool RegisterSocket( const char* name, T** value_ptr_ptr, std::vector& sockets, SocketFlags flags) { for (int i = 0; i < sockets.size(); i++) { if (sockets[i].m_name == name) { return false; } } Socket socket; socket.m_name = name; socket.m_type = GetSocketType(); socket.m_reference.ptr_ptr = (void**)(value_ptr_ptr); socket.m_type_size = sizeof(T); socket.m_flags = flags; sockets.push_back(socket); return true; } }; template struct NodeDescriptor : public NodeDescriptorBase { virtual ~NodeDescriptor() {} }; struct AnimNode; template NodeDescriptorBase* CreateNodeDescriptor(AnimNode* node) { return new NodeDescriptor(dynamic_cast(node)); } #endif //ANIMTESTBED_ANIMGRAPHDATA_H