#pragma once //////////////////////////////////////////////////////////////////////////////// // The MIT License (MIT) // // Copyright (c) 2017 Nicholas Frechette & Animation Compression Library contributors // // 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 "acl/core/iallocator.h" #include "acl/core/track_formats.h" #include "acl/core/impl/compiler_utils.h" #include "acl/core/error.h" #include "acl/compression/impl/clip_context.h" #include #include #include ACL_IMPL_FILE_PRAGMA_PUSH namespace acl { namespace acl_impl { inline bool is_rotation_track_constant(const RotationTrackStream& rotations, float threshold_angle) { // Calculating the average rotation and comparing every rotation in the track to it // to determine if we are within the threshold seems overkill. We can't use the min/max for the range // either because neither of those represents a valid rotation. Instead we grab // the first rotation, and compare everything else to it. auto sample_to_quat = [](const RotationTrackStream& track, uint32_t sample_index) { const rtm::vector4f rotation = track.get_raw_sample(sample_index); switch (track.get_rotation_format()) { case rotation_format8::quatf_full: return rtm::vector_to_quat(rotation); case rotation_format8::quatf_drop_w_full: case rotation_format8::quatf_drop_w_variable: return rtm::quat_from_positive_w(rotation); default: ACL_ASSERT(false, "Invalid or unsupported rotation format: %s", get_rotation_format_name(track.get_rotation_format())); return rtm::vector_to_quat(rotation); } }; const rtm::quatf ref_rotation = sample_to_quat(rotations, 0); const rtm::quatf inv_ref_rotation = rtm::quat_conjugate(ref_rotation); const uint32_t num_samples = rotations.get_num_samples(); for (uint32_t sample_index = 1; sample_index < num_samples; ++sample_index) { const rtm::quatf rotation = sample_to_quat(rotations, sample_index); const rtm::quatf delta = rtm::quat_normalize(rtm::quat_mul(inv_ref_rotation, rotation)); if (!rtm::quat_near_identity(delta, threshold_angle)) return false; } return true; } inline void compact_constant_streams(iallocator& allocator, clip_context& context, const track_array_qvvf& track_list, const compression_settings& settings) { ACL_ASSERT(context.num_segments == 1, "context must contain a single segment!"); SegmentContext& segment = context.segments[0]; const uint32_t num_bones = context.num_bones; const rtm::vector4f default_scale = get_default_scale(context.additive_format); uint32_t num_default_bone_scales = 0; // When a stream is constant, we only keep the first sample for (uint32_t bone_index = 0; bone_index < num_bones; ++bone_index) { const track_desc_transformf& desc = track_list[bone_index].get_description(); BoneStreams& bone_stream = segment.bone_streams[bone_index]; BoneRanges& bone_range = context.ranges[bone_index]; // We expect all our samples to have the same width of sizeof(rtm::vector4f) ACL_ASSERT(bone_stream.rotations.get_sample_size() == sizeof(rtm::vector4f), "Unexpected rotation sample size. %u != %zu", bone_stream.rotations.get_sample_size(), sizeof(rtm::vector4f)); ACL_ASSERT(bone_stream.translations.get_sample_size() == sizeof(rtm::vector4f), "Unexpected translation sample size. %u != %zu", bone_stream.translations.get_sample_size(), sizeof(rtm::vector4f)); ACL_ASSERT(bone_stream.scales.get_sample_size() == sizeof(rtm::vector4f), "Unexpected scale sample size. %u != %zu", bone_stream.scales.get_sample_size(), sizeof(rtm::vector4f)); // If we request raw data, use a 0.0 threshold for safety const float constant_rotation_threshold_angle = settings.rotation_format != rotation_format8::quatf_full ? desc.constant_rotation_threshold_angle : 0.0F; const float constant_translation_threshold = settings.translation_format != vector_format8::vector3f_full ? desc.constant_translation_threshold : 0.0F; const float constant_scale_threshold = settings.scale_format != vector_format8::vector3f_full ? desc.constant_scale_threshold : 0.0F; if (is_rotation_track_constant(bone_stream.rotations, constant_rotation_threshold_angle)) { RotationTrackStream constant_stream(allocator, 1, bone_stream.rotations.get_sample_size(), bone_stream.rotations.get_sample_rate(), bone_stream.rotations.get_rotation_format()); rtm::vector4f rotation = bone_stream.rotations.get_raw_sample(0); constant_stream.set_raw_sample(0, rotation); bone_stream.rotations = std::move(constant_stream); bone_stream.is_rotation_constant = true; bone_stream.is_rotation_default = rtm::quat_near_identity(rtm::vector_to_quat(rotation), constant_rotation_threshold_angle); bone_range.rotation = TrackStreamRange::from_min_extent(rotation, rtm::vector_zero()); } if (bone_range.translation.is_constant(constant_translation_threshold)) { TranslationTrackStream constant_stream(allocator, 1, bone_stream.translations.get_sample_size(), bone_stream.translations.get_sample_rate(), bone_stream.translations.get_vector_format()); rtm::vector4f translation = bone_stream.translations.get_raw_sample(0); constant_stream.set_raw_sample(0, translation); bone_stream.translations = std::move(constant_stream); bone_stream.is_translation_constant = true; bone_stream.is_translation_default = rtm::vector_all_near_equal3(translation, rtm::vector_zero(), constant_translation_threshold); bone_range.translation = TrackStreamRange::from_min_extent(translation, rtm::vector_zero()); } if (bone_range.scale.is_constant(constant_scale_threshold)) { ScaleTrackStream constant_stream(allocator, 1, bone_stream.scales.get_sample_size(), bone_stream.scales.get_sample_rate(), bone_stream.scales.get_vector_format()); rtm::vector4f scale = bone_stream.scales.get_raw_sample(0); constant_stream.set_raw_sample(0, scale); bone_stream.scales = std::move(constant_stream); bone_stream.is_scale_constant = true; bone_stream.is_scale_default = rtm::vector_all_near_equal3(scale, default_scale, constant_scale_threshold); bone_range.scale = TrackStreamRange::from_min_extent(scale, rtm::vector_zero()); num_default_bone_scales += bone_stream.is_scale_default ? 1 : 0; } } context.has_scale = num_default_bone_scales != num_bones; } } } ACL_IMPL_FILE_PRAGMA_POP