From 8c4a9ad626d16e8a1c98b4bf7b98439a9fce77fe Mon Sep 17 00:00:00 2001 From: PhairZ Date: Wed, 13 Nov 2024 19:51:38 +0200 Subject: [PATCH 01/33] fix fade_beats defined as int in audio_stream_interactive.h --- modules/interactive_music/audio_stream_interactive.h | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/modules/interactive_music/audio_stream_interactive.h b/modules/interactive_music/audio_stream_interactive.h index 12d3ce8aad86..9710a7fce11e 100644 --- a/modules/interactive_music/audio_stream_interactive.h +++ b/modules/interactive_music/audio_stream_interactive.h @@ -100,7 +100,7 @@ class AudioStreamInteractive : public AudioStream { TransitionFromTime from_time = TRANSITION_FROM_TIME_NEXT_BEAT; TransitionToTime to_time = TRANSITION_TO_TIME_START; FadeMode fade_mode = FADE_AUTOMATIC; - int fade_beats = 1; + float fade_beats = 1; bool use_filler_clip = false; int filler_clip = 0; bool hold_previous = false; From 0a4dd371b7a2b8d6a832e7207906b88e9649fc08 Mon Sep 17 00:00:00 2001 From: colinator27 <17358554+colinator27@users.noreply.github.com> Date: Sat, 16 Nov 2024 11:41:35 -0500 Subject: [PATCH 02/33] Implement get_bar_beats() for AudioStreamSynchronized, fix division by zero --- .../interactive_music/audio_stream_interactive.cpp | 11 ++++++++--- .../interactive_music/audio_stream_synchronized.cpp | 12 ++++++++++++ .../interactive_music/audio_stream_synchronized.h | 1 + 3 files changed, 21 insertions(+), 3 deletions(-) diff --git a/modules/interactive_music/audio_stream_interactive.cpp b/modules/interactive_music/audio_stream_interactive.cpp index 8656be988d98..995fa15015d2 100644 --- a/modules/interactive_music/audio_stream_interactive.cpp +++ b/modules/interactive_music/audio_stream_interactive.cpp @@ -691,9 +691,14 @@ void AudioStreamPlaybackInteractive::_queue(int p_to_clip_index, bool p_is_auto_ src_fade_wait = beat_sec - remainder; } break; case AudioStreamInteractive::TRANSITION_FROM_TIME_NEXT_BAR: { - float bar_sec = beat_sec * from_state.stream->get_bar_beats(); - float remainder = Math::fmod(current_pos, bar_sec); - src_fade_wait = bar_sec - remainder; + if (from_state.stream->get_bar_beats() > 0) { + float bar_sec = beat_sec * from_state.stream->get_bar_beats(); + float remainder = Math::fmod(current_pos, bar_sec); + src_fade_wait = bar_sec - remainder; + } else { + // Stream does not have a number of beats per bar - avoid NaN, and play immediately. + src_fade_wait = 0; + } } break; case AudioStreamInteractive::TRANSITION_FROM_TIME_END: { float end = from_state.stream->get_beat_count() > 0 ? float(from_state.stream->get_beat_count() * beat_sec) : from_state.stream->get_length(); diff --git a/modules/interactive_music/audio_stream_synchronized.cpp b/modules/interactive_music/audio_stream_synchronized.cpp index e38a57ba753a..ad67e778eec2 100644 --- a/modules/interactive_music/audio_stream_synchronized.cpp +++ b/modules/interactive_music/audio_stream_synchronized.cpp @@ -99,6 +99,18 @@ int AudioStreamSynchronized::get_beat_count() const { return max_beats; } +int AudioStreamSynchronized::get_bar_beats() const { + for (int i = 0; i < stream_count; i++) { + if (audio_streams[i].is_valid()) { + int bar_beats = audio_streams[i]->get_bar_beats(); + if (bar_beats != 0) { + return bar_beats; + } + } + } + return 0; +} + bool AudioStreamSynchronized::has_loop() const { for (int i = 0; i < stream_count; i++) { if (audio_streams[i].is_valid()) { diff --git a/modules/interactive_music/audio_stream_synchronized.h b/modules/interactive_music/audio_stream_synchronized.h index a2d8c5540443..83798a13bd5b 100644 --- a/modules/interactive_music/audio_stream_synchronized.h +++ b/modules/interactive_music/audio_stream_synchronized.h @@ -54,6 +54,7 @@ class AudioStreamSynchronized : public AudioStream { public: virtual double get_bpm() const override; virtual int get_beat_count() const override; + virtual int get_bar_beats() const override; virtual bool has_loop() const override; void set_stream_count(int p_count); int get_stream_count() const; From 1bffefb346c2974aad692905041ccfb84e666597 Mon Sep 17 00:00:00 2001 From: Bastiaan Olij Date: Fri, 22 Sep 2023 13:58:02 +1000 Subject: [PATCH 03/33] Adding ability to include build-in include files (precursor to custom shader templates) --- doc/classes/ShaderIncludeDB.xml | 33 +++++ servers/register_server_types.cpp | 2 + .../renderer_rd/renderer_scene_render_rd.cpp | 11 ++ servers/rendering/renderer_rd/shader_rd.cpp | 33 ++++- servers/rendering/renderer_rd/shaders/SCsub | 10 +- servers/rendering/rendering_device.cpp | 7 +- servers/rendering/rendering_device_binds.cpp | 16 ++- servers/rendering/shader_include_db.cpp | 115 ++++++++++++++++++ servers/rendering/shader_include_db.h | 53 ++++++++ 9 files changed, 271 insertions(+), 9 deletions(-) create mode 100644 doc/classes/ShaderIncludeDB.xml create mode 100644 servers/rendering/shader_include_db.cpp create mode 100644 servers/rendering/shader_include_db.h diff --git a/doc/classes/ShaderIncludeDB.xml b/doc/classes/ShaderIncludeDB.xml new file mode 100644 index 000000000000..a431eabf4e85 --- /dev/null +++ b/doc/classes/ShaderIncludeDB.xml @@ -0,0 +1,33 @@ + + + + Internal database of built in shader include files. + + + This object contains shader fragments from Godot's internal shaders. These can be used when access to internal uniform buffers and/or internal functions is required for instance when composing compositor effects or compute shaders. Only fragments for the current rendering device are loaded. + + + + + + + + + Returns the code for the built-in shader fragment. You can also access this in your shader code through [code]#include "filename"[/code]. + + + + + + + Returns [code]true[/code] if an include file with this name exists. + + + + + + Returns a list of built-in include files that are currently registered. + + + + diff --git a/servers/register_server_types.cpp b/servers/register_server_types.cpp index 18ee8630838d..3e0caeea6594 100644 --- a/servers/register_server_types.cpp +++ b/servers/register_server_types.cpp @@ -70,6 +70,7 @@ #include "rendering/renderer_rd/uniform_set_cache_rd.h" #include "rendering/rendering_device.h" #include "rendering/rendering_device_binds.h" +#include "rendering/shader_include_db.h" #include "rendering/storage/render_data.h" #include "rendering/storage/render_scene_buffers.h" #include "rendering/storage/render_scene_data.h" @@ -210,6 +211,7 @@ void register_server_types() { } GDREGISTER_ABSTRACT_CLASS(RenderingDevice); + GDREGISTER_CLASS(ShaderIncludeDB); GDREGISTER_CLASS(RDTextureFormat); GDREGISTER_CLASS(RDTextureView); GDREGISTER_CLASS(RDAttachmentFormat); diff --git a/servers/rendering/renderer_rd/renderer_scene_render_rd.cpp b/servers/rendering/renderer_rd/renderer_scene_render_rd.cpp index 4417f6832cd8..0a7dd3eadcb1 100644 --- a/servers/rendering/renderer_rd/renderer_scene_render_rd.cpp +++ b/servers/rendering/renderer_rd/renderer_scene_render_rd.cpp @@ -35,9 +35,13 @@ #include "core/os/os.h" #include "renderer_compositor_rd.h" #include "servers/rendering/renderer_rd/environment/fog.h" +#include "servers/rendering/renderer_rd/shaders/decal_data_inc.glsl.gen.h" +#include "servers/rendering/renderer_rd/shaders/light_data_inc.glsl.gen.h" +#include "servers/rendering/renderer_rd/shaders/scene_data_inc.glsl.gen.h" #include "servers/rendering/renderer_rd/storage_rd/material_storage.h" #include "servers/rendering/renderer_rd/storage_rd/texture_storage.h" #include "servers/rendering/rendering_server_default.h" +#include "servers/rendering/shader_include_db.h" #include "servers/rendering/storage/camera_attributes_storage.h" void get_vogel_disk(float *r_kernel, int p_sample_count) { @@ -1452,6 +1456,13 @@ void RendererSceneRenderRD::init() { /* Forward ID */ forward_id_storage = create_forward_id_storage(); + /* Register the include files we make available by default to our users */ + { + ShaderIncludeDB::register_built_in_include_file("godot/decal_data_inc.glsl", decal_data_inc_shader_glsl); + ShaderIncludeDB::register_built_in_include_file("godot/light_data_inc.glsl", light_data_inc_shader_glsl); + ShaderIncludeDB::register_built_in_include_file("godot/scene_data_inc.glsl", scene_data_inc_shader_glsl); + } + /* SKY SHADER */ sky.init(); diff --git a/servers/rendering/renderer_rd/shader_rd.cpp b/servers/rendering/renderer_rd/shader_rd.cpp index 6234cddee393..fb89c11a1b40 100644 --- a/servers/rendering/renderer_rd/shader_rd.cpp +++ b/servers/rendering/renderer_rd/shader_rd.cpp @@ -37,6 +37,7 @@ #include "core/version.h" #include "renderer_compositor_rd.h" #include "servers/rendering/rendering_device.h" +#include "servers/rendering/shader_include_db.h" #include "thirdparty/misc/smolv.h" #define ENABLE_SHADER_CACHE 1 @@ -46,7 +47,8 @@ void ShaderRD::_add_stage(const char *p_code, StageType p_stage_type) { String text; - for (int i = 0; i < lines.size(); i++) { + int line_count = lines.size(); + for (int i = 0; i < line_count; i++) { const String &l = lines[i]; bool push_chunk = false; @@ -78,6 +80,35 @@ void ShaderRD::_add_stage(const char *p_code, StageType p_stage_type) { chunk.type = StageTemplate::Chunk::TYPE_CODE; push_chunk = true; chunk.code = l.replace_first("#CODE", String()).replace(":", "").strip_edges().to_upper(); + } else if (l.begins_with("#include ")) { + String include_file = l.replace("#include ", "").strip_edges(); + if (include_file[0] == '"') { + int end_pos = include_file.find_char('"', 1); + if (end_pos >= 0) { + include_file = include_file.substr(1, end_pos - 1); + + String include_code = ShaderIncludeDB::get_built_in_include_file(include_file); + if (!include_code.is_empty()) { + // Add these lines into our parse list so we parse them as well. + Vector include_lines = include_code.split("\n"); + + for (int j = include_lines.size() - 1; j >= 0; j--) { + lines.insert(i + 1, include_lines[j]); + } + + line_count = lines.size(); + } else { + // Add it in as is. + text += l + "\n"; + } + } else { + // Add it in as is. + text += l + "\n"; + } + } else { + // Add it in as is. + text += l + "\n"; + } } else { text += l + "\n"; } diff --git a/servers/rendering/renderer_rd/shaders/SCsub b/servers/rendering/renderer_rd/shaders/SCsub index e102b839b59d..f1b6710383dc 100644 --- a/servers/rendering/renderer_rd/shaders/SCsub +++ b/servers/rendering/renderer_rd/shaders/SCsub @@ -4,16 +4,20 @@ from misc.utility.scons_hints import * Import("env") if "RD_GLSL" in env["BUILDERS"]: - # find all include files + # find just the include files gl_include_files = [str(f) for f in Glob("*_inc.glsl")] - # find all shader code(all glsl files excluding our include files) + # find all shader code (all glsl files excluding our include files) glsl_files = [str(f) for f in Glob("*.glsl") if str(f) not in gl_include_files] # make sure we recompile shaders if include files change env.Depends([f + ".gen.h" for f in glsl_files], gl_include_files + ["#glsl_builders.py"]) - # compile shaders + # compile include files + for glsl_file in gl_include_files: + env.GLSL_HEADER(glsl_file) + + # compile RD shader for glsl_file in glsl_files: env.RD_GLSL(glsl_file) diff --git a/servers/rendering/rendering_device.cpp b/servers/rendering/rendering_device.cpp index e9fa38475e22..6cf2f6db7e41 100644 --- a/servers/rendering/rendering_device.cpp +++ b/servers/rendering/rendering_device.cpp @@ -32,6 +32,7 @@ #include "rendering_device.compat.inc" #include "rendering_device_binds.h" +#include "shader_include_db.h" #include "core/config/project_settings.h" #include "core/io/dir_access.h" @@ -189,6 +190,10 @@ void RenderingDevice::_free_dependencies(RID p_id) { } } +/*******************************/ +/**** SHADER INFRASTRUCTURE ****/ +/*******************************/ + void RenderingDevice::shader_set_compile_to_spirv_function(ShaderCompileToSPIRVFunction p_function) { compile_to_spirv_function = p_function; } @@ -211,7 +216,7 @@ Vector RenderingDevice::shader_compile_spirv_from_source(ShaderStage p_ ERR_FAIL_NULL_V(compile_to_spirv_function, Vector()); - return compile_to_spirv_function(p_stage, p_source_code, p_language, r_error, this); + return compile_to_spirv_function(p_stage, ShaderIncludeDB::parse_include_files(p_source_code), p_language, r_error, this); } String RenderingDevice::shader_get_spirv_cache_key() const { diff --git a/servers/rendering/rendering_device_binds.cpp b/servers/rendering/rendering_device_binds.cpp index e41a56b0a32e..9ccf1e41083d 100644 --- a/servers/rendering/rendering_device_binds.cpp +++ b/servers/rendering/rendering_device_binds.cpp @@ -30,6 +30,8 @@ #include "rendering_device_binds.h" +#include "shader_include_db.h" + Error RDShaderFile::parse_versions_from_text(const String &p_text, const String p_defines, OpenIncludeFunction p_include_func, void *p_include_func_userdata) { ERR_FAIL_NULL_V_MSG( RenderingDevice::get_singleton(), @@ -144,11 +146,17 @@ Error RDShaderFile::parse_versions_from_text(const String &p_text, const String break; } include = include.substr(1, include.length() - 2).strip_edges(); - String include_text = p_include_func(include, p_include_func_userdata); - if (!include_text.is_empty()) { - stage_code[stage] += "\n" + include_text + "\n"; + + String include_code = ShaderIncludeDB::get_built_in_include_file(include); + if (!include_code.is_empty()) { + stage_code[stage] += "\n" + include_code + "\n"; } else { - base_error = "#include failed for file '" + include + "'"; + String include_text = p_include_func(include, p_include_func_userdata); + if (!include_text.is_empty()) { + stage_code[stage] += "\n" + include_text + "\n"; + } else { + base_error = "#include failed for file '" + include + "'."; + } } } else { base_error = "#include used, but no include function provided."; diff --git a/servers/rendering/shader_include_db.cpp b/servers/rendering/shader_include_db.cpp new file mode 100644 index 000000000000..bc62c1fe0295 --- /dev/null +++ b/servers/rendering/shader_include_db.cpp @@ -0,0 +1,115 @@ +/**************************************************************************/ +/* shader_include_db.cpp */ +/**************************************************************************/ +/* This file is part of: */ +/* GODOT ENGINE */ +/* https://godotengine.org */ +/**************************************************************************/ +/* Copyright (c) 2014-present Godot Engine contributors (see AUTHORS.md). */ +/* Copyright (c) 2007-2014 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. */ +/**************************************************************************/ + +#include "shader_include_db.h" + +HashMap ShaderIncludeDB::built_in_includes; + +void ShaderIncludeDB::_bind_methods() { + ClassDB::bind_static_method("ShaderIncludeDB", D_METHOD("list_built_in_include_files"), &ShaderIncludeDB::list_built_in_include_files); + ClassDB::bind_static_method("ShaderIncludeDB", D_METHOD("has_built_in_include_file", "filename"), &ShaderIncludeDB::has_built_in_include_file); + ClassDB::bind_static_method("ShaderIncludeDB", D_METHOD("get_built_in_include_file", "filename"), &ShaderIncludeDB::get_built_in_include_file); +} + +void ShaderIncludeDB::register_built_in_include_file(const String &p_filename, const String &p_shader_code) { + built_in_includes[p_filename] = p_shader_code; +} + +PackedStringArray ShaderIncludeDB::list_built_in_include_files() { + PackedStringArray ret; + + for (const KeyValue &e : built_in_includes) { + ret.push_back(e.key); + } + + return ret; +} + +bool ShaderIncludeDB::has_built_in_include_file(const String &p_filename) { + return built_in_includes.has(p_filename); +} + +String ShaderIncludeDB::get_built_in_include_file(const String &p_filename) { + const String *ptr = built_in_includes.getptr(p_filename); + + return ptr ? *ptr : String(); +} + +String ShaderIncludeDB::parse_include_files(const String &p_code) { + // Prevent needless processing if we don't have any includes. + if (p_code.find("#include ") == -1) { + return p_code; + } + + const String include = "#include "; + String parsed_code; + + Vector lines = p_code.split("\n"); + int line_count = lines.size(); + for (int i = 0; i < line_count; i++) { + const String &l = lines[i]; + + if (l.begins_with(include)) { + String include_file = l.replace(include, "").strip_edges(); + if (include_file[0] == '"') { + int end_pos = include_file.find_char('"', 1); + if (end_pos >= 0) { + include_file = include_file.substr(1, end_pos - 1); + + String include_code = ShaderIncludeDB::get_built_in_include_file(include_file); + if (!include_code.is_empty()) { + // Add these lines into our parse list so we parse them as well. + Vector include_lines = include_code.split("\n"); + + for (int j = include_lines.size() - 1; j >= 0; j--) { + lines.insert(i + 1, include_lines[j]); + } + + line_count = lines.size(); + } else { + // Just add it back in, this will cause a compile error to alert the user. + parsed_code += l + "\n"; + } + } else { + // Include as is. + parsed_code += l + "\n"; + } + } else { + // Include as is. + parsed_code += l + "\n"; + } + } else { + // Include as is. + parsed_code += l + "\n"; + } + } + + return parsed_code; +} diff --git a/servers/rendering/shader_include_db.h b/servers/rendering/shader_include_db.h new file mode 100644 index 000000000000..d1967a3296dd --- /dev/null +++ b/servers/rendering/shader_include_db.h @@ -0,0 +1,53 @@ +/**************************************************************************/ +/* shader_include_db.h */ +/**************************************************************************/ +/* This file is part of: */ +/* GODOT ENGINE */ +/* https://godotengine.org */ +/**************************************************************************/ +/* Copyright (c) 2014-present Godot Engine contributors (see AUTHORS.md). */ +/* Copyright (c) 2007-2014 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 SHADER_INCLUDE_DB_H +#define SHADER_INCLUDE_DB_H + +#include "core/object/class_db.h" + +class ShaderIncludeDB : public Object { + GDCLASS(ShaderIncludeDB, Object) + +private: + static HashMap built_in_includes; + +protected: + static void _bind_methods(); + +public: + static void register_built_in_include_file(const String &p_filename, const String &p_shader_code); + static PackedStringArray list_built_in_include_files(); + static bool has_built_in_include_file(const String &p_filename); + static String get_built_in_include_file(const String &p_filename); + static String parse_include_files(const String &p_code); +}; + +#endif // SHADER_INCLUDE_DB_H From 0ce4c6dac3c78c2e159f22abc802299737fe7c89 Mon Sep 17 00:00:00 2001 From: Dario Date: Tue, 12 Nov 2024 14:33:32 -0300 Subject: [PATCH 04/33] Improve performance of shader lighting code in Forward renderers. - Skip sampling shadows if attenuation is very small. - Skip computing diffuse and specular light if attenuation and shadow are very small. --- .../scene_forward_clustered.glsl | 12 +- .../forward_mobile/scene_forward_mobile.glsl | 15 +- .../shaders/scene_forward_lights_inc.glsl | 453 ++++++++---------- 3 files changed, 211 insertions(+), 269 deletions(-) diff --git a/servers/rendering/renderer_rd/shaders/forward_clustered/scene_forward_clustered.glsl b/servers/rendering/renderer_rd/shaders/forward_clustered/scene_forward_clustered.glsl index 1e1b6d893710..52b3d5d1e38c 100644 --- a/servers/rendering/renderer_rd/shaders/forward_clustered/scene_forward_clustered.glsl +++ b/servers/rendering/renderer_rd/shaders/forward_clustered/scene_forward_clustered.glsl @@ -2369,11 +2369,7 @@ void fragment_shader(in SceneData scene_data) { continue; // Statically baked light and object uses lightmap, skip } - float shadow = light_process_omni_shadow(light_index, vertex, normal, scene_data.taa_frame_count); - - shadow = blur_shadow(shadow); - - light_process_omni(light_index, vertex, view, normal, vertex_ddx, vertex_ddy, f0, orms, shadow, albedo, alpha, screen_uv, + light_process_omni(light_index, vertex, view, normal, vertex_ddx, vertex_ddy, f0, orms, scene_data.taa_frame_count, albedo, alpha, screen_uv, #ifdef LIGHT_BACKLIGHT_USED backlight, #endif @@ -2441,11 +2437,7 @@ void fragment_shader(in SceneData scene_data) { continue; // Statically baked light and object uses lightmap, skip } - float shadow = light_process_spot_shadow(light_index, vertex, normal, scene_data.taa_frame_count); - - shadow = blur_shadow(shadow); - - light_process_spot(light_index, vertex, view, normal, vertex_ddx, vertex_ddy, f0, orms, shadow, albedo, alpha, screen_uv, + light_process_spot(light_index, vertex, view, normal, vertex_ddx, vertex_ddy, f0, orms, scene_data.taa_frame_count, albedo, alpha, screen_uv, #ifdef LIGHT_BACKLIGHT_USED backlight, #endif diff --git a/servers/rendering/renderer_rd/shaders/forward_mobile/scene_forward_mobile.glsl b/servers/rendering/renderer_rd/shaders/forward_mobile/scene_forward_mobile.glsl index 0cb34557ea89..c0ebf917ccdc 100644 --- a/servers/rendering/renderer_rd/shaders/forward_mobile/scene_forward_mobile.glsl +++ b/servers/rendering/renderer_rd/shaders/forward_mobile/scene_forward_mobile.glsl @@ -1622,13 +1622,7 @@ void main() { uvec2 omni_indices = instances.data[draw_call.instance_index].omni_lights; for (uint i = 0; i < sc_omni_lights(); i++) { uint light_index = (i > 3) ? ((omni_indices.y >> ((i - 4) * 8)) & 0xFF) : ((omni_indices.x >> (i * 8)) & 0xFF); - - float shadow = light_process_omni_shadow(light_index, vertex, normal, scene_data.taa_frame_count); - - shadow = blur_shadow(shadow); - - // Fragment lighting - light_process_omni(light_index, vertex, view, normal, vertex_ddx, vertex_ddy, f0, orms, shadow, albedo, alpha, screen_uv, + light_process_omni(light_index, vertex, view, normal, vertex_ddx, vertex_ddy, f0, orms, scene_data.taa_frame_count, albedo, alpha, screen_uv, #ifdef LIGHT_BACKLIGHT_USED backlight, #endif @@ -1656,12 +1650,7 @@ void main() { uvec2 spot_indices = instances.data[draw_call.instance_index].spot_lights; for (uint i = 0; i < sc_spot_lights(); i++) { uint light_index = (i > 3) ? ((spot_indices.y >> ((i - 4) * 8)) & 0xFF) : ((spot_indices.x >> (i * 8)) & 0xFF); - - float shadow = light_process_spot_shadow(light_index, vertex, normal, scene_data.taa_frame_count); - - shadow = blur_shadow(shadow); - - light_process_spot(light_index, vertex, view, normal, vertex_ddx, vertex_ddy, f0, orms, shadow, albedo, alpha, screen_uv, + light_process_spot(light_index, vertex, view, normal, vertex_ddx, vertex_ddy, f0, orms, scene_data.taa_frame_count, albedo, alpha, screen_uv, #ifdef LIGHT_BACKLIGHT_USED backlight, #endif diff --git a/servers/rendering/renderer_rd/shaders/scene_forward_lights_inc.glsl b/servers/rendering/renderer_rd/shaders/scene_forward_lights_inc.glsl index 1e8fc7eab482..f58304aa8d33 100644 --- a/servers/rendering/renderer_rd/shaders/scene_forward_lights_inc.glsl +++ b/servers/rendering/renderer_rd/shaders/scene_forward_lights_inc.glsl @@ -59,16 +59,14 @@ void light_compute(vec3 N, vec3 L, vec3 V, float A, vec3 light_color, bool is_di vec3 B, vec3 T, float anisotropy, #endif inout vec3 diffuse_light, inout vec3 specular_light) { - vec4 orms_unpacked = unpackUnorm4x8(orms); - float roughness = orms_unpacked.y; float metallic = orms_unpacked.z; #if defined(LIGHT_CODE_USED) - // light is written by the light shader - + // Light is written by the user shader. mat4 inv_view_matrix = scene_data_block.data.inv_view_matrix; + mat4 read_view_matrix = scene_data_block.data.view_matrix; #ifdef USING_MOBILE_RENDERER mat4 read_model_matrix = instances.data[draw_call.instance_index].transform; @@ -76,183 +74,159 @@ void light_compute(vec3 N, vec3 L, vec3 V, float A, vec3 light_color, bool is_di mat4 read_model_matrix = instances.data[instance_index_interp].transform; #endif - mat4 read_view_matrix = scene_data_block.data.view_matrix; - #undef projection_matrix #define projection_matrix scene_data_block.data.projection_matrix #undef inv_projection_matrix #define inv_projection_matrix scene_data_block.data.inv_projection_matrix vec2 read_viewport_size = scene_data_block.data.viewport_size; - vec3 normal = N; vec3 light = L; vec3 view = V; #CODE : LIGHT +#else // !LIGHT_CODE_USED + float NdotL = min(A + dot(N, L), 1.0); + float cNdotV = max(dot(N, V), 1e-4); +#ifdef LIGHT_TRANSMITTANCE_USED + { +#ifdef SSS_MODE_SKIN + float scale = 8.25 / transmittance_depth; + float d = scale * abs(transmittance_z); + float dd = -d * d; + vec3 profile = vec3(0.233, 0.455, 0.649) * exp(dd / 0.0064) + + vec3(0.1, 0.336, 0.344) * exp(dd / 0.0484) + + vec3(0.118, 0.198, 0.0) * exp(dd / 0.187) + + vec3(0.113, 0.007, 0.007) * exp(dd / 0.567) + + vec3(0.358, 0.004, 0.0) * exp(dd / 1.99) + + vec3(0.078, 0.0, 0.0) * exp(dd / 7.41); + + diffuse_light += profile * transmittance_color.a * light_color * clamp(transmittance_boost - NdotL, 0.0, 1.0) * (1.0 / M_PI); #else - float NdotL = min(A + dot(N, L), 1.0); - float cNdotL = max(NdotL, 0.0); // clamped NdotL - float NdotV = dot(N, V); - float cNdotV = max(NdotV, 1e-4); - -#if defined(DIFFUSE_BURLEY) || defined(SPECULAR_SCHLICK_GGX) || defined(LIGHT_CLEARCOAT_USED) - vec3 H = normalize(V + L); + float scale = 8.25 / transmittance_depth; + float d = scale * abs(transmittance_z); + float dd = -d * d; + diffuse_light += exp(dd) * transmittance_color.rgb * transmittance_color.a * light_color * clamp(transmittance_boost - NdotL, 0.0, 1.0) * (1.0 / M_PI); #endif + } +#endif //LIGHT_TRANSMITTANCE_USED -#if defined(SPECULAR_SCHLICK_GGX) - float cNdotH = clamp(A + dot(N, H), 0.0, 1.0); +#if defined(LIGHT_RIM_USED) + // Epsilon min to prevent pow(0, 0) singularity which results in undefined behavior. + float rim_light = pow(max(1e-4, 1.0 - cNdotV), max(0.0, (1.0 - roughness) * 16.0)); + diffuse_light += rim_light * rim * mix(vec3(1.0), albedo, rim_tint) * light_color; #endif + // We skip checking on attenuation on directional lights to avoid a branch that is not as beneficial for directional lights as the other ones. + const float EPSILON = 1e-3f; + if (is_directional || attenuation > EPSILON) { + float cNdotL = max(NdotL, 0.0); +#if defined(DIFFUSE_BURLEY) || defined(SPECULAR_SCHLICK_GGX) || defined(LIGHT_CLEARCOAT_USED) + vec3 H = normalize(V + L); +#endif #if defined(DIFFUSE_BURLEY) || defined(SPECULAR_SCHLICK_GGX) || defined(LIGHT_CLEARCOAT_USED) - float cLdotH = clamp(A + dot(L, H), 0.0, 1.0); + float cLdotH = clamp(A + dot(L, H), 0.0, 1.0); #endif +#if defined(LIGHT_CLEARCOAT_USED) + // Clearcoat ignores normal_map, use vertex normal instead + float ccNdotL = max(min(A + dot(vertex_normal, L), 1.0), 0.0); + float ccNdotH = clamp(A + dot(vertex_normal, H), 0.0, 1.0); + float ccNdotV = max(dot(vertex_normal, V), 1e-4); + +#if !defined(SPECULAR_SCHLICK_GGX) + float cLdotH5 = SchlickFresnel(cLdotH); +#endif + float Dr = D_GGX(ccNdotH, mix(0.001, 0.1, clearcoat_roughness)); + float Gr = 0.25 / (cLdotH * cLdotH); + float Fr = mix(.04, 1.0, cLdotH5); + float clearcoat_specular_brdf_NL = clearcoat * Gr * Fr * Dr * cNdotL; + + specular_light += clearcoat_specular_brdf_NL * light_color * attenuation * specular_amount; + + // TODO: Clearcoat adds light to the scene right now (it is non-energy conserving), both diffuse and specular need to be scaled by (1.0 - FR) + // but to do so we need to rearrange this entire function +#endif // LIGHT_CLEARCOAT_USED - if (metallic < 1.0) { - float diffuse_brdf_NL; // BRDF times N.L for calculating diffuse radiance + if (metallic < 1.0) { + float diffuse_brdf_NL; // BRDF times N.L for calculating diffuse radiance #if defined(DIFFUSE_LAMBERT_WRAP) - // Energy conserving lambert wrap shader. - // https://web.archive.org/web/20210228210901/http://blog.stevemcauley.com/2011/12/03/energy-conserving-wrapped-diffuse/ - diffuse_brdf_NL = max(0.0, (NdotL + roughness) / ((1.0 + roughness) * (1.0 + roughness))) * (1.0 / M_PI); + // Energy conserving lambert wrap shader. + // https://web.archive.org/web/20210228210901/http://blog.stevemcauley.com/2011/12/03/energy-conserving-wrapped-diffuse/ + diffuse_brdf_NL = max(0.0, (NdotL + roughness) / ((1.0 + roughness) * (1.0 + roughness))) * (1.0 / M_PI); #elif defined(DIFFUSE_TOON) - diffuse_brdf_NL = smoothstep(-roughness, max(roughness, 0.01), NdotL) * (1.0 / M_PI); + diffuse_brdf_NL = smoothstep(-roughness, max(roughness, 0.01), NdotL) * (1.0 / M_PI); #elif defined(DIFFUSE_BURLEY) - - { - float FD90_minus_1 = 2.0 * cLdotH * cLdotH * roughness - 0.5; - float FdV = 1.0 + FD90_minus_1 * SchlickFresnel(cNdotV); - float FdL = 1.0 + FD90_minus_1 * SchlickFresnel(cNdotL); - diffuse_brdf_NL = (1.0 / M_PI) * FdV * FdL * cNdotL; - /* - float energyBias = mix(roughness, 0.0, 0.5); - float energyFactor = mix(roughness, 1.0, 1.0 / 1.51); - float fd90 = energyBias + 2.0 * VoH * VoH * roughness; - float f0 = 1.0; - float lightScatter = f0 + (fd90 - f0) * pow(1.0 - cNdotL, 5.0); - float viewScatter = f0 + (fd90 - f0) * pow(1.0 - cNdotV, 5.0); - - diffuse_brdf_NL = lightScatter * viewScatter * energyFactor; - */ - } + { + float FD90_minus_1 = 2.0 * cLdotH * cLdotH * roughness - 0.5; + float FdV = 1.0 + FD90_minus_1 * SchlickFresnel(cNdotV); + float FdL = 1.0 + FD90_minus_1 * SchlickFresnel(cNdotL); + diffuse_brdf_NL = (1.0 / M_PI) * FdV * FdL * cNdotL; + } #else - // lambert - diffuse_brdf_NL = cNdotL * (1.0 / M_PI); + // lambert + diffuse_brdf_NL = cNdotL * (1.0 / M_PI); #endif - diffuse_light += light_color * diffuse_brdf_NL * attenuation; + diffuse_light += light_color * diffuse_brdf_NL * attenuation; #if defined(LIGHT_BACKLIGHT_USED) - diffuse_light += light_color * (vec3(1.0 / M_PI) - diffuse_brdf_NL) * backlight * attenuation; -#endif - -#if defined(LIGHT_RIM_USED) - // Epsilon min to prevent pow(0, 0) singularity which results in undefined behavior. - float rim_light = pow(max(1e-4, 1.0 - cNdotV), max(0.0, (1.0 - roughness) * 16.0)); - diffuse_light += rim_light * rim * mix(vec3(1.0), albedo, rim_tint) * light_color; -#endif - -#ifdef LIGHT_TRANSMITTANCE_USED - - { -#ifdef SSS_MODE_SKIN - float scale = 8.25 / transmittance_depth; - float d = scale * abs(transmittance_z); - float dd = -d * d; - vec3 profile = vec3(0.233, 0.455, 0.649) * exp(dd / 0.0064) + - vec3(0.1, 0.336, 0.344) * exp(dd / 0.0484) + - vec3(0.118, 0.198, 0.0) * exp(dd / 0.187) + - vec3(0.113, 0.007, 0.007) * exp(dd / 0.567) + - vec3(0.358, 0.004, 0.0) * exp(dd / 1.99) + - vec3(0.078, 0.0, 0.0) * exp(dd / 7.41); - - diffuse_light += profile * transmittance_color.a * light_color * clamp(transmittance_boost - NdotL, 0.0, 1.0) * (1.0 / M_PI); -#else - - float scale = 8.25 / transmittance_depth; - float d = scale * abs(transmittance_z); - float dd = -d * d; - diffuse_light += exp(dd) * transmittance_color.rgb * transmittance_color.a * light_color * clamp(transmittance_boost - NdotL, 0.0, 1.0) * (1.0 / M_PI); + diffuse_light += light_color * (vec3(1.0 / M_PI) - diffuse_brdf_NL) * backlight * attenuation; #endif } -#else - -#endif //LIGHT_TRANSMITTANCE_USED - } - - if (roughness > 0.0) { // FIXME: roughness == 0 should not disable specular light entirely - - // D + if (roughness > 0.0) { +#if defined(SPECULAR_SCHLICK_GGX) + float cNdotH = clamp(A + dot(N, H), 0.0, 1.0); +#endif + // Apply specular light. + // FIXME: roughness == 0 should not disable specular light entirely #if defined(SPECULAR_TOON) - - vec3 R = normalize(-reflect(L, N)); - float RdotV = dot(R, V); - float mid = 1.0 - roughness; - mid *= mid; - float intensity = smoothstep(mid - roughness * 0.5, mid + roughness * 0.5, RdotV) * mid; - diffuse_light += light_color * intensity * attenuation * specular_amount; // write to diffuse_light, as in toon shading you generally want no reflection + vec3 R = normalize(-reflect(L, N)); + float RdotV = dot(R, V); + float mid = 1.0 - roughness; + mid *= mid; + float intensity = smoothstep(mid - roughness * 0.5, mid + roughness * 0.5, RdotV) * mid; + diffuse_light += light_color * intensity * attenuation * specular_amount; // write to diffuse_light, as in toon shading you generally want no reflection #elif defined(SPECULAR_DISABLED) - // none.. + // Do nothing. #elif defined(SPECULAR_SCHLICK_GGX) - // shlick+ggx as default - float alpha_ggx = roughness * roughness; + // shlick+ggx as default + float alpha_ggx = roughness * roughness; #if defined(LIGHT_ANISOTROPY_USED) - - float aspect = sqrt(1.0 - anisotropy * 0.9); - float ax = alpha_ggx / aspect; - float ay = alpha_ggx * aspect; - float XdotH = dot(T, H); - float YdotH = dot(B, H); - float D = D_GGX_anisotropic(cNdotH, ax, ay, XdotH, YdotH); - float G = V_GGX_anisotropic(ax, ay, dot(T, V), dot(T, L), dot(B, V), dot(B, L), cNdotV, cNdotL); + float aspect = sqrt(1.0 - anisotropy * 0.9); + float ax = alpha_ggx / aspect; + float ay = alpha_ggx * aspect; + float XdotH = dot(T, H); + float YdotH = dot(B, H); + float D = D_GGX_anisotropic(cNdotH, ax, ay, XdotH, YdotH); + float G = V_GGX_anisotropic(ax, ay, dot(T, V), dot(T, L), dot(B, V), dot(B, L), cNdotV, cNdotL); #else // LIGHT_ANISOTROPY_USED - float D = D_GGX(cNdotH, alpha_ggx); - float G = V_GGX(cNdotL, cNdotV, alpha_ggx); + float D = D_GGX(cNdotH, alpha_ggx); + float G = V_GGX(cNdotL, cNdotV, alpha_ggx); #endif // LIGHT_ANISOTROPY_USED // F - float cLdotH5 = SchlickFresnel(cLdotH); - // Calculate Fresnel using specular occlusion term from Filament: - // https://google.github.io/filament/Filament.html#lighting/occlusion/specularocclusion - float f90 = clamp(dot(f0, vec3(50.0 * 0.33)), metallic, 1.0); - vec3 F = f0 + (f90 - f0) * cLdotH5; - - vec3 specular_brdf_NL = cNdotL * D * F * G; - - specular_light += specular_brdf_NL * light_color * attenuation * specular_amount; -#endif - -#if defined(LIGHT_CLEARCOAT_USED) - // Clearcoat ignores normal_map, use vertex normal instead - float ccNdotL = max(min(A + dot(vertex_normal, L), 1.0), 0.0); - float ccNdotH = clamp(A + dot(vertex_normal, H), 0.0, 1.0); - float ccNdotV = max(dot(vertex_normal, V), 1e-4); - -#if !defined(SPECULAR_SCHLICK_GGX) - float cLdotH5 = SchlickFresnel(cLdotH); + float cLdotH5 = SchlickFresnel(cLdotH); + // Calculate Fresnel using specular occlusion term from Filament: + // https://google.github.io/filament/Filament.html#lighting/occlusion/specularocclusion + float f90 = clamp(dot(f0, vec3(50.0 * 0.33)), metallic, 1.0); + vec3 F = f0 + (f90 - f0) * cLdotH5; + vec3 specular_brdf_NL = cNdotL * D * F * G; + specular_light += specular_brdf_NL * light_color * attenuation * specular_amount; #endif - float Dr = D_GGX(ccNdotH, mix(0.001, 0.1, clearcoat_roughness)); - float Gr = 0.25 / (cLdotH * cLdotH); - float Fr = mix(.04, 1.0, cLdotH5); - float clearcoat_specular_brdf_NL = clearcoat * Gr * Fr * Dr * cNdotL; - - specular_light += clearcoat_specular_brdf_NL * light_color * attenuation * specular_amount; - // TODO: Clearcoat adds light to the scene right now (it is non-energy conserving), both diffuse and specular need to be scaled by (1.0 - FR) - // but to do so we need to rearrange this entire function -#endif // LIGHT_CLEARCOAT_USED - } + } #ifdef USE_SHADOW_TO_OPACITY - alpha = min(alpha, clamp(1.0 - attenuation, 0.0, 1.0)); + alpha = min(alpha, clamp(1.0 - attenuation, 0.0, 1.0)); #endif - -#endif //defined(LIGHT_CODE_USED) + } +#endif // LIGHT_CODE_USED } #ifndef SHADOWS_DISABLED @@ -412,9 +386,43 @@ float get_omni_attenuation(float distance, float inv_range, float decay) { return nd * pow(max(distance, 0.0001), -decay); } -float light_process_omni_shadow(uint idx, vec3 vertex, vec3 normal, float taa_frame_count) { +void light_process_omni(uint idx, vec3 vertex, vec3 eye_vec, vec3 normal, vec3 vertex_ddx, vec3 vertex_ddy, vec3 f0, uint orms, float taa_frame_count, vec3 albedo, inout float alpha, vec2 screen_uv, +#ifdef LIGHT_BACKLIGHT_USED + vec3 backlight, +#endif +#ifdef LIGHT_TRANSMITTANCE_USED + vec4 transmittance_color, + float transmittance_depth, + float transmittance_boost, +#endif +#ifdef LIGHT_RIM_USED + float rim, float rim_tint, +#endif +#ifdef LIGHT_CLEARCOAT_USED + float clearcoat, float clearcoat_roughness, vec3 vertex_normal, +#endif +#ifdef LIGHT_ANISOTROPY_USED + vec3 binormal, vec3 tangent, float anisotropy, +#endif + inout vec3 diffuse_light, inout vec3 specular_light) { + const float EPSILON = 1e-3f; + + // Omni light attenuation. + vec3 light_rel_vec = omni_lights.data[idx].position - vertex; + float light_length = length(light_rel_vec); + float omni_attenuation = get_omni_attenuation(light_length, omni_lights.data[idx].inv_radius, omni_lights.data[idx].attenuation); + + // Compute size. + float size = 0.0; + if (sc_use_light_soft_shadows() && omni_lights.data[idx].size > 0.0) { + float t = omni_lights.data[idx].size / max(0.001, light_length); + size = max(0.0, 1.0 - 1 / sqrt(1 + t * t)); + } + + float shadow = 1.0; #ifndef SHADOWS_DISABLED - if (omni_lights.data[idx].shadow_opacity > 0.001) { + // Omni light shadow. + if (omni_attenuation > EPSILON && omni_lights.data[idx].shadow_opacity > 0.001) { // there is a shadowmap vec2 texel_size = scene_data_block.data.shadow_atlas_pixel_size; vec4 base_uv_rect = omni_lights.data[idx].atlas_rect; @@ -432,8 +440,6 @@ float light_process_omni_shadow(uint idx, vec3 vertex, vec3 normal, float taa_fr vec3 local_normal = normalize(mat3(omni_lights.data[idx].shadow_matrix) * normal); vec3 normal_bias = local_normal * omni_lights.data[idx].shadow_normal_bias * (1.0 - abs(dot(local_normal, shadow_dir))); - float shadow; - if (sc_use_light_soft_shadows() && omni_lights.data[idx].soft_shadow_size > 0.0) { //soft shadow @@ -539,49 +545,14 @@ float light_process_omni_shadow(uint idx, vec3 vertex, vec3 normal, float taa_fr depth = 1.0 - depth; shadow = mix(1.0, sample_omni_pcf_shadow(shadow_atlas, omni_lights.data[idx].soft_shadow_scale / shadow_sample.z, pos, uv_rect, flip_offset, depth, taa_frame_count), omni_lights.data[idx].shadow_opacity); } - - return shadow; } #endif - return 1.0; -} - -void light_process_omni(uint idx, vec3 vertex, vec3 eye_vec, vec3 normal, vec3 vertex_ddx, vec3 vertex_ddy, vec3 f0, uint orms, float shadow, vec3 albedo, inout float alpha, vec2 screen_uv, -#ifdef LIGHT_BACKLIGHT_USED - vec3 backlight, -#endif -#ifdef LIGHT_TRANSMITTANCE_USED - vec4 transmittance_color, - float transmittance_depth, - float transmittance_boost, -#endif -#ifdef LIGHT_RIM_USED - float rim, float rim_tint, -#endif -#ifdef LIGHT_CLEARCOAT_USED - float clearcoat, float clearcoat_roughness, vec3 vertex_normal, -#endif -#ifdef LIGHT_ANISOTROPY_USED - vec3 binormal, vec3 tangent, float anisotropy, -#endif - inout vec3 diffuse_light, inout vec3 specular_light) { - vec3 light_rel_vec = omni_lights.data[idx].position - vertex; - float light_length = length(light_rel_vec); - float omni_attenuation = get_omni_attenuation(light_length, omni_lights.data[idx].inv_radius, omni_lights.data[idx].attenuation); - float light_attenuation = omni_attenuation; vec3 color = omni_lights.data[idx].color; - float size_A = 0.0; - - if (sc_use_light_soft_shadows() && omni_lights.data[idx].size > 0.0) { - float t = omni_lights.data[idx].size / max(0.001, light_length); - size_A = max(0.0, 1.0 - 1 / sqrt(1 + t * t)); - } - #ifdef LIGHT_TRANSMITTANCE_USED float transmittance_z = transmittance_depth; //no transmittance by default - transmittance_color.a *= light_attenuation; + transmittance_color.a *= omni_attenuation; #ifndef SHADOWS_DISABLED if (omni_lights.data[idx].shadow_opacity > 0.001) { // Redo shadowmapping, but shrink the model a bit to avoid artifacts. @@ -673,9 +644,8 @@ void light_process_omni(uint idx, vec3 vertex, vec3 eye_vec, vec3 normal, vec3 v } } - light_attenuation *= shadow; - - light_compute(normal, normalize(light_rel_vec), eye_vec, size_A, color, false, light_attenuation, f0, orms, omni_lights.data[idx].specular_amount, albedo, alpha, screen_uv, + vec3 light_rel_vec_norm = light_rel_vec / light_length; + light_compute(normal, light_rel_vec_norm, eye_vec, size, color, false, omni_attenuation * shadow, f0, orms, omni_lights.data[idx].specular_amount, albedo, alpha, screen_uv, #ifdef LIGHT_BACKLIGHT_USED backlight, #endif @@ -698,15 +668,66 @@ void light_process_omni(uint idx, vec3 vertex, vec3 eye_vec, vec3 normal, vec3 v specular_light); } -float light_process_spot_shadow(uint idx, vec3 vertex, vec3 normal, float taa_frame_count) { -#ifndef SHADOWS_DISABLED - if (spot_lights.data[idx].shadow_opacity > 0.001) { - vec3 light_rel_vec = spot_lights.data[idx].position - vertex; - float light_length = length(light_rel_vec); - vec3 spot_dir = spot_lights.data[idx].direction; +vec2 normal_to_panorama(vec3 n) { + n = normalize(n); + vec2 panorama_coords = vec2(atan(n.x, n.z), acos(-n.y)); + + if (panorama_coords.x < 0.0) { + panorama_coords.x += M_PI * 2.0; + } + + panorama_coords /= vec2(M_PI * 2.0, M_PI); + return panorama_coords; +} + +void light_process_spot(uint idx, vec3 vertex, vec3 eye_vec, vec3 normal, vec3 vertex_ddx, vec3 vertex_ddy, vec3 f0, uint orms, float taa_frame_count, vec3 albedo, inout float alpha, vec2 screen_uv, +#ifdef LIGHT_BACKLIGHT_USED + vec3 backlight, +#endif +#ifdef LIGHT_TRANSMITTANCE_USED + vec4 transmittance_color, + float transmittance_depth, + float transmittance_boost, +#endif +#ifdef LIGHT_RIM_USED + float rim, float rim_tint, +#endif +#ifdef LIGHT_CLEARCOAT_USED + float clearcoat, float clearcoat_roughness, vec3 vertex_normal, +#endif +#ifdef LIGHT_ANISOTROPY_USED + vec3 binormal, vec3 tangent, float anisotropy, +#endif + inout vec3 diffuse_light, + inout vec3 specular_light) { + const float EPSILON = 1e-3f; + + // Spot light attenuation. + vec3 light_rel_vec = spot_lights.data[idx].position - vertex; + float light_length = length(light_rel_vec); + vec3 light_rel_vec_norm = light_rel_vec / light_length; + float spot_attenuation = get_omni_attenuation(light_length, spot_lights.data[idx].inv_radius, spot_lights.data[idx].attenuation); + vec3 spot_dir = spot_lights.data[idx].direction; - vec3 shadow_dir = light_rel_vec / light_length; - vec3 normal_bias = normal * light_length * spot_lights.data[idx].shadow_normal_bias * (1.0 - abs(dot(normal, shadow_dir))); + // This conversion to a highp float is crucial to prevent light leaking + // due to precision errors in the following calculations (cone angle is mediump). + highp float cone_angle = spot_lights.data[idx].cone_angle; + float scos = max(dot(-light_rel_vec_norm, spot_dir), cone_angle); + float spot_rim = max(0.0001, (1.0 - scos) / (1.0 - cone_angle)); + spot_attenuation *= 1.0 - pow(spot_rim, spot_lights.data[idx].cone_attenuation); + + // Compute size. + float size = 0.0; + if (sc_use_light_soft_shadows() && spot_lights.data[idx].size > 0.0) { + float t = spot_lights.data[idx].size / max(0.001, light_length); + size = max(0.0, 1.0 - 1 / sqrt(1 + t * t)); + } + + float shadow = 1.0; +#ifndef SHADOWS_DISABLED + // Spot light shadow. + if (spot_attenuation > EPSILON && spot_lights.data[idx].shadow_opacity > 0.001) { + vec3 normal_bias = normal * light_length * spot_lights.data[idx].shadow_normal_bias * (1.0 - abs(dot(normal, light_rel_vec_norm))); //there is a shadowmap vec4 v = vec4(vertex + normal_bias, 1.0); @@ -715,7 +736,6 @@ float light_process_spot_shadow(uint idx, vec3 vertex, vec3 normal, float taa_fr splane.z += spot_lights.data[idx].shadow_bias / (light_length * spot_lights.data[idx].inv_radius); splane /= splane.w; - float shadow; if (sc_use_light_soft_shadows() && spot_lights.data[idx].soft_shadow_size > 0.0) { //soft shadow @@ -772,73 +792,15 @@ float light_process_spot_shadow(uint idx, vec3 vertex, vec3 normal, float taa_fr vec3 shadow_uv = vec3(splane.xy * spot_lights.data[idx].atlas_rect.zw + spot_lights.data[idx].atlas_rect.xy, splane.z); shadow = mix(1.0, sample_pcf_shadow(shadow_atlas, spot_lights.data[idx].soft_shadow_scale * scene_data_block.data.shadow_atlas_pixel_size, shadow_uv, taa_frame_count), spot_lights.data[idx].shadow_opacity); } - - return shadow; } - #endif // SHADOWS_DISABLED - return 1.0; -} - -vec2 normal_to_panorama(vec3 n) { - n = normalize(n); - vec2 panorama_coords = vec2(atan(n.x, n.z), acos(-n.y)); - - if (panorama_coords.x < 0.0) { - panorama_coords.x += M_PI * 2.0; - } - - panorama_coords /= vec2(M_PI * 2.0, M_PI); - return panorama_coords; -} - -void light_process_spot(uint idx, vec3 vertex, vec3 eye_vec, vec3 normal, vec3 vertex_ddx, vec3 vertex_ddy, vec3 f0, uint orms, float shadow, vec3 albedo, inout float alpha, vec2 screen_uv, -#ifdef LIGHT_BACKLIGHT_USED - vec3 backlight, -#endif -#ifdef LIGHT_TRANSMITTANCE_USED - vec4 transmittance_color, - float transmittance_depth, - float transmittance_boost, -#endif -#ifdef LIGHT_RIM_USED - float rim, float rim_tint, -#endif -#ifdef LIGHT_CLEARCOAT_USED - float clearcoat, float clearcoat_roughness, vec3 vertex_normal, -#endif -#ifdef LIGHT_ANISOTROPY_USED - vec3 binormal, vec3 tangent, float anisotropy, -#endif - inout vec3 diffuse_light, - inout vec3 specular_light) { - vec3 light_rel_vec = spot_lights.data[idx].position - vertex; - float light_length = length(light_rel_vec); - float spot_attenuation = get_omni_attenuation(light_length, spot_lights.data[idx].inv_radius, spot_lights.data[idx].attenuation); - vec3 spot_dir = spot_lights.data[idx].direction; - - // This conversion to a highp float is crucial to prevent light leaking - // due to precision errors in the following calculations (cone angle is mediump). - highp float cone_angle = spot_lights.data[idx].cone_angle; - float scos = max(dot(-normalize(light_rel_vec), spot_dir), cone_angle); - float spot_rim = max(0.0001, (1.0 - scos) / (1.0 - cone_angle)); - - spot_attenuation *= 1.0 - pow(spot_rim, spot_lights.data[idx].cone_attenuation); - float light_attenuation = spot_attenuation; vec3 color = spot_lights.data[idx].color; float specular_amount = spot_lights.data[idx].specular_amount; - float size_A = 0.0; - - if (sc_use_light_soft_shadows() && spot_lights.data[idx].size > 0.0) { - float t = spot_lights.data[idx].size / max(0.001, light_length); - size_A = max(0.0, 1.0 - 1 / sqrt(1 + t * t)); - } - #ifdef LIGHT_TRANSMITTANCE_USED float transmittance_z = transmittance_depth; - transmittance_color.a *= light_attenuation; + transmittance_color.a *= spot_attenuation; #ifndef SHADOWS_DISABLED if (spot_lights.data[idx].shadow_opacity > 0.001) { vec4 splane = (spot_lights.data[idx].shadow_matrix * vec4(vertex - normalize(normal) * spot_lights.data[idx].transmittance_bias, 1.0)); @@ -882,9 +844,8 @@ void light_process_spot(uint idx, vec3 vertex, vec3 eye_vec, vec3 normal, vec3 v color *= proj.rgb * proj.a; } } - light_attenuation *= shadow; - light_compute(normal, normalize(light_rel_vec), eye_vec, size_A, color, false, light_attenuation, f0, orms, spot_lights.data[idx].specular_amount, albedo, alpha, screen_uv, + light_compute(normal, light_rel_vec_norm, eye_vec, size, color, false, spot_attenuation * shadow, f0, orms, spot_lights.data[idx].specular_amount, albedo, alpha, screen_uv, #ifdef LIGHT_BACKLIGHT_USED backlight, #endif From a694cd7120c110d9965df582de4739024ee9fae4 Mon Sep 17 00:00:00 2001 From: landervr <31851431+CpnWaffle@users.noreply.github.com> Date: Mon, 2 Dec 2024 11:52:11 +0100 Subject: [PATCH 05/33] Clean up UI of ReflectionProbe --- .../gizmos/reflection_probe_gizmo_plugin.cpp | 23 ++++++++----------- 1 file changed, 9 insertions(+), 14 deletions(-) diff --git a/editor/plugins/gizmos/reflection_probe_gizmo_plugin.cpp b/editor/plugins/gizmos/reflection_probe_gizmo_plugin.cpp index 85b84262f7c2..d74c434410df 100644 --- a/editor/plugins/gizmos/reflection_probe_gizmo_plugin.cpp +++ b/editor/plugins/gizmos/reflection_probe_gizmo_plugin.cpp @@ -47,7 +47,7 @@ ReflectionProbeGizmoPlugin::ReflectionProbeGizmoPlugin() { gizmo_color.a = 0.5; create_material("reflection_internal_material", gizmo_color); - gizmo_color.a = 0.1; + gizmo_color.a = 0.025; create_material("reflection_probe_solid_material", gizmo_color); create_icon_material("reflection_probe_icon", EditorNode::get_singleton()->get_editor_theme()->get_icon(SNAME("GizmoReflectionProbe"), EditorStringName(EditorIcons))); @@ -165,22 +165,17 @@ void ReflectionProbeGizmoPlugin::redraw(EditorNode3DGizmo *p_gizmo) { aabb.position = -size / 2; aabb.size = size; - for (int i = 0; i < 8; i++) { - Vector3 ep = aabb.get_endpoint(i); - internal_lines.push_back(probe->get_origin_offset()); - internal_lines.push_back(ep); - } - Vector handles = helper->box_get_handles(probe->get_size()); - for (int i = 0; i < 3; i++) { - Vector3 orig_handle = probe->get_origin_offset(); - orig_handle[i] -= 0.25; - lines.push_back(orig_handle); - handles.push_back(orig_handle); + if (probe->get_origin_offset() != Vector3(0.0, 0.0, 0.0)) { + for (int i = 0; i < 3; i++) { + Vector3 orig_handle = probe->get_origin_offset(); + orig_handle[i] -= 0.25; + lines.push_back(orig_handle); - orig_handle[i] += 0.5; - lines.push_back(orig_handle); + orig_handle[i] += 0.5; + lines.push_back(orig_handle); + } } Ref material = get_material("reflection_probe_material", p_gizmo); From 2b39314461576be071d15ffc0ae18ca1c92fc088 Mon Sep 17 00:00:00 2001 From: Lukas Tenbrink Date: Fri, 29 Nov 2024 02:06:35 +0100 Subject: [PATCH 06/33] Abstract the implementation of 3 ustring string length checks with strlen, adding an implementation for char32_t. --- core/string/ustring.cpp | 29 ++++++++++++----------------- 1 file changed, 12 insertions(+), 17 deletions(-) diff --git a/core/string/ustring.cpp b/core/string/ustring.cpp index e6f7492a189b..9056d5b393bb 100644 --- a/core/string/ustring.cpp +++ b/core/string/ustring.cpp @@ -64,6 +64,15 @@ const char16_t Char16String::_null = 0; const char32_t String::_null = 0; const char32_t String::_replacement_char = 0xfffd; +// strlen equivalent function for char32_t * arguments. +_FORCE_INLINE_ size_t strlen(const char32_t *p_str) { + const char32_t *ptr = p_str; + while (*ptr != 0) { + ++ptr; + } + return ptr - p_str; +} + bool select_word(const String &p_s, int p_col, int &r_beg, int &r_end) { const String &s = p_s; int beg = CLAMP(p_col, 0, s.length()); @@ -423,11 +432,7 @@ void String::copy_from(const char32_t *p_cstr) { return; } - int len = 0; - const char32_t *ptr = p_cstr; - while (*(ptr++) != 0) { - len++; - } + const int len = strlen(p_cstr); if (len == 0) { resize(0); @@ -628,12 +633,7 @@ String &String::operator+=(char32_t p_char) { bool String::operator==(const char *p_str) const { // compare Latin-1 encoded c-string - int len = 0; - const char *aux = p_str; - - while (*(aux++) != 0) { - len++; - } + int len = strlen(p_str); if (length() != len) { return false; @@ -667,12 +667,7 @@ bool String::operator==(const wchar_t *p_str) const { } bool String::operator==(const char32_t *p_str) const { - int len = 0; - const char32_t *aux = p_str; - - while (*(aux++) != 0) { - len++; - } + const int len = strlen(p_str); if (length() != len) { return false; From df2c2ca3c32468d73137eff82fa6861b4e97ae3d Mon Sep 17 00:00:00 2001 From: Pablo Andres Fuente Date: Tue, 3 Dec 2024 00:12:52 -0300 Subject: [PATCH 07/33] Fix TCPServer "Should disconnect client" test "Should disconnect client" test was failing randomly on Mac CI tests, so this PR is making it more reliable reading on the closed client instead of writing to it --- tests/core/io/test_tcp_server.h | 9 ++++++--- 1 file changed, 6 insertions(+), 3 deletions(-) diff --git a/tests/core/io/test_tcp_server.h b/tests/core/io/test_tcp_server.h index 69833f44119f..9bf1c2ecad5c 100644 --- a/tests/core/io/test_tcp_server.h +++ b/tests/core/io/test_tcp_server.h @@ -85,8 +85,8 @@ Error poll(Ref p_client) { const uint64_t time = OS::get_singleton()->get_ticks_usec(); Error err = p_client->poll(); while (err != Error::OK && (OS::get_singleton()->get_ticks_usec() - time) < MAX_WAIT_USEC) { - err = p_client->poll(); OS::get_singleton()->delay_usec(SLEEP_DURATION); + err = p_client->poll(); } return err; } @@ -183,8 +183,8 @@ TEST_CASE("[TCPServer] When stopped shouldn't accept new connections") { time = OS::get_singleton()->get_ticks_usec(); Error err = new_client->poll(); while (err != Error::OK && err != Error::ERR_CONNECTION_ERROR && (OS::get_singleton()->get_ticks_usec() - time) < MAX_WAIT_USEC) { - err = new_client->poll(); OS::get_singleton()->delay_usec(SLEEP_DURATION); + err = new_client->poll(); } REQUIRE((err == Error::OK || err == Error::ERR_CONNECTION_ERROR)); StreamPeerTCP::Status status = new_client->get_status(); @@ -210,7 +210,10 @@ TEST_CASE("[TCPServer] Should disconnect client") { server->stop(); CHECK_FALSE(server->is_listening()); - client->put_string(hello_world); + // Reading for a closed connection will print an error. + ERR_PRINT_OFF; + CHECK_EQ(client->get_string(), String()); + ERR_PRINT_ON; REQUIRE_EQ(poll(client), Error::OK); CHECK_EQ(client->get_status(), StreamPeerTCP::STATUS_NONE); } From fa7615be9e1627734565500690b34701d587e247 Mon Sep 17 00:00:00 2001 From: Dario Date: Tue, 3 Dec 2024 12:59:56 -0300 Subject: [PATCH 08/33] Add specialization for directional light split blend and fog. --- .../forward_mobile/render_forward_mobile.cpp | 15 +++-- .../scene_shader_forward_mobile.cpp | 8 ++- .../scene_shader_forward_mobile.h | 20 ++++++- .../forward_mobile/scene_forward_mobile.glsl | 14 +++-- .../scene_forward_mobile_inc.glsl | 55 ++++++++++++++----- .../renderer_rd/storage_rd/light_storage.h | 10 ++++ 6 files changed, 96 insertions(+), 26 deletions(-) diff --git a/servers/rendering/renderer_rd/forward_mobile/render_forward_mobile.cpp b/servers/rendering/renderer_rd/forward_mobile/render_forward_mobile.cpp index e93b7f0339d5..e2d6796566b6 100644 --- a/servers/rendering/renderer_rd/forward_mobile/render_forward_mobile.cpp +++ b/servers/rendering/renderer_rd/forward_mobile/render_forward_mobile.cpp @@ -1005,13 +1005,20 @@ void RenderForwardMobile::_render_scene(RenderDataRD *p_render_data, const Color { base_specialization.use_directional_soft_shadows = p_render_data->directional_light_count > 0 ? p_render_data->directional_light_soft_shadows : false; base_specialization.directional_lights = p_render_data->directional_light_count; + base_specialization.directional_light_blend_splits = light_storage->get_directional_light_blend_splits(p_render_data->directional_light_count); if (!is_environment(p_render_data->environment) || !environment_get_fog_enabled(p_render_data->environment)) { base_specialization.disable_fog = true; - } - - if (p_render_data->environment.is_valid() && environment_get_fog_mode(p_render_data->environment) == RS::EnvironmentFogMode::ENV_FOG_MODE_DEPTH) { - base_specialization.use_depth_fog = true; + base_specialization.use_fog_aerial_perspective = false; + base_specialization.use_fog_sun_scatter = false; + base_specialization.use_fog_height_density = false; + base_specialization.use_depth_fog = false; + } else { + base_specialization.disable_fog = false; + base_specialization.use_fog_aerial_perspective = environment_get_fog_aerial_perspective(p_render_data->environment) > 0.0; + base_specialization.use_fog_sun_scatter = environment_get_fog_sun_scatter(p_render_data->environment) > 0.001; + base_specialization.use_fog_height_density = abs(environment_get_fog_height_density(p_render_data->environment)) >= 0.0001; + base_specialization.use_depth_fog = p_render_data->environment.is_valid() && environment_get_fog_mode(p_render_data->environment) == RS::EnvironmentFogMode::ENV_FOG_MODE_DEPTH; } base_specialization.scene_use_ambient_cubemap = use_ambient_cubemap; diff --git a/servers/rendering/renderer_rd/forward_mobile/scene_shader_forward_mobile.cpp b/servers/rendering/renderer_rd/forward_mobile/scene_shader_forward_mobile.cpp index 4525b50b6e5e..6274dd764407 100644 --- a/servers/rendering/renderer_rd/forward_mobile/scene_shader_forward_mobile.cpp +++ b/servers/rendering/renderer_rd/forward_mobile/scene_shader_forward_mobile.cpp @@ -238,6 +238,7 @@ void SceneShaderForwardMobile::ShaderData::_create_pipeline(PipelineKey p_pipeli "SPEC PACKED #0:", p_pipeline_key.shader_specialization.packed_0, "SPEC PACKED #1:", p_pipeline_key.shader_specialization.packed_1, "SPEC PACKED #2:", p_pipeline_key.shader_specialization.packed_2, + "SPEC PACKED #3:", p_pipeline_key.shader_specialization.packed_3, "RENDER PASS:", p_pipeline_key.render_pass, "WIREFRAME:", p_pipeline_key.wireframe); #endif @@ -328,7 +329,12 @@ void SceneShaderForwardMobile::ShaderData::_create_pipeline(PipelineKey p_pipeli specialization_constants.push_back(sc); sc.constant_id = 2; - sc.float_value = p_pipeline_key.shader_specialization.packed_2; + sc.int_value = p_pipeline_key.shader_specialization.packed_2; + sc.type = RD::PIPELINE_SPECIALIZATION_CONSTANT_TYPE_INT; + specialization_constants.push_back(sc); + + sc.constant_id = 3; + sc.float_value = p_pipeline_key.shader_specialization.packed_3; sc.type = RD::PIPELINE_SPECIALIZATION_CONSTANT_TYPE_FLOAT; specialization_constants.push_back(sc); diff --git a/servers/rendering/renderer_rd/forward_mobile/scene_shader_forward_mobile.h b/servers/rendering/renderer_rd/forward_mobile/scene_shader_forward_mobile.h index e2549d1f0062..a40c5446c74a 100644 --- a/servers/rendering/renderer_rd/forward_mobile/scene_shader_forward_mobile.h +++ b/servers/rendering/renderer_rd/forward_mobile/scene_shader_forward_mobile.h @@ -67,6 +67,9 @@ class SceneShaderForwardMobile { uint32_t projector_use_mipmaps : 1; uint32_t disable_fog : 1; uint32_t use_depth_fog : 1; + uint32_t use_fog_aerial_perspective : 1; + uint32_t use_fog_sun_scatter : 1; + uint32_t use_fog_height_density : 1; uint32_t use_lightmap_bicubic_filter : 1; uint32_t multimesh : 1; uint32_t multimesh_format_2d : 1; @@ -75,7 +78,7 @@ class SceneShaderForwardMobile { uint32_t scene_use_ambient_cubemap : 1; uint32_t scene_use_reflection_cubemap : 1; uint32_t scene_roughness_limiter_enabled : 1; - uint32_t padding : 5; + uint32_t padding_0 : 2; uint32_t soft_shadow_samples : 6; uint32_t penumbra_shadow_samples : 6; }; @@ -97,9 +100,18 @@ class SceneShaderForwardMobile { uint32_t packed_1; }; + union { + struct { + uint32_t directional_light_blend_splits : 8; + uint32_t padding_1 : 24; + }; + + uint32_t packed_2; + }; + union { float luminance_multiplier; - float packed_2; + float packed_3; }; }; @@ -111,6 +123,10 @@ class SceneShaderForwardMobile { uint32_t packed_0; }; + + uint32_t padding_1; + uint32_t padding_2; + uint32_t padding_3; }; struct ShaderData : public RendererRD::MaterialStorage::ShaderData { diff --git a/servers/rendering/renderer_rd/shaders/forward_mobile/scene_forward_mobile.glsl b/servers/rendering/renderer_rd/shaders/forward_mobile/scene_forward_mobile.glsl index 0cb34557ea89..f72b576f28f9 100644 --- a/servers/rendering/renderer_rd/shaders/forward_mobile/scene_forward_mobile.glsl +++ b/servers/rendering/renderer_rd/shaders/forward_mobile/scene_forward_mobile.glsl @@ -768,7 +768,7 @@ layout(location = 0) out mediump vec4 frag_color; vec4 fog_process(vec3 vertex) { vec3 fog_color = scene_data_block.data.fog_light_color; - if (scene_data_block.data.fog_aerial_perspective > 0.0) { + if (sc_use_fog_aerial_perspective()) { vec3 sky_fog_color = vec3(0.0); vec3 cube_view = scene_data_block.data.radiance_inverse_xform * vertex; // mip_level always reads from the second mipmap and higher so the fog is always slightly blurred @@ -784,7 +784,7 @@ vec4 fog_process(vec3 vertex) { fog_color = mix(fog_color, sky_fog_color, scene_data_block.data.fog_aerial_perspective); } - if (scene_data_block.data.fog_sun_scatter > 0.001) { + if (sc_use_fog_sun_scatter()) { vec4 sun_scatter = vec4(0.0); float sun_total = 0.0; vec3 view = normalize(vertex); @@ -806,7 +806,7 @@ vec4 fog_process(vec3 vertex) { fog_amount = 1 - exp(min(0.0, -length(vertex) * scene_data_block.data.fog_density)); } - if (abs(scene_data_block.data.fog_height_density) >= 0.0001) { + if (sc_use_fog_height_density()) { float y = (scene_data_block.data.inv_view_matrix * vec4(vertex, 1.0)).y; float y_dist = y - scene_data_block.data.fog_height; @@ -1497,9 +1497,11 @@ void main() { pssm_coord /= pssm_coord.w; - shadow = sample_directional_pcf_shadow(directional_shadow_atlas, scene_data.directional_shadow_pixel_size * directional_lights.data[i].soft_shadow_scale * (blur_factor + (1.0 - blur_factor) * float(directional_lights.data[i].blend_splits)), pssm_coord, scene_data.taa_frame_count); + bool blend_split = sc_directional_light_blend_split(i); + float blend_split_weight = blend_split ? 1.0f : 0.0f; + shadow = sample_directional_pcf_shadow(directional_shadow_atlas, scene_data.directional_shadow_pixel_size * directional_lights.data[i].soft_shadow_scale * (blur_factor + (1.0 - blur_factor) * blend_split_weight), pssm_coord, scene_data.taa_frame_count); - if (directional_lights.data[i].blend_splits) { + if (blend_split) { float pssm_blend; float blur_factor2; @@ -1531,7 +1533,7 @@ void main() { pssm_coord /= pssm_coord.w; - float shadow2 = sample_directional_pcf_shadow(directional_shadow_atlas, scene_data.directional_shadow_pixel_size * directional_lights.data[i].soft_shadow_scale * (blur_factor2 + (1.0 - blur_factor2) * float(directional_lights.data[i].blend_splits)), pssm_coord, scene_data.taa_frame_count); + float shadow2 = sample_directional_pcf_shadow(directional_shadow_atlas, scene_data.directional_shadow_pixel_size * directional_lights.data[i].soft_shadow_scale * (blur_factor2 + (1.0 - blur_factor2) * blend_split_weight), pssm_coord, scene_data.taa_frame_count); shadow = mix(shadow, shadow2, pssm_blend); } diff --git a/servers/rendering/renderer_rd/shaders/forward_mobile/scene_forward_mobile_inc.glsl b/servers/rendering/renderer_rd/shaders/forward_mobile/scene_forward_mobile_inc.glsl index 2cc86482f6f0..5e4719c49881 100644 --- a/servers/rendering/renderer_rd/shaders/forward_mobile/scene_forward_mobile_inc.glsl +++ b/servers/rendering/renderer_rd/shaders/forward_mobile/scene_forward_mobile_inc.glsl @@ -23,8 +23,12 @@ layout(push_constant, std430) uniform DrawCall { #ifdef UBERSHADER uint sc_packed_0; uint sc_packed_1; - float sc_packed_2; + uint sc_packed_2; + float sc_packed_3; uint uc_packed_0; + uint uc_padding_1; + uint uc_padding_2; + uint uc_padding_3; #endif } draw_call; @@ -46,10 +50,14 @@ uint sc_packed_1() { return draw_call.sc_packed_1; } -float sc_packed_2() { +uint sc_packed_2() { return draw_call.sc_packed_2; } +float sc_packed_3() { + return draw_call.sc_packed_3; +} + uint uc_cull_mode() { return (draw_call.uc_packed_0 >> 0) & 3U; } @@ -59,7 +67,8 @@ uint uc_cull_mode() { // Pull the constants from the pipeline's specialization constants. layout(constant_id = 0) const uint pso_sc_packed_0 = 0; layout(constant_id = 1) const uint pso_sc_packed_1 = 0; -layout(constant_id = 2) const float pso_sc_packed_2 = 2.0; +layout(constant_id = 2) const uint pso_sc_packed_2 = 0; +layout(constant_id = 3) const float pso_sc_packed_3 = 2.0; uint sc_packed_0() { return pso_sc_packed_0; @@ -69,10 +78,14 @@ uint sc_packed_1() { return pso_sc_packed_1; } -float sc_packed_2() { +uint sc_packed_2() { return pso_sc_packed_2; } +float sc_packed_3() { + return pso_sc_packed_3; +} + #endif bool sc_use_light_projector() { @@ -103,38 +116,50 @@ bool sc_use_depth_fog() { return ((sc_packed_0() >> 6) & 1U) != 0; } -bool sc_use_lightmap_bicubic_filter() { +bool sc_use_fog_aerial_perspective() { return ((sc_packed_0() >> 7) & 1U) != 0; } -bool sc_multimesh() { +bool sc_use_fog_sun_scatter() { return ((sc_packed_0() >> 8) & 1U) != 0; } -bool sc_multimesh_format_2d() { +bool sc_use_fog_height_density() { return ((sc_packed_0() >> 9) & 1U) != 0; } -bool sc_multimesh_has_color() { +bool sc_use_lightmap_bicubic_filter() { return ((sc_packed_0() >> 10) & 1U) != 0; } -bool sc_multimesh_has_custom_data() { +bool sc_multimesh() { return ((sc_packed_0() >> 11) & 1U) != 0; } -bool sc_scene_use_ambient_cubemap() { +bool sc_multimesh_format_2d() { return ((sc_packed_0() >> 12) & 1U) != 0; } -bool sc_scene_use_reflection_cubemap() { +bool sc_multimesh_has_color() { return ((sc_packed_0() >> 13) & 1U) != 0; } -bool sc_scene_roughness_limiter_enabled() { +bool sc_multimesh_has_custom_data() { return ((sc_packed_0() >> 14) & 1U) != 0; } +bool sc_scene_use_ambient_cubemap() { + return ((sc_packed_0() >> 15) & 1U) != 0; +} + +bool sc_scene_use_reflection_cubemap() { + return ((sc_packed_0() >> 16) & 1U) != 0; +} + +bool sc_scene_roughness_limiter_enabled() { + return ((sc_packed_0() >> 17) & 1U) != 0; +} + uint sc_soft_shadow_samples() { return (sc_packed_0() >> 20) & 63U; } @@ -171,8 +196,12 @@ uint sc_decals() { return (sc_packed_1() >> 28) & 15U; } +bool sc_directional_light_blend_split(uint i) { + return ((sc_packed_2() >> i) & 1U) != 0; +} + float sc_luminance_multiplier() { - return sc_packed_2(); + return sc_packed_3(); } /* Set 0: Base Pass (never changes) */ diff --git a/servers/rendering/renderer_rd/storage_rd/light_storage.h b/servers/rendering/renderer_rd/storage_rd/light_storage.h index 1a92c5470dcb..b5f1523123de 100644 --- a/servers/rendering/renderer_rd/storage_rd/light_storage.h +++ b/servers/rendering/renderer_rd/storage_rd/light_storage.h @@ -799,6 +799,16 @@ class LightStorage : public RendererLightStorage { RID get_spot_light_buffer() { return spot_light_buffer; } RID get_directional_light_buffer() { return directional_light_buffer; } uint32_t get_max_directional_lights() { return max_directional_lights; } + uint32_t get_directional_light_blend_splits(uint32_t p_directional_light_count) const { + uint32_t blend_splits = 0; + for (uint32_t i = 0; i < p_directional_light_count; i++) { + if (directional_lights[i].blend_splits) { + blend_splits |= 1U << i; + } + } + + return blend_splits; + } bool has_directional_shadows(const uint32_t p_directional_light_count) { for (uint32_t i = 0; i < p_directional_light_count; i++) { if (directional_lights[i].shadow_opacity > 0.001) { From 274064ae7f4407977147b3d0c873f5ce04424421 Mon Sep 17 00:00:00 2001 From: hakro Date: Tue, 3 Dec 2024 20:06:34 +0100 Subject: [PATCH 09/33] Ignore custom tooltip if its text is empty in signals tab --- editor/connections_dialog.cpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/editor/connections_dialog.cpp b/editor/connections_dialog.cpp index d76c324be05c..b96c715bda7d 100644 --- a/editor/connections_dialog.cpp +++ b/editor/connections_dialog.cpp @@ -907,7 +907,7 @@ ConnectDialog::~ConnectDialog() { Control *ConnectionsDockTree::make_custom_tooltip(const String &p_text) const { // If it's not a doc tooltip, fallback to the default one. - if (p_text.contains("::")) { + if (p_text.is_empty() || p_text.contains("::")) { return nullptr; } From 121e6406370f5c9fe2b536bfc4e5de85a4b92a50 Mon Sep 17 00:00:00 2001 From: bruvzg <7645683+bruvzg@users.noreply.github.com> Date: Mon, 9 Sep 2024 00:04:07 +0300 Subject: [PATCH 10/33] [RTL] Fix indent in tables and tables in indent. --- scene/gui/rich_text_label.cpp | 50 ++++++++++++++++++++--------------- scene/gui/rich_text_label.h | 2 ++ 2 files changed, 30 insertions(+), 22 deletions(-) diff --git a/scene/gui/rich_text_label.cpp b/scene/gui/rich_text_label.cpp index 155824db3f33..ea0b8cd80864 100644 --- a/scene/gui/rich_text_label.cpp +++ b/scene/gui/rich_text_label.cpp @@ -364,7 +364,8 @@ float RichTextLabel::_resize_line(ItemFrame *p_frame, int p_line, const Reflines[p_line]; MutexLock lock(l.text_buf->get_mutex()); - l.offset.x = _find_margin(l.from, p_base_font, p_base_font_size) + l.prefix_width; + l.indent = _find_margin(l.from, p_base_font, p_base_font_size) + l.prefix_width; + l.offset.x = l.indent; l.text_buf->set_width(p_width - l.offset.x); PackedFloat32Array tab_stops = _find_tab_stops(l.from); @@ -501,7 +502,8 @@ float RichTextLabel::_shape_line(ItemFrame *p_frame, int p_line, const Ref } // Add indent. - l.offset.x = _find_margin(l.from, p_base_font, p_base_font_size) + l.prefix_width; + l.indent = _find_margin(l.from, p_base_font, p_base_font_size) + l.prefix_width; + l.offset.x = l.indent; l.text_buf->set_width(p_width - l.offset.x); l.text_buf->set_alignment(_find_alignment(l.from)); l.text_buf->set_direction(_find_direction(l.from)); @@ -625,8 +627,8 @@ float RichTextLabel::_shape_line(ItemFrame *p_frame, int p_line, const Ref t_char_count += cell_ch; remaining_characters -= cell_ch; - table->columns[column].min_width = MAX(table->columns[column].min_width, ceil(frame->lines[i].text_buf->get_size().x)); - table->columns[column].max_width = MAX(table->columns[column].max_width, ceil(frame->lines[i].text_buf->get_non_wrapped_size().x)); + table->columns[column].min_width = MAX(table->columns[column].min_width, frame->lines[i].indent + ceil(frame->lines[i].text_buf->get_size().x)); + table->columns[column].max_width = MAX(table->columns[column].max_width, frame->lines[i].indent + ceil(frame->lines[i].text_buf->get_non_wrapped_size().x)); } idx++; } @@ -977,6 +979,7 @@ int RichTextLabel::_draw_line(ItemFrame *p_frame, int p_line, const Vector2 &p_o if (frame->lines.size() != 0 && row < row_count) { Vector2 coff = frame->lines[0].offset; + coff.x -= frame->lines[0].indent; if (rtl) { coff.x = rect.size.width - table->columns[col].width - coff.x; } @@ -2548,6 +2551,10 @@ int RichTextLabel::_find_margin(Item *p_item, const Ref &p_base_font, int float margin = 0.0; while (item) { + if (item->type == ITEM_FRAME) { + break; + } + if (item->type == ITEM_INDENT) { Ref font = p_base_font; int font_size = p_base_font_size; @@ -4295,7 +4302,6 @@ void RichTextLabel::append_text(const String &p_bbcode) { parsing_bbcode.store(true); int pos = 0; - int indent_level = 0; bool in_bold = false; bool in_italics = false; @@ -4377,7 +4383,7 @@ void RichTextLabel::append_text(const String &p_bbcode) { in_italics = false; } if ((tag_stack.front()->get() == "indent") || (tag_stack.front()->get() == "ol") || (tag_stack.front()->get() == "ul")) { - indent_level--; + current_frame->indent_level--; } if (!tag_ok) { @@ -4650,44 +4656,44 @@ void RichTextLabel::append_text(const String &p_bbcode) { pos = brk_end + 1; tag_stack.push_front(tag); } else if (tag == "ul") { - indent_level++; - push_list(indent_level, LIST_DOTS, false); + current_frame->indent_level++; + push_list(current_frame->indent_level, LIST_DOTS, false); pos = brk_end + 1; tag_stack.push_front(tag); } else if (tag.begins_with("ul bullet=")) { String bullet = _get_tag_value(tag); - indent_level++; - push_list(indent_level, LIST_DOTS, false, bullet); + current_frame->indent_level++; + push_list(current_frame->indent_level, LIST_DOTS, false, bullet); pos = brk_end + 1; tag_stack.push_front("ul"); } else if ((tag == "ol") || (tag == "ol type=1")) { - indent_level++; - push_list(indent_level, LIST_NUMBERS, false); + current_frame->indent_level++; + push_list(current_frame->indent_level, LIST_NUMBERS, false); pos = brk_end + 1; tag_stack.push_front("ol"); } else if (tag == "ol type=a") { - indent_level++; - push_list(indent_level, LIST_LETTERS, false); + current_frame->indent_level++; + push_list(current_frame->indent_level, LIST_LETTERS, false); pos = brk_end + 1; tag_stack.push_front("ol"); } else if (tag == "ol type=A") { - indent_level++; - push_list(indent_level, LIST_LETTERS, true); + current_frame->indent_level++; + push_list(current_frame->indent_level, LIST_LETTERS, true); pos = brk_end + 1; tag_stack.push_front("ol"); } else if (tag == "ol type=i") { - indent_level++; - push_list(indent_level, LIST_ROMAN, false); + current_frame->indent_level++; + push_list(current_frame->indent_level, LIST_ROMAN, false); pos = brk_end + 1; tag_stack.push_front("ol"); } else if (tag == "ol type=I") { - indent_level++; - push_list(indent_level, LIST_ROMAN, true); + current_frame->indent_level++; + push_list(current_frame->indent_level, LIST_ROMAN, true); pos = brk_end + 1; tag_stack.push_front("ol"); } else if (tag == "indent") { - indent_level++; - push_indent(indent_level); + current_frame->indent_level++; + push_indent(current_frame->indent_level); pos = brk_end + 1; tag_stack.push_front(tag); } else if (tag.begins_with("lang=")) { diff --git a/scene/gui/rich_text_label.h b/scene/gui/rich_text_label.h index 00b3280a22f3..d9aac546ca07 100644 --- a/scene/gui/rich_text_label.h +++ b/scene/gui/rich_text_label.h @@ -153,6 +153,7 @@ class RichTextLabel : public Control { Color dc_ol_color; Vector2 offset; + float indent = 0.0; int char_offset = 0; int char_count = 0; @@ -205,6 +206,7 @@ class RichTextLabel : public Control { Size2 min_size_over = Size2(-1, -1); Size2 max_size_over = Size2(-1, -1); Rect2 padding; + int indent_level = 0; ItemFrame() { type = ITEM_FRAME; From b0cee57d814109b4670e0ca51ec64b59e32a4f77 Mon Sep 17 00:00:00 2001 From: kobewi Date: Tue, 3 Dec 2024 22:54:54 +0100 Subject: [PATCH 11/33] Fix extensions when saving resource as --- editor/editor_node.cpp | 15 ++++++--------- 1 file changed, 6 insertions(+), 9 deletions(-) diff --git a/editor/editor_node.cpp b/editor/editor_node.cpp index 222696a60846..bcc29e797c57 100644 --- a/editor/editor_node.cpp +++ b/editor/editor_node.cpp @@ -1427,7 +1427,7 @@ void EditorNode::save_resource_as(const Ref &p_resource, const String if (p_resource->get_path().is_resource_file()) { file->set_current_file(p_resource->get_path().get_file()); } else { - if (extensions.size()) { + if (!preferred.is_empty()) { String resource_name_snake_case = p_resource->get_class().to_snake_case(); file->set_current_file("new_" + resource_name_snake_case + "." + preferred.front()->get().to_lower()); } else { @@ -1436,18 +1436,15 @@ void EditorNode::save_resource_as(const Ref &p_resource, const String } } else if (!p_resource->get_path().is_empty()) { file->set_current_path(p_resource->get_path()); - if (extensions.size()) { - String ext = p_resource->get_path().get_extension().to_lower(); + if (!extensions.is_empty()) { + const String ext = p_resource->get_path().get_extension().to_lower(); if (extensions.find(ext) == nullptr) { file->set_current_path(p_resource->get_path().replacen("." + ext, "." + extensions.front()->get())); } } - } else if (preferred.size()) { - String existing; - if (extensions.size()) { - String resource_name_snake_case = p_resource->get_class().to_snake_case(); - existing = "new_" + resource_name_snake_case + "." + preferred.front()->get().to_lower(); - } + } else if (!preferred.is_empty()) { + const String resource_name_snake_case = p_resource->get_class().to_snake_case(); + const String existing = "new_" + resource_name_snake_case + "." + preferred.front()->get().to_lower(); file->set_current_path(existing); } file->set_title(TTR("Save Resource As...")); From e4106f8d61fb1f0163c5b00b4585a8cae3d962ca Mon Sep 17 00:00:00 2001 From: Hugo Locurcio Date: Tue, 18 Apr 2023 19:45:48 +0200 Subject: [PATCH 12/33] Show String properties' text in a tooltip in the inspector This allows previewing single-line or multipline strings that are too long too fit within the box in the inspector. --- editor/editor_inspector.cpp | 6 ++++++ editor/editor_inspector.h | 2 ++ editor/editor_properties.cpp | 10 ++++++++++ 3 files changed, 18 insertions(+) diff --git a/editor/editor_inspector.cpp b/editor/editor_inspector.cpp index 1e9c661b2401..f936a06e20e1 100644 --- a/editor/editor_inspector.cpp +++ b/editor/editor_inspector.cpp @@ -69,6 +69,12 @@ bool EditorInspector::_property_path_matches(const String &p_property_path, cons return false; } +String EditorProperty::get_tooltip_string(const String &p_string) const { + // Trim to 100 characters to prevent the tooltip from being too long. + constexpr int TOOLTIP_MAX_LENGTH = 100; + return p_string.left(TOOLTIP_MAX_LENGTH).strip_edges() + String((p_string.length() > TOOLTIP_MAX_LENGTH) ? "..." : ""); +} + Size2 EditorProperty::get_minimum_size() const { Size2 ms; Ref font = get_theme_font(SceneStringName(font), SNAME("Tree")); diff --git a/editor/editor_inspector.h b/editor/editor_inspector.h index 2e4633ccea98..e87d90dc8f3d 100644 --- a/editor/editor_inspector.h +++ b/editor/editor_inspector.h @@ -160,6 +160,8 @@ class EditorProperty : public Container { public: void emit_changed(const StringName &p_property, const Variant &p_value, const StringName &p_field = StringName(), bool p_changing = false); + String get_tooltip_string(const String &p_string) const; + virtual Size2 get_minimum_size() const override; void set_label(const String &p_label); diff --git a/editor/editor_properties.cpp b/editor/editor_properties.cpp index e6c944581cd4..c54c97437179 100644 --- a/editor/editor_properties.cpp +++ b/editor/editor_properties.cpp @@ -91,6 +91,10 @@ void EditorPropertyText::_text_changed(const String &p_string) { return; } + // Set tooltip so that the full text is displayed in a tooltip if hovered. + // This is useful when using a narrow inspector, as the text can be trimmed otherwise. + text->set_tooltip_text(get_tooltip_string(text->get_text())); + if (string_name) { emit_changed(get_edited_property(), StringName(p_string)); } else { @@ -104,6 +108,7 @@ void EditorPropertyText::update_property() { if (text->get_text() != s) { int caret = text->get_caret_column(); text->set_text(s); + text->set_tooltip_text(get_tooltip_string(s)); text->set_caret_column(caret); } text->set_editable(!is_read_only()); @@ -150,10 +155,14 @@ void EditorPropertyMultilineText::_set_read_only(bool p_read_only) { void EditorPropertyMultilineText::_big_text_changed() { text->set_text(big_text->get_text()); + // Set tooltip so that the full text is displayed in a tooltip if hovered. + // This is useful when using a narrow inspector, as the text can be trimmed otherwise. + text->set_tooltip_text(get_tooltip_string(big_text->get_text())); emit_changed(get_edited_property(), big_text->get_text(), "", true); } void EditorPropertyMultilineText::_text_changed() { + text->set_tooltip_text(get_tooltip_string(text->get_text())); emit_changed(get_edited_property(), text->get_text(), "", true); } @@ -182,6 +191,7 @@ void EditorPropertyMultilineText::update_property() { String t = get_edited_property_value(); if (text->get_text() != t) { text->set_text(t); + text->set_tooltip_text(get_tooltip_string(t)); if (big_text && big_text->is_visible_in_tree()) { big_text->set_text(t); } From 4051b43879fbc0070111bc93362d254c5a0946ef Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?R=C3=A9mi=20Verschelde?= Date: Wed, 4 Dec 2024 02:19:04 +0100 Subject: [PATCH 13/33] ufbx: Update to upstream 0.15.0 --- thirdparty/README.md | 2 +- thirdparty/ufbx/ufbx.c | 1898 +++++++++++++++++++++++++--------------- thirdparty/ufbx/ufbx.h | 120 +-- 3 files changed, 1269 insertions(+), 751 deletions(-) diff --git a/thirdparty/README.md b/thirdparty/README.md index 79d7e5b7b6ad..d87068ed167a 100644 --- a/thirdparty/README.md +++ b/thirdparty/README.md @@ -928,7 +928,7 @@ number and run the script. ## ufbx - Upstream: https://github.com/ufbx/ufbx -- Version: 0.14.3 (19bdb7e7ef02eb914d5e7211a3685f50ee6d27e3, 2024) +- Version: 0.15.0 (24eea6f40929fe0f679b7950def378edb003afdb, 2024) - License: MIT Files extracted from upstream source: diff --git a/thirdparty/ufbx/ufbx.c b/thirdparty/ufbx/ufbx.c index 2d13c78bd8c3..77bd3dea623f 100644 --- a/thirdparty/ufbx/ufbx.c +++ b/thirdparty/ufbx/ufbx.c @@ -20,6 +20,21 @@ // UFBX_TRACE Log calls of `ufbxi_check()` for tracing execution // UFBX_LITTLE_ENDIAN=0/1 Explicitly define little/big endian architecture // UFBX_PATH_SEPARATOR='' Specify default platform path separator +// UFBX_NO_SSE Do not try to include SSE + +// Dependencies: +// UFBX_NO_MALLOC Disable default malloc/realloc/free +// UFBX_NO_STDIO Disable stdio FILE API +// UFBX_EXTERNAL_MALLOC Link to external ufbx_malloc() interface +// UFBX_EXTERNAL_STDIO Link to external ufbx_stdio_() interface +// UFBX_EXTERNAL_MATH Link to external interface +// UFBX_EXTERNAL_STRING Link to external interface + +// Freestanding: +// UFBX_MATH_PREFIX='ufbx_' Prefix for external functions used +// UFBX_STRING_PREFIX='ufbx_' Prefix for external functions used +// UFBX_NO_LIBC Do not include libc (implies UFBX_EXTERNAL_MATH/STRING/MALLOC/STDIO by default) +// UFBX_NO_LIBC_TYPES Do not include any libc headers, you must define all types in // Mostly internal for debugging: // UFBX_STATIC_ANALYSIS Enable static analysis augmentation @@ -175,28 +190,71 @@ // -- Headers -#include -#include -#include -#include -#include -#include - -#if !defined(UFBX_NO_MATH_H) - #include - #define UFBX_INFINITY INFINITY - #define UFBX_NAN NAN +// Legacy mapping +#if !defined(UFBX_EXTERNAL_MATH) && defined(UFBX_NO_MATH_H) + #define UFBX_EXTERNAL_MATH #endif -#if !defined(UFBX_MATH_PREFIX) - #define UFBX_MATH_PREFIX +#if !defined(UFBX_NO_LIBC_TYPES) + #include #endif -#define ufbxi_math_cat2(a, b) a##b -#define ufbxi_math_cat(a, b) ufbxi_math_cat2(a, b) -#define ufbxi_math_fn(name) ufbxi_math_cat(UFBX_MATH_PREFIX, name) +#if !defined(UFBX_NO_LIBC) + #if !defined(UFBX_NO_FLOAT_H) + #include + #endif + #if !defined(UFBX_EXTERNAL_MATH) + #include + #endif + #if !defined(UFBX_EXTERNAL_STRING) + #include + #endif + #if !defined(UFBX_NO_STDIO) && !defined(UFBX_EXTERNAL_STDIO) + #include + #endif + #if !defined(UFBX_NO_MALLOC) && !defined(UFBX_EXTERNAL_MALLOC) + #include + #endif +#else + #if !defined(UFBX_EXTERNAL_MATH) && !defined(UFBX_NO_EXTERNAL_MATH) + #define UFBX_EXTERNAL_MATH + #endif + #if !defined(UFBX_EXTERNAL_STRING) && !defined(UFBX_NO_EXTERNAL_STRING) + #define UFBX_EXTERNAL_STRING + #endif + #if !defined(UFBX_EXTERNAL_MALLOC) && !defined(UFBX_NO_EXTERNAL_MALLOC) && !defined(UFBX_NO_MALLOC) + #define UFBX_EXTERNAL_MALLOC + #endif + #if !defined(UFBX_EXTERNAL_STDIO) && !defined(UFBX_NO_EXTERNAL_STDIO) && !defined(UFBX_NO_STDIO) + #define UFBX_EXTERNAL_STDIO + #endif +#endif + +#if defined(UFBX_EXTERNAL_STRING) && !defined(UFBX_STRING_PREFIX) + #define UFBX_STRING_PREFIX ufbx_ +#endif -#if !defined(UFBX_NO_MATH_DEFINES) +#if !defined(UFBX_EXTERNAL_MATH) + #if !defined(UFBX_MATH_PREFIX) + #define UFBX_MATH_PREFIX + #endif +#endif + +#define ufbxi_pre_cat2(a, b) a##b +#define ufbxi_pre_cat(a, b) ufbxi_pre_cat2(a, b) + +// -- External functions + +#ifndef ufbx_extern_abi + #if defined(UFBX_STATIC) + #define ufbx_extern_abi static + #else + #define ufbx_extern_abi + #endif +#endif + +#if defined(UFBX_MATH_PREFIX) + #define ufbxi_math_fn(name) ufbxi_pre_cat(UFBX_MATH_PREFIX, name) #define ufbx_sqrt ufbxi_math_fn(sqrt) #define ufbx_fabs ufbxi_math_fn(fabs) #define ufbx_pow ufbxi_math_fn(pow) @@ -216,31 +274,123 @@ #define ufbx_isnan ufbxi_math_fn(isnan) #endif -#if defined(UFBX_NO_MATH_H) && !defined(UFBX_NO_MATH_DECLARATIONS) - double ufbx_sqrt(double x); - double ufbx_sin(double x); - double ufbx_cos(double x); - double ufbx_tan(double x); - double ufbx_asin(double x); - double ufbx_acos(double x); - double ufbx_atan(double x); - double ufbx_atan2(double y, double x); - double ufbx_pow(double x, double y); - double ufbx_fmin(double a, double b); - double ufbx_fmax(double a, double b); - double ufbx_fabs(double x); - double ufbx_copysign(double x, double y); - double ufbx_nextafter(double x, double y); - double ufbx_rint(double x); - double ufbx_ceil(double x); - int ufbx_isnan(double x); +#if !defined(UFBX_NO_EXTERNAL_DEFINES) + +#if defined(__cplusplus) +extern "C" { +#endif + +#if defined(UFBX_EXTERNAL_MATH) + ufbx_extern_abi double ufbx_sqrt(double x); + ufbx_extern_abi double ufbx_sin(double x); + ufbx_extern_abi double ufbx_cos(double x); + ufbx_extern_abi double ufbx_tan(double x); + ufbx_extern_abi double ufbx_asin(double x); + ufbx_extern_abi double ufbx_acos(double x); + ufbx_extern_abi double ufbx_atan(double x); + ufbx_extern_abi double ufbx_atan2(double y, double x); + ufbx_extern_abi double ufbx_pow(double x, double y); + ufbx_extern_abi double ufbx_fmin(double a, double b); + ufbx_extern_abi double ufbx_fmax(double a, double b); + ufbx_extern_abi double ufbx_fabs(double x); + ufbx_extern_abi double ufbx_copysign(double x, double y); + ufbx_extern_abi double ufbx_nextafter(double x, double y); + ufbx_extern_abi double ufbx_rint(double x); + ufbx_extern_abi double ufbx_ceil(double x); + ufbx_extern_abi int ufbx_isnan(double x); +#endif + +#if defined(UFBX_EXTERNAL_STRING) + ufbx_extern_abi size_t ufbx_strlen(const char *str); + ufbx_extern_abi void *ufbx_memcpy(void *dst, const void *src, size_t count); + ufbx_extern_abi void *ufbx_memmove(void *dst, const void *src, size_t count); + ufbx_extern_abi void *ufbx_memset(void *dst, int ch, size_t count); + ufbx_extern_abi const void *ufbx_memchr(const void *ptr, int value, size_t count); + ufbx_extern_abi int ufbx_memcmp(const void *a, const void *b, size_t count); + ufbx_extern_abi int ufbx_strcmp(const char *a, const char *b); + ufbx_extern_abi int ufbx_strncmp(const char *a, const char *b, size_t count); +#endif + +#if defined(UFBX_EXTERNAL_MALLOC) + ufbx_extern_abi void *ufbx_malloc(size_t size); + ufbx_extern_abi void *ufbx_realloc(void *ptr, size_t old_size, size_t new_size); + ufbx_extern_abi void ufbx_free(void *ptr, size_t old_size); +#endif + +#if defined(UFBX_EXTERNAL_STDIO) + ufbx_extern_abi void *ufbx_stdio_open(const char *path, size_t path_len); + ufbx_extern_abi size_t ufbx_stdio_read(void *file, void *data, size_t size); + ufbx_extern_abi bool ufbx_stdio_skip(void *file, size_t size); + ufbx_extern_abi uint64_t ufbx_stdio_size(void *file); + ufbx_extern_abi void ufbx_stdio_close(void *file); +#endif + +#if defined(__cplusplus) +} +#endif + #endif #if !defined(UFBX_INFINITY) - #define UFBX_INFINITY (1e+300 * 1e+300) + #if defined(INFINITY) + #define UFBX_INFINITY INFINITY + #else + #define UFBX_INFINITY (1e+300 * 1e+300) + #endif #endif #if !defined(UFBX_NAN) - #define UFBX_NAN (UFBX_INFINITY * 0.0f) + #if defined(NAN) + #define UFBX_NAN NAN + #else + #define UFBX_NAN (UFBX_INFINITY * 0.0f) + #endif +#endif +#if !defined(UFBX_FLT_EPSILON) + #if defined(FLT_EPSILON) + #define UFBX_FLT_EPSILON FLT_EPSILON + #else + #define UFBX_FLT_EPSILON 1.192092896e-07f + #endif +#endif +#if !defined(UFBX_FLT_EVAL_METHOD) + #if defined(FLT_EVAL_METHOD) + #define UFBX_FLT_EVAL_METHOD FLT_EVAL_METHOD + #elif defined(__FLT_EVAL_METHOD__) + #define UFBX_FLT_EVAL_METHOD __FLT_EVAL_METHOD__ + #elif defined(_MSC_VER) && (defined(_M_X64) || defined(_M_ARM64)) + #define UFBX_FLT_EVAL_METHOD 0 + #else + #define UFBX_FLT_EVAL_METHOD -1 + #endif +#endif + +#if defined(ufbx_malloc) || defined(ufbx_realloc) || defined(ufbx_free) + // User provided allocators + #if !defined(ufbx_malloc) || !defined(ufbx_realloc) || !defined(ufbx_free) + #error Inconsistent custom global allocator + #endif +#elif defined(UFBX_NO_MALLOC) + #define ufbx_malloc(size) ((void)(size), (void*)NULL) + #define ufbx_realloc(ptr, old_size, new_size) ((void)(ptr), (void)(old_size), (void)(new_size), (void*)NULL) + #define ufbx_free(ptr, old_size) ((void)(ptr), (void*)(old_size)) +#elif defined(UFBX_EXTERNAL_MALLOC) + // Nop +#else + #define ufbx_malloc(size) malloc((size)) + #define ufbx_realloc(ptr, old_size, new_size) realloc((ptr), (new_size)) + #define ufbx_free(ptr, old_size) free((ptr)) +#endif + +#if !defined(ufbx_panic_handler) + static void ufbxi_panic_handler(const char *message) + { + (void)message; + #if !defined(UFBX_NO_STDIO) && !defined(UFBX_EXTERNAL_STDIO) + fprintf(stderr, "ufbx panic: %s\n", message); + #endif + ufbx_assert(false && "ufbx panic: See stderr for more information"); + } + #define ufbx_panic_handler ufbxi_panic_handler #endif // -- Platform @@ -414,8 +564,16 @@ #if defined(UFBX_STATIC_ANALYSIS) bool ufbxi_analysis_opaque; #define ufbxi_maybe_null(ptr) (ufbxi_analysis_opaque ? (ptr) : NULL) + #define ufbxi_analysis_assert(cond) ufbx_assert(cond) #else #define ufbxi_maybe_null(ptr) (ptr) + #define ufbxi_analysis_assert(cond) (void)0 +#endif + +#if defined(UFBX_STATIC_ANALYSIS) || defined(UFBX_UBSAN) + #define ufbxi_maybe_uninit(cond, value, def) ((cond) ? (value) : (def)) +#else + #define ufbxi_maybe_uninit(cond, value, def) (value) #endif #if !defined(ufbxi_trace) @@ -442,7 +600,7 @@ #endif #endif -#if !defined(UFBX_STANDARD_C) && (defined(_MSC_VER) && defined(_M_X64)) || ((defined(__GNUC__) || defined(__clang__)) && defined(__x86_64__)) || defined(UFBX_USE_SSE) +#if !defined(UFBX_STANDARD_C) && !defined(UFBX_NO_SSE) && (defined(_MSC_VER) && defined(_M_X64) && !defined(_M_ARM64EC)) || ((defined(__GNUC__) || defined(__clang__)) && defined(__x86_64__)) || defined(UFBX_USE_SSE) #define UFBXI_HAS_SSE 1 #include #include @@ -450,6 +608,108 @@ #define UFBXI_HAS_SSE 0 #endif +// -- Atomic counter + +#define UFBXI_THREAD_SAFE 1 + +#if defined(__cplusplus) + #define ufbxi_extern_c extern "C" +#else + #define ufbxi_extern_c +#endif + +#if !defined(UFBX_STANDARD_C) && (defined(__GNUC__) || defined(__clang__) || defined(__INTEL_COMPILER)) + typedef size_t ufbxi_atomic_counter; + #define ufbxi_atomic_counter_init(ptr) (*(ptr) = 0) + #define ufbxi_atomic_counter_free(ptr) (*(ptr) = 0) + #define ufbxi_atomic_counter_inc(ptr) __sync_fetch_and_add((ptr), 1) + #define ufbxi_atomic_counter_dec(ptr) __sync_fetch_and_sub((ptr), 1) + #define ufbxi_atomic_counter_load(ptr) __sync_fetch_and_add((ptr), 0) // TODO: Proper atomic load +#elif !defined(UFBX_STANDARD_C) && defined(_MSC_VER) + #if defined(_M_X64) || defined(_M_ARM64) + ufbxi_extern_c __int64 _InterlockedIncrement64(__int64 volatile * lpAddend); + ufbxi_extern_c __int64 _InterlockedDecrement64(__int64 volatile * lpAddend); + ufbxi_extern_c __int64 _InterlockedExchangeAdd64(__int64 volatile * lpAddend, __int64 Value); + typedef volatile __int64 ufbxi_atomic_counter; + #define ufbxi_atomic_counter_init(ptr) (*(ptr) = 0) + #define ufbxi_atomic_counter_free(ptr) (*(ptr) = 0) + #define ufbxi_atomic_counter_inc(ptr) ((size_t)_InterlockedIncrement64(ptr) - 1) + #define ufbxi_atomic_counter_dec(ptr) ((size_t)_InterlockedDecrement64(ptr) + 1) + #define ufbxi_atomic_counter_load(ptr) ((size_t)_InterlockedExchangeAdd64((ptr), 0)) + #else + ufbxi_extern_c long __cdecl _InterlockedIncrement(long volatile * lpAddend); + ufbxi_extern_c long __cdecl _InterlockedDecrement(long volatile * lpAddend); + ufbxi_extern_c long __cdecl _InterlockedExchangeAdd(long volatile * lpAddend, long Value); + typedef volatile long ufbxi_atomic_counter; + #define ufbxi_atomic_counter_init(ptr) (*(ptr) = 0) + #define ufbxi_atomic_counter_free(ptr) (*(ptr) = 0) + #define ufbxi_atomic_counter_inc(ptr) ((size_t)_InterlockedIncrement(ptr) - 1) + #define ufbxi_atomic_counter_dec(ptr) ((size_t)_InterlockedDecrement(ptr) + 1) + #define ufbxi_atomic_counter_load(ptr) ((size_t)_InterlockedExchangeAdd((ptr), 0)) + #endif +#elif !defined(UFBX_STANDARD_C) && defined(__TINYC__) + #if defined(__x86_64__) || defined(_AMD64_) + static size_t ufbxi_tcc_atomic_add(volatile size_t *dst, size_t value) { + __asm__ __volatile__("lock; xaddq %0, %1;" : "+r" (value), "=m" (*dst) : "m" (dst)); + return value; + } + #elif defined(__i386__) || defined(_X86_) + static size_t ufbxi_tcc_atomic_add(volatile size_t *dst, size_t value) { + __asm__ __volatile__("lock; xaddl %0, %1;" : "+r" (value), "=m" (*dst) : "m" (dst)); + return value; + } + #else + #error Unexpected TCC architecture + #endif + typedef volatile size_t ufbxi_atomic_counter; + #define ufbxi_atomic_counter_init(ptr) (*(ptr) = 0) + #define ufbxi_atomic_counter_free(ptr) (*(ptr) = 0) + #define ufbxi_atomic_counter_inc(ptr) ufbxi_tcc_atomic_add((ptr), 1) + #define ufbxi_atomic_counter_dec(ptr) ufbxi_tcc_atomic_add((ptr), SIZE_MAX) + #define ufbxi_atomic_counter_load(ptr) ufbxi_tcc_atomic_add((ptr), 0) +#elif defined(__cplusplus) && (__cplusplus >= 201103L) + #include + #include + typedef struct { alignas(std::atomic_size_t) char data[sizeof(std::atomic_size_t)]; } ufbxi_atomic_counter; + #define ufbxi_atomic_counter_init(ptr) (new (&(ptr)->data) std::atomic_size_t(0)) + #define ufbxi_atomic_counter_free(ptr) (((std::atomic_size_t*)(ptr)->data)->~atomic()) + #define ufbxi_atomic_counter_inc(ptr) ((std::atomic_size_t*)(ptr)->data)->fetch_add(1) + #define ufbxi_atomic_counter_dec(ptr) ((std::atomic_size_t*)(ptr)->data)->fetch_sub(1) + #define ufbxi_atomic_counter_load(ptr) ((std::atomic_size_t*)(ptr)->data)->load(std::memory_order_acquire) +#elif defined(__STDC_VERSION__) && (__STDC_VERSION__ >= 201112L) && !defined(__STDC_NO_ATOMICS__) + #include + typedef volatile atomic_size_t ufbxi_atomic_counter; + #define ufbxi_atomic_counter_init(ptr) atomic_init(ptr, 0) + #define ufbxi_atomic_counter_free(ptr) (void)(ptr) + #define ufbxi_atomic_counter_inc(ptr) atomic_fetch_add((ptr), 1) + #define ufbxi_atomic_counter_dec(ptr) atomic_fetch_sub((ptr), 1) + #define ufbxi_atomic_counter_load(ptr) atomic_load_explicit((ptr), memory_order_acquire) +#else + typedef volatile size_t ufbxi_atomic_counter; + #define ufbxi_atomic_counter_init(ptr) (*(ptr) = 0) + #define ufbxi_atomic_counter_free(ptr) (*(ptr) = 0) + #define ufbxi_atomic_counter_inc(ptr) ((*(ptr))++) + #define ufbxi_atomic_counter_dec(ptr) ((*(ptr))--) + #define ufbxi_atomic_counter_load(ptr) (*(ptr)) + #undef UFBXI_THREAD_SAFE + #define UFBXI_THREAD_SAFE 0 +#endif + +// ^^ No references to before this point ^^ +// vv No more includes past this point vv + +#if defined(UFBX_STRING_PREFIX) + #define ufbxi_string_fn(name) ufbxi_pre_cat(UFBX_STRING_PREFIX, name) + #define strlen ufbxi_string_fn(strlen) + #define memcpy ufbxi_string_fn(memcpy) + #define memmove ufbxi_string_fn(memmove) + #define memset ufbxi_string_fn(memset) + #define memchr ufbxi_string_fn(memchr) + #define memcmp ufbxi_string_fn(memcmp) + #define strcmp ufbxi_string_fn(strcmp) + #define strncmp ufbxi_string_fn(strncmp) +#endif + #if !defined(UFBX_LITTLE_ENDIAN) #if !defined(UFBX_STANDARD_C) && (defined(_M_IX86) || defined(__i386__) || defined(_M_X64) || defined(__x86_64__) || defined(_M_ARM64) || defined(__aarch64__) || defined(__wasm__) || defined(__EMSCRIPTEN__)) #define UFBX_LITTLE_ENDIAN 1 @@ -570,7 +830,7 @@ ufbx_static_assert(sizeof_f64, sizeof(double) == 8); // -- Version -#define UFBX_SOURCE_VERSION ufbx_pack_version(0, 14, 3) +#define UFBX_SOURCE_VERSION ufbx_pack_version(0, 15, 0) ufbx_abi_data_def const uint32_t ufbx_source_version = UFBX_SOURCE_VERSION; ufbx_static_assert(source_header_version, UFBX_SOURCE_VERSION/1000u == UFBX_HEADER_VERSION/1000u); @@ -607,98 +867,16 @@ ufbx_static_assert(source_header_version, UFBX_SOURCE_VERSION/1000u == UFBX_HEAD #define ufbxi_wrap_shr64(a, b) ((a) >> ((b) & 63)) #endif -// -- Atomic counter - -#define UFBXI_THREAD_SAFE 1 - -#if defined(__cplusplus) - #define ufbxi_extern_c extern "C" -#else - #define ufbxi_extern_c -#endif - -#if !defined(UFBX_STANDARD_C) && (defined(__GNUC__) || defined(__clang__) || defined(__INTEL_COMPILER)) - typedef size_t ufbxi_atomic_counter; - #define ufbxi_atomic_counter_init(ptr) (*(ptr) = 0) - #define ufbxi_atomic_counter_free(ptr) (*(ptr) = 0) - #define ufbxi_atomic_counter_inc(ptr) __sync_fetch_and_add((ptr), 1) - #define ufbxi_atomic_counter_dec(ptr) __sync_fetch_and_sub((ptr), 1) - #define ufbxi_atomic_counter_load(ptr) __sync_fetch_and_add((ptr), 0) // TODO: Proper atomic load -#elif !defined(UFBX_STANDARD_C) && defined(_MSC_VER) - #if defined(_M_X64) || defined(_M_ARM64) - ufbxi_extern_c __int64 _InterlockedIncrement64(__int64 volatile * lpAddend); - ufbxi_extern_c __int64 _InterlockedDecrement64(__int64 volatile * lpAddend); - ufbxi_extern_c __int64 _InterlockedExchangeAdd64(__int64 volatile * lpAddend, __int64 Value); - typedef volatile __int64 ufbxi_atomic_counter; - #define ufbxi_atomic_counter_init(ptr) (*(ptr) = 0) - #define ufbxi_atomic_counter_free(ptr) (*(ptr) = 0) - #define ufbxi_atomic_counter_inc(ptr) ((size_t)_InterlockedIncrement64(ptr) - 1) - #define ufbxi_atomic_counter_dec(ptr) ((size_t)_InterlockedDecrement64(ptr) + 1) - #define ufbxi_atomic_counter_load(ptr) ((size_t)_InterlockedExchangeAdd64((ptr), 0)) - #else - ufbxi_extern_c long __cdecl _InterlockedIncrement(long volatile * lpAddend); - ufbxi_extern_c long __cdecl _InterlockedDecrement(long volatile * lpAddend); - ufbxi_extern_c long __cdecl _InterlockedExchangeAdd(long volatile * lpAddend, long Value); - typedef volatile long ufbxi_atomic_counter; - #define ufbxi_atomic_counter_init(ptr) (*(ptr) = 0) - #define ufbxi_atomic_counter_free(ptr) (*(ptr) = 0) - #define ufbxi_atomic_counter_inc(ptr) ((size_t)_InterlockedIncrement(ptr) - 1) - #define ufbxi_atomic_counter_dec(ptr) ((size_t)_InterlockedDecrement(ptr) + 1) - #define ufbxi_atomic_counter_load(ptr) ((size_t)_InterlockedExchangeAdd((ptr), 0)) - #endif -#elif !defined(UFBX_STANDARD_C) && defined(__TINYC__) - #if defined(__x86_64__) || defined(_AMD64_) - static size_t ufbxi_tcc_atomic_add(volatile size_t *dst, size_t value) { - __asm__ __volatile__("lock; xaddq %0, %1;" : "+r" (value), "=m" (*dst) : "m" (dst)); - return value; - } - #elif defined(__i386__) || defined(_X86_) - static size_t ufbxi_tcc_atomic_add(volatile size_t *dst, size_t value) { - __asm__ __volatile__("lock; xaddl %0, %1;" : "+r" (value), "=m" (*dst) : "m" (dst)); - return value; - } - #else - #error Unexpected TCC architecture - #endif - typedef volatile size_t ufbxi_atomic_counter; - #define ufbxi_atomic_counter_init(ptr) (*(ptr) = 0) - #define ufbxi_atomic_counter_free(ptr) (*(ptr) = 0) - #define ufbxi_atomic_counter_inc(ptr) ufbxi_tcc_atomic_add((ptr), 1) - #define ufbxi_atomic_counter_dec(ptr) ufbxi_tcc_atomic_add((ptr), SIZE_MAX) - #define ufbxi_atomic_counter_load(ptr) ufbxi_tcc_atomic_add((ptr), 0) -#elif defined(__cplusplus) && (__cplusplus >= 201103L) - #include - #include - typedef struct { alignas(std::atomic_size_t) char data[sizeof(std::atomic_size_t)]; } ufbxi_atomic_counter; - #define ufbxi_atomic_counter_init(ptr) (new (&(ptr)->data) std::atomic_size_t(0)) - #define ufbxi_atomic_counter_free(ptr) (((std::atomic_size_t*)(ptr)->data)->~atomic()) - #define ufbxi_atomic_counter_inc(ptr) ((std::atomic_size_t*)(ptr)->data)->fetch_add(1) - #define ufbxi_atomic_counter_dec(ptr) ((std::atomic_size_t*)(ptr)->data)->fetch_sub(1) - #define ufbxi_atomic_counter_load(ptr) ((std::atomic_size_t*)(ptr)->data)->load(std::memory_order_acquire) -#elif defined(__STDC_VERSION__) && (__STDC_VERSION__ >= 201112L) && !defined(__STDC_NO_ATOMICS__) - #include - typedef volatile atomic_size_t ufbxi_atomic_counter; - #define ufbxi_atomic_counter_init(ptr) atomic_init(ptr, 0) - #define ufbxi_atomic_counter_free(ptr) (void)(ptr) - #define ufbxi_atomic_counter_inc(ptr) atomic_fetch_add((ptr), 1) - #define ufbxi_atomic_counter_dec(ptr) atomic_fetch_sub((ptr), 1) - #define ufbxi_atomic_counter_load(ptr) atomic_load_explicit((ptr), memory_order_acquire) -#else - typedef volatile size_t ufbxi_atomic_counter; - #define ufbxi_atomic_counter_init(ptr) (*(ptr) = 0) - #define ufbxi_atomic_counter_free(ptr) (*(ptr) = 0) - #define ufbxi_atomic_counter_inc(ptr) ((*(ptr))++) - #define ufbxi_atomic_counter_dec(ptr) ((*(ptr))--) - #define ufbxi_atomic_counter_load(ptr) (*(ptr)) - #undef UFBXI_THREAD_SAFE - #define UFBXI_THREAD_SAFE 0 -#endif - // -- Bit manipulation #if !defined(UFBX_STANDARD_C) && defined(_MSC_VER) && (defined(_M_X64) || defined(_M_IX86)) ufbxi_extern_c unsigned char _BitScanReverse(unsigned long * _Index, unsigned long _Mask); ufbxi_extern_c unsigned char _BitScanReverse64(unsigned long * _Index, unsigned __int64 _Mask); + static ufbxi_forceinline ufbxi_unused uint32_t ufbxi_lzcnt32(uint32_t v) { + unsigned long index; + _BitScanReverse(&index, (unsigned long)v); + return 31 - (uint32_t)index; + } static ufbxi_forceinline ufbxi_unused uint32_t ufbxi_lzcnt64(uint64_t v) { unsigned long index; #if defined(_M_X64) @@ -713,14 +891,26 @@ ufbx_static_assert(source_header_version, UFBX_SOURCE_VERSION/1000u == UFBX_HEAD return 63 - (uint32_t)index; } #elif !defined(UFBX_STANDARD_C) && (defined(__GNUC__) || defined(__clang__)) + #define ufbxi_lzcnt32(v) ((uint32_t)__builtin_clz((unsigned)(v))) #define ufbxi_lzcnt64(v) ((uint32_t)__builtin_clzll((unsigned long long)(v))) #else // DeBrujin table lookup - static const uint8_t ufbxi_lzcnt_table[] = { + static const uint8_t ufbxi_lzcnt32_table[] = { + 31, 22, 30, 21, 18, 10, 29, 2, 20, 17, 15, 13, 9, 6, 28, 1, 23, 19, 11, 3, 16, 14, 7, 24, 12, 4, 8, 25, 5, 26, 27, 0, + }; + static const uint8_t ufbxi_lzcnt64_table[] = { 63, 16, 62, 7, 15, 36, 61, 3, 6, 14, 22, 26, 35, 47, 60, 2, 9, 5, 28, 11, 13, 21, 42, 19, 25, 31, 34, 40, 46, 52, 59, 1, 17, 8, 37, 4, 23, 27, 48, 10, 29, 12, 43, 20, 32, 41, 53, 18, 38, 24, 49, 30, 44, 33, 54, 39, 50, 45, 55, 51, 56, 57, 58, 0, }; + static ufbxi_noinline ufbxi_unused uint32_t ufbxi_lzcnt32(uint32_t v) { + v |= v >> 1; + v |= v >> 2; + v |= v >> 4; + v |= v >> 8; + v |= v >> 16; + return ufbxi_lzcnt32_table[(v * 0x07c4acddu) >> 27]; + } static ufbxi_noinline ufbxi_unused uint32_t ufbxi_lzcnt64(uint64_t v) { v |= v >> 1; v |= v >> 2; @@ -728,10 +918,20 @@ ufbx_static_assert(source_header_version, UFBX_SOURCE_VERSION/1000u == UFBX_HEAD v |= v >> 8; v |= v >> 16; v |= v >> 32; - return ufbxi_lzcnt_table[(v * UINT64_C(0x03f79d71b4cb0a89)) >> 58]; + return ufbxi_lzcnt64_table[(v * UINT64_C(0x03f79d71b4cb0a89)) >> 58]; } #endif +// -- Bit conversion + +#if defined(__cplusplus) + #define ufbxi_bit_cast(m_dst_type, m_dst, m_src_type, m_src) memcpy(&(m_dst), &(m_src), sizeof(m_dst_type)) +#else + #define ufbxi_bit_cast(m_dst_type, m_dst, m_src_type, m_src) do { \ + union { m_dst_type mi_dst; m_src_type mi_src; } mi_union; \ + mi_union.mi_src = (m_src); (m_dst) = mi_union.mi_dst; } while (0) +#endif + // -- Debug #if defined(UFBX_DEBUG_BINARY_SEARCH) || defined(UFBX_REGRESSION) @@ -1081,321 +1281,389 @@ static ufbxi_noinline void ufbxi_unstable_sort(void *in_data, size_t size, size_ } // -- Float parsing -// -// Custom float parsing that handles floats up to (-)ddddddddddddddddddd.ddddddddddddddddddd -// If larger or scientific notation is used then it defers to `strtod()`. -// For the algorithm we need 128-bit division that is either provided by hardware on x64 or -// a custom implementation below. - -#if !defined(UFBX_STANDARD_C) && UFBXI_MSC_VER >= 1920 && defined(_M_X64) && !defined(__clang__) - ufbxi_extern_c extern unsigned __int64 __cdecl _udiv128(unsigned __int64 highdividend, - unsigned __int64 lowdividend, unsigned __int64 divisor, unsigned __int64 *remainder); - #define ufbxi_div128(a_hi, a_lo, b, p_rem) (_udiv128((a_hi), (a_lo), (b), (p_rem))) -#elif !defined(UFBX_STANDARD_C) && (defined(__GNUC__) || defined(__clang__)) && (defined(__x86_64__) || defined(_M_X64)) - static ufbxi_forceinline uint64_t ufbxi_div128(uint64_t a_hi, uint64_t a_lo, uint64_t b, uint64_t *p_rem) { - uint64_t quot, rem; - __asm__("divq %[v]" : "=a"(quot), "=d"(rem) : [v] "r"(b), "a"(a_lo), "d"(a_hi)); - *p_rem = rem; - return quot; - } -#else - static ufbxi_forceinline uint64_t ufbxi_div128(uint64_t a_hi, uint64_t a_lo, uint64_t b, uint64_t *p_rem) { - // Divide `(a_hi << 64 | a_lo)` by `b`, returns quotinent and stores reminder in `p_rem`. - // Based on TAOCP 2.4 multi-word division single algorithm digit step. - // - // Notation: - // b is the base (2^32) in this case - // aN is the Nth digit (base b) of a from the least significant digit - // { x y z } is a multi-digit number b^2*x + b*y + z - // ie. for a 64-bit number a = { a1 a0 } = b*a1 + a0 - // - // We do the division in two steps by dividing three digits in each iteration: - // - // q1, r = { a3 a2 a1 } / { b1 b0 } - // q0, r = { r1 r0 a0 } / { b1 b0 } - // - // In each step we want to compute the expression: - // - // q, r = { u2 u1 u0 } / { v1 v0 } - // - // However we cannot rely on being able to do `u96 / u64` division we estimate - // the result by considering only the leading digits: - // - // q^ = { u2 u1 } / v1 [A] - // r^ = { u2 u1 } % v1 = { u2 u1 } - v1 * q^ [B] - // - // As long as `v1 >= b/2` the estimate `q^` is at most two larger than the actual `q` - // (proof in TAOCP 2.4) so we can compute the correction amount `c`: - // - // q <= q^ <= q + 2 - // q = q^ - c [C] - // - // We can compute the final remainder (that must be non-negative) as follows: - // - // r = { u2 u1 u0 } - v*q - // r = { u2 u1 u0 } - v*(q^ - c) - // r = { u2 u1 u0 } - v*q^ + v*c - // r = { u2 u1 u0 } - { v1 v0 } * q^ + v*c - // r = b^2*u2 + b*u1 + u0 - b*v1*q^ - v0*q^ + v*c - // r = b*(b*u2 + u1 - v1*q^) + u0 - v0*q^ + v*c - // r = b*({ u2 u1 } - v1*q^) + u0 - v0*q^ + v*c - // r = b*r^ + u0 - v0*q^ + v*c - // r = { r^ u0 } - v0*q^ + v*c [D] - // - // As we know `0 <= c <= 2` we can first check if `r < 0` requiring `c >= 1`: - // - // { r^ u0 } - v0*q^ < 0 - // { r^ u0 } < v0*q^ [E] - // - // If we know that `r < 0` we can check if `r < -v` requiring `c = 2`: - // - // { r^ u0 } - v0*q^ < -v - // v0*q^ - { r^ u0 } > v [F] - // - - // First we need to make sure `v1 >= b/2`, we can do this by multiplying the whole - // expression by `2^shift` so that the high bit of `v` is set. - uint32_t shift = ufbxi_lzcnt64(b); - a_hi = (a_hi << shift) | (shift ? a_lo >> (64 - shift) : 0); - a_lo <<= shift; - b <<= shift; - - uint64_t v = b; - uint32_t v1 = (uint32_t)(v >> 32); - uint32_t v0 = (uint32_t)(v); - uint64_t q1, q0, r; - - // q1, r = { a3 a2 a1 } / { b1 b0 } - { - uint64_t u2_u1 = a_hi; - uint32_t u0 = (uint32_t)(a_lo >> 32u); - uint64_t qh = u2_u1 / v1; // q^ = { u2 u1 } / v1 [A] - uint64_t rh = u2_u1 % v1; // r^ = { u2 u1 } % v1 [B] - uint64_t rh_u0 = rh << 32u | u0; // { r^ u0 } - uint64_t v0qh = v0 * qh; // v0*q^ - uint32_t c = rh_u0 < v0qh ? 1 : 0; // { r^ u0 } < v0*q^ [E] - c += c & (v0qh - rh_u0 > v ? 1 : 0); // v0*q^ - { r^ u0 } > v [F] - q1 = qh - c; // q1 = q^ - c [C] - r = rh_u0 - v0qh + v*c; // r = { r^ u0 } - v0*q^ + v*c [D] - } +#define UFBXI_BIGINT_LIMB_BITS 32 +#define UFBXI_BIGINT_ACCUM_BITS (UFBXI_BIGINT_LIMB_BITS * 2) +#define UFBXI_BIGINT_LIMB_MAX (ufbxi_bigint_limb)(((ufbxi_bigint_accum)1 << UFBXI_BIGINT_LIMB_BITS) - 1) +typedef uint32_t ufbxi_bigint_limb; +typedef uint64_t ufbxi_bigint_accum; - // q0, r = { r1 r0 a0 } / { b1 b0 } - { - uint64_t u2_u1 = r; - uint32_t u0 = (uint32_t)a_lo; - - uint64_t qh = u2_u1 / v1; // q^ = { u2 u1 } / v1 [A] - uint64_t rh = u2_u1 % v1; // r^ = { u2 u1 } % v1 [B] - uint64_t rh_u0 = rh << 32u | u0; // { r^ u0 } - uint64_t v0qh = v0 * qh; // v0*q^ - uint32_t c = rh_u0 < v0qh ? 1 : 0; // { r^ u0 } < v0*q^ [E] - c += c & (v0qh - rh_u0 > v ? 1 : 0); // v0*q^ - { r^ u0 } > v [F] - q0 = qh - c; // q0 = q^ - c [C] - r = rh_u0 - v0qh + v*c; // r = { r^ u0 } - v0*q^ + v*c [D] - } +typedef struct { + ufbxi_bigint_limb *limbs; + uint32_t capacity; + uint32_t length; +} ufbxi_bigint; - // Un-normalize the remainder and return the quotinent - *p_rem = r >> shift; - return q1 << 32u | q0; - } -#endif +static ufbxi_bigint ufbxi_bigint_make(ufbxi_bigint_limb *limbs, size_t capacity) +{ + ufbxi_bigint bi = { limbs, (uint32_t)capacity }; + return bi; +} -typedef enum { - UFBXI_PARSE_DOUBLE_ALLOW_FAST_PATH = 0x1, - UFBXI_PARSE_DOUBLE_VERIFY_LENGTH = 0x2, -} ufbxi_parse_double_flag; +#define ufbxi_bigint_array(arr) ufbxi_bigint_make((arr), sizeof(arr) / sizeof(*(arr))) static const uint64_t ufbxi_pow5_tab[] = { - UINT64_C(0x8000000000000000), // 5^0 * 2^63 - UINT64_C(0xa000000000000000), // 5^1 * 2^61 - UINT64_C(0xc800000000000000), // 5^2 * 2^59 - UINT64_C(0xfa00000000000000), // 5^3 * 2^57 - UINT64_C(0x9c40000000000000), // 5^4 * 2^54 - UINT64_C(0xc350000000000000), // 5^5 * 2^52 - UINT64_C(0xf424000000000000), // 5^6 * 2^50 - UINT64_C(0x9896800000000000), // 5^7 * 2^47 - UINT64_C(0xbebc200000000000), // 5^8 * 2^45 - UINT64_C(0xee6b280000000000), // 5^9 * 2^43 - UINT64_C(0x9502f90000000000), // 5^10 * 2^40 - UINT64_C(0xba43b74000000000), // 5^11 * 2^38 - UINT64_C(0xe8d4a51000000000), // 5^12 * 2^36 - UINT64_C(0x9184e72a00000000), // 5^13 * 2^33 - UINT64_C(0xb5e620f480000000), // 5^14 * 2^31 - UINT64_C(0xe35fa931a0000000), // 5^15 * 2^29 - UINT64_C(0x8e1bc9bf04000000), // 5^16 * 2^26 - UINT64_C(0xb1a2bc2ec5000000), // 5^17 * 2^24 - UINT64_C(0xde0b6b3a76400000), // 5^18 * 2^22 - UINT64_C(0x8ac7230489e80000), // 5^19 * 2^19 - UINT64_C(0xad78ebc5ac620000), // 5^20 * 2^17 - UINT64_C(0xd8d726b7177a8000), // 5^21 * 2^15 - UINT64_C(0x878678326eac9000), // 5^22 * 2^12 - UINT64_C(0xa968163f0a57b400), // 5^23 * 2^10 - UINT64_C(0xd3c21bcecceda100), // 5^24 * 2^8 - UINT64_C(0x84595161401484a0), // 5^25 * 2^5 - UINT64_C(0xa56fa5b99019a5c8), // 5^26 * 2^3 - UINT64_C(0xcecb8f27f4200f3a), // 5^27 * 2^1 -}; -static const int8_t ufbxi_pow2_tab[] = { - 62, 59, 56, 53, 49, 46, 43, 39, 36, 33, 29, 26, 23, 19, 16, 13, 9, 6, 3, -1, -4, -7, -11, -14, -17, -21, -24, -27, + UINT64_C(0x1), UINT64_C(0x5), UINT64_C(0x19), UINT64_C(0x7d), UINT64_C(0x271), UINT64_C(0xc35), UINT64_C(0x3d09), UINT64_C(0x1312d), UINT64_C(0x5f5e1), + UINT64_C(0x1dcd65), UINT64_C(0x9502f9), UINT64_C(0x2e90edd), UINT64_C(0xe8d4a51), UINT64_C(0x48c27395), UINT64_C(0x16bcc41e9), UINT64_C(0x71afd498d), + UINT64_C(0x2386f26fc1), UINT64_C(0xb1a2bc2ec5), UINT64_C(0x3782dace9d9), UINT64_C(0x1158e460913d), UINT64_C(0x56bc75e2d631), UINT64_C(0x1b1ae4d6e2ef5), + UINT64_C(0x878678326eac9), UINT64_C(0x2a5a058fc295ed), UINT64_C(0xd3c21bcecceda1), UINT64_C(0x422ca8b0a00a425), UINT64_C(0x14adf4b7320334b9), UINT64_C(0x6765c793fa10079d), }; + static const double ufbxi_pow10_tab_f64[] = { 1e0, 1e1, 1e2, 1e3, 1e4, 1e5, 1e6, 1e7, 1e8, 1e9, 1e10, 1e11, 1e12, 1e13, 1e14, 1e15, 1e16, 1e17, 1e18, 1e19, 1e20, 1e21, 1e22, }; -static ufbxi_noinline uint32_t ufbxi_parse_double_init_flags() +static ufbxi_noinline void ufbxi_bigint_mad(ufbxi_bigint *bigint, ufbxi_bigint_accum multiplicand, ufbxi_bigint_accum addend) +{ + ufbxi_dev_assert((multiplicand | addend) >> (UFBXI_BIGINT_ACCUM_BITS - 1) == 0); + ufbxi_bigint b = *bigint; + ufbxi_bigint_limb m_lo = (ufbxi_bigint_limb)multiplicand; + ufbxi_bigint_limb m_hi = (ufbxi_bigint_limb)(multiplicand >> UFBXI_BIGINT_LIMB_BITS); + ufbxi_bigint_accum carry = addend; + for (uint32_t i = 0; i < b.length; i++) { + ufbxi_bigint_accum limb = (ufbxi_bigint_accum)b.limbs[i]; + ufbxi_bigint_accum lo = limb * m_lo + (carry & UFBXI_BIGINT_LIMB_MAX); + ufbxi_bigint_accum hi = limb * m_hi; + b.limbs[i] = (ufbxi_bigint_limb)lo; + carry = (carry >> 32u) + (lo >> 32u) + hi; + } + while (carry) { + b.limbs[b.length++] = (ufbxi_bigint_limb)carry; + ufbxi_dev_assert(b.length < b.capacity); + carry >>= 32u; + } + bigint->length = b.length; +} + +static ufbxi_noinline bool ufbxi_bigint_div(ufbxi_bigint *q, ufbxi_bigint *u, ufbxi_bigint *v) +{ + int32_t n = (int32_t)v->length; + int32_t m = (int32_t)u->length - n; + ufbxi_bigint_limb v_hi = v->limbs[v->length - 1]; + ufbxi_bigint_limb *un = u->limbs, *vn = v->limbs; + ufbxi_dev_assert(n >= 2 && m >= 1 && v_hi >> (UFBXI_BIGINT_LIMB_BITS - 1) != 0 && un[n+m - 1] >> (UFBXI_BIGINT_LIMB_BITS - 1) == 0); + un[n + m] = 0; + q->length = 0; + for (int32_t j = m - 1; j >= 0; j--) { + ufbxi_bigint_accum u_hi = ((ufbxi_bigint_accum)un[n+j] << UFBXI_BIGINT_LIMB_BITS) | un[n+j-1]; + ufbxi_bigint_accum t, qhat = u_hi / v_hi, rhat = u_hi % v_hi; + while (qhat >> UFBXI_BIGINT_LIMB_BITS != 0 || qhat*vn[n-2] > ((rhat<> UFBXI_BIGINT_LIMB_BITS != 0) break; + } + ufbxi_bigint_limb carry = 0; + for (int32_t i = 0; i < n; i++) { + ufbxi_bigint_accum p = qhat * vn[i]; + t = (ufbxi_bigint_accum)un[i+j] - carry - (ufbxi_bigint_limb)p; + un[i+j] = (ufbxi_bigint_limb)t; + carry = (ufbxi_bigint_limb)((p >> UFBXI_BIGINT_LIMB_BITS) - (t >> UFBXI_BIGINT_LIMB_BITS)); + } + t = (ufbxi_bigint_accum)un[j+n] - carry; + un[j+n] = (ufbxi_bigint_limb)t; + if (t >> UFBXI_BIGINT_LIMB_BITS != 0) { + qhat -= 1; + carry = 0; + for (int32_t i = 0; i < n; i++) { + t = (ufbxi_bigint_accum)un[i+j] + vn[i] + carry; + un[i+j] = (ufbxi_bigint_limb)t; + carry = (ufbxi_bigint_limb)(t >> UFBXI_BIGINT_LIMB_BITS); + } + un[j+n] += carry; + } + q->limbs[j] = (ufbxi_bigint_limb)qhat; + if (qhat && !q->length) { + ufbxi_dev_assert(j + 1 < (int32_t)q->capacity); + q->length = (uint32_t)(j + 1); + } + } + for (int32_t i = 0; i < n; i++) { + if (un[i]) return true; + } + return false; +} + +static void ufbxi_bigint_mul_pow5(ufbxi_bigint *b, uint32_t power) { - // We require evaluation in double precision, either for doubles (0) or always (1) - // and rounding to nearest, which we can check for with `1 + eps == 1 - eps`. - #if defined(FLT_EVAL_METHOD) - #if FLT_EVAL_METHOD == 0 || FLT_EVAL_METHOD == 1 - static volatile double ufbxi_volatile_eps = 2.2250738585072014e-308; - if (1.0 + ufbxi_volatile_eps == 1.0 - ufbxi_volatile_eps) return UFBXI_PARSE_DOUBLE_ALLOW_FAST_PATH; - #endif - #endif + for (; power > 27; power -= 27) { + ufbxi_bigint_mad(b, ufbxi_pow5_tab[27], 0); + } + ufbxi_bigint_mad(b, ufbxi_pow5_tab[power], 0); +} - return 0; +static ufbxi_noinline void ufbxi_bigint_shift_left(ufbxi_bigint *bigint, uint32_t amount) +{ + uint32_t words = amount / UFBXI_BIGINT_LIMB_BITS, bits = amount % UFBXI_BIGINT_LIMB_BITS; + ufbxi_bigint b = *bigint; + ufbxi_dev_assert(b.length + words + 1 < b.capacity && b.capacity >= 4); + uint32_t bits_down = UFBXI_BIGINT_LIMB_BITS - bits - 1; + bigint->length += words + (b.limbs[b.length - 1] >> 1 >> bits_down != 0 ? 1 : 0); + b.limbs[b.length] = 0; + if (b.length <= 3 && words <= 3) { + ufbxi_bigint_limb l0 = ufbxi_maybe_uninit(b.length >= 0, b.limbs[0], ~0u); + ufbxi_bigint_limb l1 = ufbxi_maybe_uninit(b.length >= 1, b.limbs[1], ~0u); + ufbxi_bigint_limb l2 = ufbxi_maybe_uninit(b.length >= 2, b.limbs[2], ~0u); + b.limbs[0] = 0; + b.limbs[1] = 0; + b.limbs[2] = 0; + b.limbs[words + 0] = l0 << bits; + b.limbs[words + 1] = (l1 << bits) | (l0 >> 1 >> bits_down); + b.limbs[words + 2] = (l2 << bits) | (l1 >> 1 >> bits_down); + b.limbs[words + 3] = (l2 >> 1 >> bits_down); + } else { + for (uint32_t i = b.length + 1; i-- > 1; ) { + b.limbs[i + words] = (b.limbs[i] << bits) | (b.limbs[i - 1] >> 1 >> bits_down); + } + b.limbs[words] = b.limbs[0] << bits; + for (uint32_t i = 0; i < words; i++) { + b.limbs[i] = 0; + } + } +} + +static ufbxi_bigint_limb ufbxi_bigint_top_limb(const ufbxi_bigint b, uint32_t index) { + return index < b.length ? b.limbs[b.length - 1 - index] : 0; +} + +static ufbxi_noinline uint64_t ufbxi_bigint_extract_high(const ufbxi_bigint b, int32_t *p_exponent, bool *p_tail) +{ + ufbxi_dev_assert(b.length != 0); + uint64_t result = 0; + const uint32_t limb_count = 64 / UFBXI_BIGINT_LIMB_BITS; + for (uint32_t i = 0; i < limb_count; i++) { + result = (result << UFBXI_BIGINT_LIMB_BITS) | ufbxi_bigint_top_limb(b, i); + } + uint32_t shift = ufbxi_lzcnt64(result); + result <<= shift; + ufbxi_bigint_limb lo = ufbxi_bigint_top_limb(b, limb_count); + if (shift > 0) { + result |= lo >> (UFBXI_BIGINT_LIMB_BITS - shift); + } + *p_tail |= (ufbxi_bigint_limb)(lo << shift) != 0; + for (uint32_t i = limb_count + 1; i < b.length; i++) { + *p_tail |= ufbxi_bigint_top_limb(b, i) != 0; + } + *p_exponent += (int32_t)(b.length * UFBXI_BIGINT_LIMB_BITS - shift - 1); + return result; } -static ufbxi_noinline double ufbxi_parse_double_slow(const char *str, char **end) +static uint64_t ufbxi_shift_right_round(uint64_t value, uint32_t shift, bool tail) { - // TODO: Locales - return strtod(str, end); + if (shift == 0) return value; + if (shift > 64) return 0; + uint64_t result = value >> (shift - 1); + uint64_t tail_mask = (UINT64_C(1) << (shift - 1)) - 1; + + bool r_odd = (result & 0x2) != 0; + bool r_round = (result & 0x1) != 0; + bool r_tail = tail || (value & tail_mask) != 0; + uint64_t round_bit = (r_round && (r_odd || r_tail)) ? 1u : 0u; + + return (result >> 1u) + round_bit; } +typedef enum { + UFBXI_PARSE_DOUBLE_ALLOW_FAST_PATH = 0x1, + UFBXI_PARSE_DOUBLE_VERIFY_LENGTH = 0x2, + UFBXI_PARSE_DOUBLE_AS_BINARY32 = 0x4, +} ufbxi_parse_double_flag; + static ufbxi_noinline double ufbxi_parse_double(const char *str, size_t max_length, char **end, uint32_t flags) { - // TODO: Use this for optimizing digit parsing - (void)max_length; + const uint32_t max_limbs = 14; - uint64_t integer = 0; - uint32_t n_integer = 0; - int32_t n_decimals = 0; - uint32_t n_exp = 0; - bool negative = false; + ufbxi_bigint_limb mantissa_limbs[42], divisor_limbs[42], quotient_limbs[42]; + ufbxi_bigint big_mantissa = ufbxi_bigint_array(mantissa_limbs); + ufbxi_bigint big_quotient = ufbxi_bigint_array(quotient_limbs); + int32_t dec_exponent = 0, has_dot = 0; + bool negative = false, tail = false, digits_valid = true; + uint64_t digits = 0; + uint32_t num_digits = 0; - // Parse /[+-]?[0-9]*(\.[0-9]*)([eE][+-]?[0-9]*)?/ retaining all digits - // in `integer` and number of decimals in `n_decimals`, exponent simply - // modifies `n_decimals` accordingly. const char *p = str; - if (*p == '-') { - negative = true; - p++; - } else if (*p == '+') { - p++; - } - while (((uint32_t)*p - '0') < 10) { - integer = integer * 10 + (uint64_t)(*p++ - '0'); - n_integer++; + if (*p == '+' || *p == '-') { + negative = *p++ == '-'; } - if (*p == '.') { - p++; - while (((uint32_t)*p - '0') < 10) { - integer = integer * 10 + (uint64_t)(*p++ - '0'); - n_integer++; - n_decimals++; + for (;;) { + char c = *p++; + if (c >= '0' && c <= '9') { + if (big_mantissa.length < max_limbs) { + digits = digits * 10 + (uint64_t)(c - '0'); + num_digits++; + if (num_digits >= 18) { + ufbxi_bigint_mad(&big_mantissa, ufbxi_pow5_tab[num_digits] << num_digits, digits); + digits = 0; + num_digits = 0; + digits_valid = false; + } + dec_exponent -= has_dot; + } else { + dec_exponent += 1 - has_dot; + } + } else if (c == '.' && !has_dot) { + has_dot = true; + } else { + break; } } - if ((*p | 0x20) == 'e') { + p--; + if (*p == 'e' || *p == 'E') { p++; - int32_t exp = 0; - int32_t exp_sign = -1; - if (*p == '-') { - p++; - exp_sign = 1; - } else if (*p == '+') { + bool exp_negative = false; + if (*p == '+' || *p == '-') { + exp_negative = *p == '-'; p++; } - while (((uint32_t)*p - '0') < 10) { - exp = exp * 10 + (int32_t)(*p++ - '0'); - n_exp++; + int32_t exp = 0; + for (;;) { + char c = *p; + if (c >= '0' && c <= '9') { + p++; + exp = exp * 10 + (c - '0'); + if (exp >= 10000) break; + } else { + break; + } } - n_decimals += exp * exp_sign; + dec_exponent += exp_negative ? -exp : exp; } - *end = (char*)p; + *end = (char*)p; // Check that the number is not potentially truncated. if (ufbxi_to_size(p - str) >= max_length && (flags & UFBXI_PARSE_DOUBLE_VERIFY_LENGTH) != 0) { *end = NULL; return 0.0; } - // Overflowed either 64-bit `integer` or 31-bit `exp`. - if (n_integer > 19 || n_exp > 9 || (integer >> 63) != 0) { - return ufbxi_parse_double_slow(str, end); - } - // Both power of 10 and integer are exactly representable as doubles // Powers of 10 are factored as 2*5, and 2^N can be always exactly represented. - if ((flags & UFBXI_PARSE_DOUBLE_ALLOW_FAST_PATH) != 0 && n_decimals >= -22 && n_decimals <= 22 && (integer >> 53) == 0) { + if ((flags & UFBXI_PARSE_DOUBLE_ALLOW_FAST_PATH) != 0 && big_mantissa.length == 0 && dec_exponent >= -22 && dec_exponent <= 22 && (digits >> 53) == 0) { double value; - if (n_decimals > 0) { - value = (double)integer / ufbxi_pow10_tab_f64[n_decimals]; + if (dec_exponent < 0) { + value = (double)digits / ufbxi_pow10_tab_f64[-dec_exponent]; } else { - value = (double)integer * ufbxi_pow10_tab_f64[-n_decimals]; + value = (double)digits * ufbxi_pow10_tab_f64[dec_exponent]; } return negative ? -value : value; } - // Cannot handle positive exponents here, fortunately the fast case should - // take care of most of them, for negative exponents we can only handle - // up to e-27 as `5^28 > 2^64` and cannot be used as a divisor below. - if (n_decimals < 0) { - return ufbxi_parse_double_slow(str, end); - } else if (!n_decimals || !integer) { - double value = (double)integer; - return negative ? -value : value; - } else if (n_decimals > 27) { - return ufbxi_parse_double_slow(str, end); - } - - // We want to compute `integer / 10^N` precisely, we can do this - // using 128-bit division `2^64 * dividend / divisor`: - // dividend = integer * 2^S (S set such that highest bit is 62) - // divisor = 10^N * 2^T (T set such that highest bit is 63) - // We have to compensate for the shifts in the exponent: - // (2^64 * integer * 2^S) / (10^N * 2^T) * 2^(-1 - S + T) - // To get larger exponent range split 10^N to 5^N * 2^N and move 2^N to the exponent - // (2^64 * integer * 2^S) / (5^N * 2^T) * 2^(-1 - S + T - N) - uint32_t shift = ufbxi_lzcnt64(integer) - 1; - uint64_t dividend = integer << shift; - uint64_t divisor = ufbxi_pow5_tab[n_decimals]; - int32_t exponent = (int32_t)ufbxi_pow2_tab[n_decimals] - (int32_t)shift; // (-1 + T - N) - S - uint64_t rem_hi; - uint64_t b_hi = ufbxi_div128(dividend, 0, divisor, &rem_hi); - - // Align the mantissa so that high bit is set, due to the shifting of the - // divisor and dividend the smallest result is `2^62 + N`, so we need to - // shift at most by one bit. - uint64_t b_bit = 1 - (b_hi >> 63u); - uint64_t mantissa = b_hi << b_bit; - exponent -= (int32_t)b_bit; - - // Round to 53 bits, accounting for potential remainder. - bool nonzero_tail = rem_hi != 0; - bool r_odd = mantissa & (1 << 11u); - bool r_round = mantissa & (1 << 10u); - bool r_tail = (mantissa & ((1 << 10u) - 1)) != 0 || nonzero_tail; - uint64_t round = (r_round && (r_odd || r_tail)) ? 1u : 0u; - - // Assemble the IEEE 754 binary64 number. - uint64_t bits - = (uint64_t)negative << 63u - | (uint64_t)(exponent + 1023) << 52u - | ((mantissa >> 11u) & ~(UINT64_C(1) << 52u)); - bits += round; - - // Type punning via unions is safe in C but in C++ the only safe way - // (pre std::bit_cast) is to use `memcpy()` and hope it gets optimized out. -#if defined(__cplusplus) - double result; - memcpy(&result, &bits, 8); - return result; -#else - union { uint64_t u; double d; } u_to_d; - u_to_d.u = bits; - return u_to_d.d; -#endif + if (big_mantissa.length == 0) { + big_mantissa.limbs[0] = (ufbxi_bigint_limb)digits; + big_mantissa.limbs[1] = (ufbxi_bigint_limb)(digits >> 32u); + big_mantissa.length = (digits >> 32u) ? 2 : digits ? 1 : 0; + if (big_mantissa.length == 0) return negative ? -0.0 : 0.0; + } else { + ufbxi_bigint_mad(&big_mantissa, ufbxi_pow5_tab[num_digits] << num_digits, digits); + } + + uint32_t enc_sign_shift = 63; + uint32_t enc_mantissa_bits = 53; + int32_t enc_max_exponent = 1023; + if (flags & UFBXI_PARSE_DOUBLE_AS_BINARY32) { + enc_sign_shift = 31; + enc_mantissa_bits = 24; + enc_max_exponent = 127; + } + + int32_t exponent = 0; + if (dec_exponent < 0) { + if (dec_exponent + (int32_t)big_mantissa.length * 10 <= -325) return negative ? -0.0 : 0.0; + + ufbxi_bigint big_divisor = ufbxi_bigint_array(divisor_limbs); + uint32_t pow5 = (uint32_t)-dec_exponent; + uint32_t initial_pow5 = pow5 <= 27 ? pow5 : 27; + uint64_t pow5_value = ufbxi_pow5_tab[initial_pow5]; + pow5 -= initial_pow5; + exponent += dec_exponent; + + if (pow5 == 0 && digits_valid && digits >> 63 == 0) { + uint32_t divisor_zeros = ufbxi_lzcnt64(pow5_value); + uint64_t mantissa_zeros = ufbxi_lzcnt64(digits) - 1; + uint64_t divisor_bits = pow5_value << divisor_zeros; + uint64_t mantissa_bits = digits << mantissa_zeros; + big_divisor.limbs[0] = (ufbxi_bigint_limb)divisor_bits; + big_divisor.limbs[1] = (ufbxi_bigint_limb)(divisor_bits >> 32u); + big_divisor.length = 2; + big_mantissa.limbs[0] = 0; + big_mantissa.limbs[1] = 0; + big_mantissa.limbs[2] = (ufbxi_bigint_limb)mantissa_bits; + big_mantissa.limbs[3] = (ufbxi_bigint_limb)(mantissa_bits >> 32u); + big_mantissa.length = 4; + exponent += (int32_t)divisor_zeros - (int32_t)mantissa_zeros - 64; + } else { + big_divisor.limbs[0] = (ufbxi_bigint_limb)pow5_value; + big_divisor.limbs[1] = (ufbxi_bigint_limb)(pow5_value >> 32u); + big_divisor.length = (pow5_value >> 32u) != 0 ? 2 : 1; + if (pow5 > 0) { + ufbxi_bigint_mul_pow5(&big_divisor, pow5); + } + + uint32_t divisor_zeros = ufbxi_lzcnt32(big_divisor.limbs[big_divisor.length - 1]); + if (big_divisor.length == 1) divisor_zeros += UFBXI_BIGINT_LIMB_BITS; + ufbxi_bigint_shift_left(&big_divisor, divisor_zeros); + uint32_t divisor_bits = big_divisor.length * UFBXI_BIGINT_LIMB_BITS; + + uint32_t mantissa_zeros = ufbxi_lzcnt32(big_mantissa.limbs[big_mantissa.length - 1]); + uint32_t mantissa_bits = big_mantissa.length * UFBXI_BIGINT_LIMB_BITS - mantissa_zeros; + uint32_t mantissa_min_bits = divisor_bits + enc_mantissa_bits + 2; + uint32_t mantissa_shift = mantissa_bits < mantissa_min_bits ? mantissa_min_bits - mantissa_bits : 0; + // Align mantissa to never have a high bit, this means we can skip the first digit during division. + mantissa_shift += ((mantissa_shift - mantissa_zeros) & (UFBXI_BIGINT_LIMB_BITS - 1)) == 0 ? 1 : 0; + if (mantissa_shift > 0) { + ufbxi_bigint_shift_left(&big_mantissa, mantissa_shift); + } + exponent += (int32_t)divisor_zeros - (int32_t)mantissa_shift; + } + + tail = ufbxi_bigint_div(&big_quotient, &big_mantissa, &big_divisor); + big_mantissa = big_quotient; + } else if (dec_exponent > 0) { + if (dec_exponent + (int32_t)(big_mantissa.length - 1) * 9 >= 310) return negative ? -UFBX_INFINITY : UFBX_INFINITY; + + exponent += dec_exponent; + ufbxi_bigint_mul_pow5(&big_mantissa, (uint32_t)dec_exponent); + } + + uint64_t mantissa = ufbxi_bigint_extract_high(big_mantissa, &exponent, &tail); + uint64_t sign_bit = (uint64_t)(negative ? 1u : 0u) << enc_sign_shift; + + uint32_t mantissa_shift = 64 - enc_mantissa_bits; + if (exponent > enc_max_exponent) { + return negative ? -UFBX_INFINITY : UFBX_INFINITY; + } else if (exponent <= -enc_max_exponent) { + mantissa_shift += (uint32_t)(-enc_max_exponent + 1 - exponent); + exponent = -enc_max_exponent + 1; + } + + mantissa = ufbxi_shift_right_round(mantissa, mantissa_shift, tail); + if (mantissa == 0) return negative ? -0.0 : 0.0; + + uint64_t bits = mantissa; + bits += (uint64_t)(exponent + enc_max_exponent - 1) << (enc_mantissa_bits - 1); + bits |= sign_bit; + + if (flags & UFBXI_PARSE_DOUBLE_AS_BINARY32) { + uint32_t bits_lo = (uint32_t)bits; + float result; + ufbxi_bit_cast(float, result, uint32_t, bits_lo); + return result; + } else { + double result; + ufbxi_bit_cast(double, result, uint64_t, bits); + return result; + } +} + +static ufbxi_noinline uint32_t ufbxi_parse_double_init_flags() +{ + // We require evaluation in double precision, either for doubles (0) or always (1) + // and rounding to nearest, which we can check for with `1 + eps == 1 - eps`. + #if UFBX_FLT_EVAL_METHOD == 0 || UFBX_FLT_EVAL_METHOD == 1 + static volatile double ufbxi_volatile_eps = 2.2250738585072014e-308; + if (1.0 + ufbxi_volatile_eps == 1.0 - ufbxi_volatile_eps) return UFBXI_PARSE_DOUBLE_ALLOW_FAST_PATH; + #endif + + return 0; } static ufbxi_forceinline int64_t ufbxi_parse_int64(const char *str, char **end) @@ -1421,6 +1689,24 @@ static ufbxi_forceinline int64_t ufbxi_parse_int64(const char *str, char **end) return negative ? -(int64_t)abs_val : (int64_t)abs_val; } +static ufbxi_noinline uint32_t ufbxi_parse_uint32_radix(const char *str, uint32_t radix) +{ + uint32_t value = 0; + for (const char *p = str; ; p++) { + char c = *p; + if (c >= '0' && c <= '9') { + value = value * radix + (uint32_t)(c - '0'); + } else if (radix == 16 && (c >= 'a' && c <= 'f')) { + value = value * radix + (uint32_t)(c + (10 - 'a')); + } else if (radix == 16 && (c >= 'A' && c <= 'F')) { + value = value * radix + (uint32_t)(c + (10 - 'A')); + } else { + break; + } + } + return value; +} + // -- DEFLATE implementation #if !defined(ufbx_inflate) @@ -2838,35 +3124,111 @@ ufbxi_extern_c ptrdiff_t ufbx_inflate(void *dst, size_t dst_size, const ufbx_inf uint32_t ref = (uint32_t)bits; ref = (ref>>24) | ((ref>>8)&0xff00) | ((ref<<8)&0xff0000) | (ref<<24); - uint32_t checksum = ufbxi_adler32(dc.out_begin, ufbxi_to_size(dc.out_ptr - dc.out_begin)); - if (ref != checksum) { - return -9; + uint32_t checksum = ufbxi_adler32(dc.out_begin, ufbxi_to_size(dc.out_ptr - dc.out_begin)); + if (ref != checksum) { + return -9; + } + } + } + + return dc.out_ptr - dc.out_begin; +} + +#endif // !defined(ufbx_inflate) + +// -- Printf + +typedef struct { + char *dst; + size_t length; + size_t pos; +} ufbxi_print_buffer; + +#define UFBXI_PRINT_UNSIGNED 0x1 +#define UFBXI_PRINT_STRING 0x2 +#define UFBXI_PRINT_SIZE_T 0x10 + +static void ufbxi_print_append(ufbxi_print_buffer *buf, size_t min_width, size_t max_width, const char *str) +{ + size_t width = 0; + for (width = 0; width < max_width; width++) { + if (!str[width]) break; + } + size_t pad = min_width > width ? min_width - width : 0; + for (size_t i = 0; i < pad; i++) { + if (buf->pos < buf->length) buf->dst[buf->pos++] = ' '; + } + for (size_t i = 0; i < width; i++) { + if (buf->pos < buf->length) buf->dst[buf->pos++] = str[i]; + } +} + +static char *ufbxi_print_format_int(char *buffer, uint64_t value) +{ + *--buffer = '\0'; + do { + uint32_t digit = (uint32_t)(value % 10); + value = value / 10; + *--buffer = (char)('0' + digit); + } while (value > 0); + return buffer; +} + +static void ufbxi_vprint(ufbxi_print_buffer *buf, const char *fmt, va_list args) +{ + char buffer[96]; // ufbxi_uninit + for (const char *p = fmt; *p;) { + if (*p == '%' && *++p != '%') { + size_t min_width = 0, max_width = SIZE_MAX; + if (*p == '*') { + p++; + min_width = (size_t)va_arg(args, int); + } + if (*p == '.') { + ufbxi_dev_assert(p[1] == '*'); + p += 2; + max_width = (size_t)va_arg(args, int); + } + uint32_t flags = 0; + switch (*p) { + case 'z': p++; flags |= UFBXI_PRINT_SIZE_T; break; + default: break; + } + switch (*p++) { + case 'u': flags |= UFBXI_PRINT_UNSIGNED; break; + case 's': flags |= UFBXI_PRINT_STRING; break; + default: break; + } + if (flags & UFBXI_PRINT_STRING) { + const char *str = va_arg(args, const char*); + ufbxi_print_append(buf, min_width, max_width, str); + } else if (flags & UFBXI_PRINT_UNSIGNED) { + uint64_t value = (flags & UFBXI_PRINT_SIZE_T) != 0 ? (uint64_t)va_arg(args, size_t) : (uint64_t)va_arg(args, uint32_t); + char *str = ufbxi_print_format_int(buffer + sizeof(buffer), value); + ufbxi_print_append(buf, min_width, max_width, str); + } else { + ufbxi_unreachable("Bad printf format"); } + } else { + if (buf->pos < buf->length) buf->dst[buf->pos++] = *p; + p++; } } - - return dc.out_ptr - dc.out_begin; + if (buf->length && buf->dst) { + size_t end = buf->pos <= buf->length - 1 ? buf->pos : buf->length - 1; + buf->dst[end] = '\0'; + } } -#endif // !defined(ufbx_inflate) - // -- Errors static const char ufbxi_empty_char[1] = { '\0' }; static ufbxi_noinline int ufbxi_vsnprintf(char *buf, size_t buf_size, const char *fmt, va_list args) { - int result = vsnprintf(buf, buf_size, fmt, args); - - if (result < 0) result = 0; - if ((size_t)result >= buf_size - 1) result = (int)buf_size - 1; - - // HACK: On some MSYS/MinGW implementations `vsnprintf` is broken and does - // not write the null terminator on truncation, it's always safe to do so - // let's just do it unconditionally here... - buf[result] = '\0'; - - return result; + ufbxi_print_buffer buffer = { buf, buf_size }; + ufbxi_vprint(&buffer, fmt, args); + return (int)ufbxi_min_sz(buffer.pos, buf_size - 1); } static ufbxi_noinline int ufbxi_snprintf(char *buf, size_t buf_size, const char *fmt, ...) @@ -2883,21 +3245,19 @@ static ufbxi_noinline void ufbxi_panicf_imp(ufbx_panic *panic, const char *fmt, if (panic && panic->did_panic) return; va_list args; // ufbxi_uninit - va_start(args, fmt); if (panic) { + va_start(args, fmt); panic->did_panic = true; panic->message_length = (size_t)ufbxi_vsnprintf(panic->message, sizeof(panic->message), fmt, args); + va_end(args); } else { - fprintf(stderr, "ufbx panic: "); - vfprintf(stderr, fmt, args); - fprintf(stderr, "\n"); - } - - va_end(args); + va_start(args, fmt); + char message[UFBX_PANIC_MESSAGE_LENGTH]; + ufbxi_vsnprintf(message, sizeof(message), fmt, args); + va_end(args); - if (!panic) { - ufbx_assert(false && "ufbx panic: See stderr for more information"); + ufbx_panic_handler(message); } } @@ -3046,7 +3406,7 @@ static ufbxi_noinline void ufbxi_clear_error(ufbx_error *err) #define ufbxi_fail_err_msg(err, desc, msg) return ufbxi_fail_imp_err(err, ufbxi_error_msg(desc, msg), ufbxi_function, ufbxi_line) #define ufbxi_report_err_msg(err, desc, msg) (void)ufbxi_fail_imp_err(err, ufbxi_error_msg(desc, msg), ufbxi_function, ufbxi_line) -static ufbxi_noinline void ufbxi_fix_error_type(ufbx_error *error, const char *default_desc) +static ufbxi_noinline void ufbxi_fix_error_type(ufbx_error *error, const char *default_desc, ufbx_error *p_error) { const char *desc = error->description.data; if (!desc) desc = default_desc; @@ -3096,6 +3456,9 @@ static ufbxi_noinline void ufbxi_fix_error_type(ufbx_error *error, const char *d } error->description.data = desc; error->description.length = strlen(desc); + if (p_error) { + memcpy(p_error, error, sizeof(ufbx_error)); + } } // -- Allocator @@ -3168,7 +3531,7 @@ static ufbxi_noinline void *ufbxi_alloc_size(ufbxi_allocator *ator, size_t size, } else if (ator->ator.allocator.realloc_fn) { ptr = ator->ator.allocator.realloc_fn(ator->ator.allocator.user, NULL, 0, total); } else { - ptr = malloc(total); + ptr = ufbx_malloc(total); } if (!ptr) { @@ -3215,7 +3578,7 @@ static ufbxi_noinline void *ufbxi_realloc_size(ufbxi_allocator *ator, size_t siz ator->ator.allocator.free_fn(ator->ator.allocator.user, old_ptr, old_total); } } else { - ptr = realloc(old_ptr, total); + ptr = ufbx_realloc(old_ptr, old_total, total); } ufbxi_check_return_err_msg(ator->error, ptr, NULL, "Out of memory"); @@ -3249,7 +3612,7 @@ static ufbxi_noinline void ufbxi_free_size(ufbxi_allocator *ator, size_t size, v ator->ator.allocator.realloc_fn(ator->ator.allocator.user, ptr, total, 0); } } else { - free(ptr); + ufbx_free(ptr, total); } } @@ -3888,7 +4251,7 @@ static ufbxi_noinline void ufbxi_map_init(ufbxi_map *map, ufbxi_allocator *ator, // allocation counts. We can work around this using a local allocator that doesn't // count the allocations. { - ufbxi_allocator *regression_ator = (ufbxi_allocator*)malloc(sizeof(ufbxi_allocator)); + ufbxi_allocator *regression_ator = (ufbxi_allocator*)ufbx_malloc(sizeof(ufbxi_allocator)); ufbx_assert(regression_ator); memset(regression_ator, 0, sizeof(ufbxi_allocator)); regression_ator->name = "regression"; @@ -3922,7 +4285,7 @@ static ufbxi_noinline void ufbxi_map_free(ufbxi_map *map) #if defined(UFBX_REGRESSION) if (regression_ator) { ufbxi_free_ator(regression_ator); - free(regression_ator); + ufbx_free(regression_ator, sizeof(ufbxi_allocator)); } #endif } @@ -5923,10 +6286,8 @@ typedef struct { // IO uint64_t data_offset; - ufbx_read_fn *read_fn; ufbx_skip_fn *skip_fn; - ufbx_close_fn *close_fn; void *read_user; char *read_buffer; @@ -6026,6 +6387,10 @@ typedef struct { // Temporary per-element flags uint8_t *tmp_element_flag; + // IO (cold) + ufbx_close_fn *close_fn; + ufbx_size_fn *size_fn; + ufbxi_ascii ascii; bool has_geometry_transform_nodes; @@ -6065,6 +6430,10 @@ typedef struct { ufbxi_warnings warnings; bool deferred_failure; + bool deferred_load; + + const char *load_filename; + size_t load_filename_len; bool parse_threaded; ufbxi_thread_pool thread_pool; @@ -6348,8 +6717,6 @@ ufbxi_nodiscard static ufbxi_noinline int ufbxi_read_to(ufbxi_context *uc, void return 1; } -// -- File IO - static ufbxi_noinline void ufbxi_init_ator(ufbx_error *error, ufbxi_allocator *ator, const ufbx_allocator_opts *opts, const char *name) { ufbx_allocator_opts zero_opts; @@ -6369,19 +6736,56 @@ static ufbxi_noinline void ufbxi_init_ator(ufbx_error *error, ufbxi_allocator *a ator->name = name; } -static ufbxi_noinline FILE *ufbxi_fopen(const char *path, size_t path_len, ufbxi_allocator *tmp_ator) +typedef struct { + ufbx_error error; + + ufbxi_allocator *parent_ator; + ufbxi_allocator ator; +} ufbxi_file_context; + +static ufbxi_noinline void ufbxi_begin_file_context(ufbxi_file_context *fc, ufbx_open_file_context ctx, const ufbx_allocator_opts *ator_opts) { -#if !defined(UFBX_STANDARD_C) && defined(_WIN32) - wchar_t wpath_buf[256]; - wchar_t *wpath = NULL; + memset(fc, 0, sizeof(ufbxi_file_context)); + if (ctx) { + fc->parent_ator = (ufbxi_allocator*)ctx; + fc->ator = *fc->parent_ator; + fc->ator.error = &fc->error; + } else { + ufbxi_init_ator(&fc->error, &fc->ator, ator_opts, "file"); + } +} - if (path_len == SIZE_MAX) { - path_len = strlen(path); +static ufbxi_noinline void ufbxi_end_file_context(ufbxi_file_context *fc, ufbx_error *error, bool ok) +{ + if (fc->parent_ator) { + fc->ator.error = fc->parent_ator->error; + *fc->parent_ator = fc->ator; + } else { + ufbxi_free_ator(&fc->ator); + } + if (error) { + if (!ok) { + ufbxi_fix_error_type(&fc->error, "Failed to open file", error); + } else { + ufbxi_clear_error(error); + } } +} + +// -- File IO + +#if !defined(UFBX_NO_STDIO) && !defined(UFBX_EXTERNAL_STDIO) + +static ufbxi_noinline FILE *ufbxi_fopen(ufbxi_file_context *fc, const char *path, size_t path_len, bool null_terminated) +{ + FILE *file = NULL; +#if !defined(UFBX_STANDARD_C) && defined(_WIN32) + (void)null_terminated; + wchar_t wpath_buf[256], *wpath = NULL; // ufbxi_uninit if (path_len < ufbxi_arraycount(wpath_buf) - 1) { wpath = wpath_buf; } else { - wpath = ufbxi_alloc(tmp_ator, wchar_t, path_len + 1); + wpath = ufbxi_alloc(&fc->ator, wchar_t, path_len + 1); if (!wpath) return NULL; } @@ -6416,45 +6820,38 @@ static ufbxi_noinline FILE *ufbxi_fopen(const char *path, size_t path_len, ufbxi } wpath[wlen] = 0; - FILE *file = NULL; -#if UFBXI_MSC_VER >= 1400 - if (_wfopen_s(&file, wpath, L"rb") != 0) { - file = NULL; - } -#else - file = _wfopen(wpath, L"rb"); -#endif - + #if UFBXI_MSC_VER >= 1400 + if (_wfopen_s(&file, wpath, L"rb") != 0) file = NULL; + #else + file = _wfopen(wpath, L"rb"); + #endif if (wpath != wpath_buf) { - ufbxi_free(tmp_ator, wchar_t, wpath, path_len + 1); + ufbxi_free(&fc->ator, wchar_t, wpath, path_len + 1); } - - return file; #else - if (path_len == SIZE_MAX) { - return fopen(path, "rb"); - } - - char copy_buf[256]; // ufbxi_uninit - char *copy = NULL; - - if (path_len < ufbxi_arraycount(copy_buf) - 1) { - copy = copy_buf; + char copy_buf[256], *copy = NULL; // ufbxi_uninit + if (null_terminated) { + copy = (char*)path; } else { - copy = ufbxi_alloc(tmp_ator, char, path_len + 1); - if (!copy) return NULL; + if (path_len < ufbxi_arraycount(copy_buf) - 1) { + copy = copy_buf; + } else { + copy = ufbxi_alloc(&fc->ator, char, path_len + 1); + if (!copy) return NULL; + } + memcpy(copy, path, path_len); + copy[path_len] = '\0'; } - memcpy(copy, path, path_len); - copy[path_len] = '\0'; - - FILE *file = fopen(copy, "rb"); - - if (copy != copy_buf) { - ufbxi_free(tmp_ator, char, copy, path_len + 1); + file = fopen(copy, "rb"); + if (!null_terminated && copy != copy_buf) { + ufbxi_free(&fc->ator, char, copy, path_len + 1); } - - return file; #endif + if (!file) { + ufbxi_set_err_info(&fc->error, path, path_len); + ufbxi_report_err_msg(&fc->error, "file", "File not found"); + } + return file; } static uint64_t ufbxi_ftell(FILE *file) @@ -6466,20 +6863,20 @@ static uint64_t ufbxi_ftell(FILE *file) int64_t result = _ftelli64(file); if (result >= 0) return (uint64_t)result; #else - long result = ftell(file); + int64_t result = ftell(file); if (result >= 0) return (uint64_t)result; #endif return UINT64_MAX; } -static size_t ufbxi_file_read(void *user, void *data, size_t max_size) +static size_t ufbxi_stdio_read(void *user, void *data, size_t max_size) { FILE *file = (FILE*)user; if (ferror(file)) return SIZE_MAX; return fread(data, 1, max_size, file); } -static bool ufbxi_file_skip(void *user, size_t size) +static bool ufbxi_stdio_skip(void *user, size_t size) { FILE *file = (FILE*)user; ufbx_assert(size <= UFBXI_MAX_SKIP_SIZE); @@ -6488,12 +6885,94 @@ static bool ufbxi_file_skip(void *user, size_t size) return true; } -static void ufbxi_file_close(void *user) +static uint64_t ufbxi_stdio_size(void *user) +{ + FILE *file = (FILE*)user; + uint64_t result = 0; + uint64_t begin = ufbxi_ftell(file); + if (begin < UINT64_MAX) { + fpos_t pos; // ufbxi_uninit + if (fgetpos(file, &pos) == 0) { + if (fseek(file, 0, SEEK_END) == 0) { + uint64_t end = ufbxi_ftell(file); + if (end != UINT64_MAX && begin < end) { + result = end - begin; + } + // Both `rewind()` and `fsetpos()` to reset error and EOF + rewind(file); + fsetpos(file, &pos); + } + } + } + return result; +} + +static void ufbxi_stdio_close(void *user) { FILE *file = (FILE*)user; fclose(file); } +static ufbxi_noinline void ufbxi_stdio_init(ufbx_stream *stream, void *file, bool close) +{ + stream->read_fn = &ufbxi_stdio_read; + stream->skip_fn = &ufbxi_stdio_skip; + stream->size_fn = &ufbxi_stdio_size; + stream->close_fn = close ? &ufbxi_stdio_close : NULL; + stream->user = file; +} + +static ufbxi_noinline bool ufbxi_stdio_open(ufbxi_file_context *fc, ufbx_stream *stream, const char *path, size_t path_len, bool null_terminated) +{ + FILE *file = ufbxi_fopen(fc, path, path_len, null_terminated); + if (!file) return false; + ufbxi_stdio_init(stream, file, true); + return true; +} + +#elif defined(UFBX_EXTERNAL_STDIO) + +static ufbxi_noinline void ufbxi_stdio_init(ufbx_stream *stream, void *file, bool close) +{ + stream->read_fn = &ufbx_stdio_read; + stream->skip_fn = &ufbx_stdio_skip; + stream->size_fn = &ufbx_stdio_size; + stream->close_fn = close ? &ufbx_stdio_close : NULL; + stream->user = file; +} + +static ufbxi_noinline bool ufbxi_stdio_open(ufbxi_file_context *fc, ufbx_stream *stream, const char *path, size_t path_len, bool null_terminated) +{ + char copy_buf[256], *copy = NULL; // ufbxi_uninit + if (null_terminated) { + copy = (char*)path; + } else { + if (path_len < ufbxi_arraycount(copy_buf) - 1) { + copy = copy_buf; + } else { + copy = ufbxi_alloc(&fc->ator, char, path_len + 1); + if (!copy) return false; + } + memcpy(copy, path, path_len); + copy[path_len] = '\0'; + } + void *file = ufbx_stdio_open(copy, path_len); + if (!null_terminated && copy != copy_buf) { + ufbxi_free(&fc->ator, char, copy, path_len + 1); + } + if (!file) { + ufbxi_set_err_info(&fc->error, path, path_len); + ufbxi_report_err_msg(&fc->error, "file", "File not found"); + return false; + } + ufbxi_stdio_init(stream, file, true); + return true; +} + +#endif + +// -- Memory IO + typedef struct { const void *data; size_t size; @@ -6502,7 +6981,8 @@ typedef struct { // Own allocation information size_t self_size; - ufbxi_allocator ator; + ufbxi_allocator *parent_ator; + ufbxi_allocator local_ator; ufbx_error error; char data_copy[]; } ufbxi_memory_stream; @@ -6524,6 +7004,12 @@ static bool ufbxi_memory_skip(void *user, size_t size) return true; } +static uint64_t ufbxi_memory_size(void *user) +{ + ufbxi_memory_stream *stream = (ufbxi_memory_stream*)user; + return stream->size; +} + static void ufbxi_memory_close(void *user) { ufbxi_memory_stream *stream = (ufbxi_memory_stream*)user; @@ -6531,9 +7017,13 @@ static void ufbxi_memory_close(void *user) stream->close_cb.fn(stream->close_cb.user, (void*)stream->data, stream->size); } - ufbxi_allocator ator = stream->ator; - ufbxi_free(&ator, char, stream, stream->self_size); - ufbxi_free_ator(&ator); + if (stream->parent_ator) { + ufbxi_free(stream->parent_ator, char, stream, stream->self_size); + } else { + ufbxi_allocator ator = stream->local_ator; + ufbxi_free(&ator, char, stream, stream->self_size); + ufbxi_free_ator(&ator); + } } // -- XML @@ -6703,9 +7193,9 @@ static ufbxi_noinline int ufbxi_xml_read_until(ufbxi_xml_context *xc, ufbx_strin if (entity[0] == '#') { unsigned long code = 0; if (entity[1] == 'x') { - code = strtoul(entity + 2, NULL, 16); + code = ufbxi_parse_uint32_radix(entity + 2, 16); } else { - code = strtoul(entity + 1, NULL, 10); + code = ufbxi_parse_uint32_radix(entity + 1, 10); } char bytes[5] = { 0 }; @@ -9098,11 +9588,9 @@ ufbxi_nodiscard ufbxi_noinline static int ufbxi_ascii_next_token(ufbxi_context * token->value.i64 = ufbxi_parse_int64(token->str_data, &end); ufbxi_check(end == token->str_data + token->str_len - 1); } else if (token->type == UFBXI_ASCII_FLOAT) { - if (ua->parse_as_f32) { - token->value.f64 = strtof(token->str_data, &end); - } else { - token->value.f64 = ufbxi_parse_double(token->str_data, token->str_len, &end, uc->double_parse_flags); - } + uint32_t flags = uc->double_parse_flags; + if (ua->parse_as_f32) flags = UFBXI_PARSE_DOUBLE_AS_BINARY32; + token->value.f64 = ufbxi_parse_double(token->str_data, token->str_len, &end, flags); ufbxi_check(end == token->str_data + token->str_len - 1); } } @@ -9499,8 +9987,6 @@ ufbxi_nodiscard static ufbxi_noinline int ufbxi_ascii_read_float_array(ufbxi_con *v = (float)val; } - // TODO: Collect ASCII numbers to deferred parse integer/string segments - // Try to parse the next value, we don't commit this until we find a comma after it above. char *num_end = NULL; size_t left = ufbxi_to_size(end - src_scan); @@ -15725,44 +16211,12 @@ ufbxi_nodiscard ufbxi_noinline static int ufbxi_resolve_relative_filename(ufbxi_ // Open file utility -static void *ufbxi_ator_alloc(void *user, size_t size) -{ - ufbxi_allocator *ator = (ufbxi_allocator*)user; - return ufbxi_alloc(ator, char, size); -} - -static void *ufbxi_ator_realloc(void *user, void *old_ptr, size_t old_size, size_t new_size) -{ - ufbxi_allocator *ator = (ufbxi_allocator*)user; - return ufbxi_realloc(ator, char, old_ptr, old_size, new_size); -} - -static void ufbxi_ator_free(void *user, void *ptr, size_t size) -{ - ufbxi_allocator *ator = (ufbxi_allocator*)user; - ufbxi_free(ator, char, ptr, size); -} - -static ufbxi_noinline void ufbxi_setup_ator_allocator(ufbx_allocator *allocator, ufbxi_allocator *ator) -{ - allocator->alloc_fn = &ufbxi_ator_alloc; - allocator->realloc_fn = &ufbxi_ator_realloc; - allocator->free_fn = &ufbxi_ator_free; - allocator->free_allocator_fn = NULL; - allocator->user = ator; -} - static ufbxi_noinline bool ufbxi_open_file(const ufbx_open_file_cb *cb, ufbx_stream *stream, const char *path, size_t path_len, const ufbx_blob *original_filename, ufbxi_allocator *ator, ufbx_open_file_type type) { if (!cb || !cb->fn) return false; ufbx_open_file_info info; // ufbxi_uninit - if (ator) { - ufbxi_setup_ator_allocator(&info.temp_allocator, ator); - } else { - memset(&info.temp_allocator, 0, sizeof(info.temp_allocator)); - } - + info.context = (uintptr_t)ator; if (original_filename) { info.original_filename = *original_filename; } else { @@ -17084,6 +17538,7 @@ ufbxi_nodiscard static ufbxi_noinline int ufbxi_obj_load_mtl(ufbxi_context *uc) if (!has_stream && uc->opts.load_external_files && uc->opts.obj_search_mtl_by_filename && path.length > 4) { ufbx_string ext = { path.data + path.length - 4, 4 }; if (ufbxi_match(&ext, "\\c.obj")) { + ufbxi_analysis_assert(path.length < SIZE_MAX - 1); char *copy = ufbxi_push_copy(&uc->tmp, char, path.length + 1, path.data); ufbxi_check(copy); copy[path.length - 3] = copy[path.length - 3] == 'O' ? 'M' : 'm'; @@ -23237,9 +23692,9 @@ ufbxi_nodiscard static ufbxi_noinline int ufbxi_cache_load_xml_imp(ufbxi_cache_c if (tag_fps) { ufbxi_xml_attrib *fps = ufbxi_xml_find_attrib(tag_fps, "TimePerFrame"); if (fps) { - int value = atoi(fps->value.data); + uint32_t value = ufbxi_parse_uint32_radix(fps->value.data, 10); if (value > 0) { - cc->xml_ticks_per_frame = (uint32_t)value; + cc->xml_ticks_per_frame = value; } } } @@ -23264,9 +23719,9 @@ ufbxi_nodiscard static ufbxi_noinline int ufbxi_cache_load_xml_imp(ufbxi_cache_c ufbxi_xml_attrib *start_time = ufbxi_xml_find_attrib(tag, "StartTime"); ufbxi_xml_attrib *end_time = ufbxi_xml_find_attrib(tag, "EndTime"); if (sampling_rate && start_time && end_time) { - channel->sample_rate = (uint32_t)atoi(sampling_rate->value.data); - channel->start_time = (uint32_t)atoi(start_time->value.data); - channel->end_time = (uint32_t)atoi(end_time->value.data); + channel->sample_rate = ufbxi_parse_uint32_radix(sampling_rate->value.data, 10); + channel->start_time = ufbxi_parse_uint32_radix(start_time->value.data, 10); + channel->end_time = ufbxi_parse_uint32_radix(end_time->value.data, 10); channel->current_time = channel->start_time; channel->try_load = true; } @@ -23591,7 +24046,7 @@ ufbxi_noinline static ufbx_geometry_cache *ufbxi_cache_load(ufbxi_cache_context if (ok) { return &cc->imp->cache; } else { - ufbxi_fix_error_type(&cc->error, "Failed to load geometry cache"); + ufbxi_fix_error_type(&cc->error, "Failed to load geometry cache", NULL); if (!cc->owned_by_scene) { ufbxi_buf_free(&cc->string_pool.buf); ufbxi_free_ator(&cc->ator_result); @@ -24090,6 +24545,50 @@ ufbxi_nodiscard static ufbxi_noinline int ufbxi_load_imp(ufbxi_context *uc) { // Check for deferred failure if (uc->deferred_failure) return 0; + if (uc->deferred_load) { + ufbx_stream stream = { 0 }; + ufbx_open_file_opts opts = { 0 }; + const char *filename = uc->load_filename; + size_t filename_len = uc->load_filename_len; + bool ok = false; + if (filename_len == SIZE_MAX) { + opts.filename_null_terminated = true; + filename_len = strlen(filename); + } + if (uc->opts.filename.length == 0 || uc->opts.filename.data == NULL) { + uc->opts.filename.data = filename; + uc->opts.filename.length = filename_len; + } + ufbx_error error; + error.type = UFBX_ERROR_NONE; + if (uc->opts.open_main_file_with_default || uc->opts.open_file_cb.fn == &ufbx_default_open_file) { + ufbx_open_file_context ctx = (ufbx_open_file_context)&uc->ator_tmp; + ok = ufbx_open_file_ctx(&stream, ctx, filename, filename_len, &opts, &error); + } else { + ok = ufbxi_open_file(&uc->opts.open_file_cb, &stream, uc->load_filename, filename_len, NULL, &uc->ator_tmp, UFBX_OPEN_FILE_MAIN_MODEL); + } + if (!ok) { + if (error.type != UFBX_ERROR_NONE) { + // cppcheck-suppress uninitStructMember + uc->error = error; + } else { + ufbxi_set_err_info(&uc->error, filename, filename_len); + } + ufbxi_fail_msg("open_file_fn()", "File not found"); + } + uc->read_fn = stream.read_fn; + uc->skip_fn = stream.skip_fn; + uc->size_fn = stream.size_fn; + uc->close_fn = stream.close_fn; + uc->read_user = stream.user; + } + + if (uc->opts.progress_cb.fn && uc->progress_bytes_total == 0 && uc->size_fn) { + uint64_t total = uc->size_fn(uc->read_user); + ufbxi_check(total != UINT64_MAX); + uc->progress_bytes_total = total; + } + ufbxi_check(uc->opts.path_separator >= 0x20 && uc->opts.path_separator <= 0x7e); ufbxi_check(ufbxi_fixup_opts_string(uc, &uc->opts.filename, false)); @@ -24431,34 +24930,24 @@ static ufbxi_noinline ufbx_scene *ufbxi_load(ufbxi_context *uc, const ufbx_load_ int ok = ufbxi_load_imp(uc); - ufbxi_free_temp(uc); - if (uc->close_fn) { uc->close_fn(uc->read_user); } + ufbxi_free_temp(uc); + if (ok) { if (p_error) { ufbxi_clear_error(p_error); } return &uc->scene_imp->scene; } else { - ufbxi_fix_error_type(&uc->error, "Failed to load"); - if (p_error) *p_error = uc->error; + ufbxi_fix_error_type(&uc->error, "Failed to load", p_error); ufbxi_free_result(uc); return NULL; } } -static ufbxi_noinline ufbx_scene *ufbxi_load_not_found(const char *filename, size_t filename_len, ufbx_error *p_error) -{ - ufbxi_context uc = { UFBX_ERROR_NONE }; - ufbxi_set_err_info(&uc.error, filename, filename_len); - ufbxi_report_err_msg(&uc.error, "File not found", "File not found"); - uc.deferred_failure = true; - return ufbxi_load(&uc, NULL, p_error); -} - // -- Animation evaluation static ufbxi_forceinline bool ufbxi_override_less_than_prop(const ufbx_prop_override *over, uint32_t element_id, const ufbx_prop *prop) @@ -25240,8 +25729,7 @@ ufbxi_nodiscard static ufbxi_noinline ufbx_scene *ufbxi_evaluate_scene(ufbxi_eva } return &ec->scene_imp->scene; } else { - ufbxi_fix_error_type(&ec->error, "Failed to evaluate"); - if (p_error) *p_error = ec->error; + ufbxi_fix_error_type(&ec->error, "Failed to evaluate", p_error); ufbxi_buf_free(&ec->tmp); ufbxi_buf_free(&ec->result); ufbxi_free_ator(&ec->ator_tmp); @@ -25310,11 +25798,11 @@ static bool ufbxi_prop_override_less(void *user, const void *va, const void *vb) return strcmp(a->prop_name.data, b->prop_name.data) < 0; } -static int ufbxi_cmp_transform_override(const void *va, const void *vb) +static bool ufbxi_transform_override_less(void *user, const void *va, const void *vb) { + (void)user; const ufbx_transform_override *a = (const ufbx_transform_override*)va, *b = (const ufbx_transform_override*)vb; - if (a->node_id != b->node_id) return a->node_id < b->node_id ? -1 : 1; - return 0; + return a->node_id < b->node_id; } ufbxi_nodiscard static ufbxi_noinline int ufbxi_create_anim_imp(ufbxi_create_anim_context *ac) @@ -25419,8 +25907,7 @@ ufbxi_nodiscard static ufbxi_noinline int ufbxi_create_anim_imp(ufbxi_create_ani anim->transform_overrides.count = ac->opts.transform_overrides.count; anim->transform_overrides.data = ufbxi_push_copy(&ac->result, ufbx_transform_override, anim->transform_overrides.count, ac->opts.transform_overrides.data); ufbxi_check_err(&ac->error, anim->transform_overrides.data); - - qsort(anim->transform_overrides.data, anim->transform_overrides.count, sizeof(ufbx_transform_override), &ufbxi_cmp_transform_override); + ufbxi_unstable_sort(anim->transform_overrides.data, anim->transform_overrides.count, sizeof(ufbx_transform_override), &ufbxi_transform_override_less, NULL); } ac->imp = ufbxi_push(&ac->result, ufbxi_anim_imp, 1); @@ -25473,6 +25960,9 @@ typedef struct { ufbx_baked_node **baked_nodes; bool *nodes_to_bake; + char *tmp_arr; + size_t tmp_arr_size; + const ufbx_scene *scene; const ufbx_anim *anim; ufbx_bake_opts opts; @@ -25495,14 +25985,15 @@ typedef struct { ufbx_anim_value *anim_value; } ufbxi_bake_prop; -static int ufbxi_cmp_bake_prop(const void *va, const void *vb) +static bool ufbxi_bake_prop_less(void *user, const void *va, const void *vb) { + (void)user; const ufbxi_bake_prop *a = (const ufbxi_bake_prop*)va; const ufbxi_bake_prop *b = (const ufbxi_bake_prop*)vb; - if (a->sort_id != b->sort_id) return a->sort_id < b->sort_id ? -1 : 1; - if (a->element_id != b->element_id) return a->element_id < b->element_id ? -1 : 1; - if (a->prop_name != b->prop_name) return strcmp(a->prop_name, b->prop_name); - return a->anim_value < b->anim_value; + if (a->sort_id != b->sort_id) return a->sort_id < b->sort_id; + if (a->element_id != b->element_id) return a->element_id < b->element_id; + if (a->prop_name != b->prop_name) return strcmp(a->prop_name, b->prop_name) < 0; + return false; } ufbx_static_assert(bake_step_left, UFBX_BAKED_KEY_STEP_LEFT == 0x1); @@ -25518,13 +26009,6 @@ static ufbxi_forceinline int ufbxi_cmp_bake_time(ufbxi_bake_time a, ufbxi_bake_t return 0; } -static int ufbxi_cmp_bake_time_fn(const void *va, const void *vb) -{ - const ufbxi_bake_time a = *(const ufbxi_bake_time*)va; - const ufbxi_bake_time b = *(const ufbxi_bake_time*)vb; - return ufbxi_cmp_bake_time(a, b); -} - ufbxi_nodiscard static ufbxi_forceinline int ufbxi_bake_push_time(ufbxi_bake_context *bc, double time, uint32_t flags) { ufbxi_bake_time *p_key = ufbxi_push_fast(&bc->tmp_times, ufbxi_bake_time, 1); @@ -25609,6 +26093,13 @@ ufbxi_nodiscard static ufbxi_noinline bool ufbxi_in_list(const char *const *item return false; } +ufbxi_nodiscard static ufbxi_noinline int ufbxi_sort_bake_times(ufbxi_bake_context *bc, ufbxi_bake_time *times, size_t count) +{ + ufbxi_check_err(&bc->error, ufbxi_grow_array(&bc->ator_tmp, &bc->tmp_arr, &bc->tmp_arr_size, count * sizeof(ufbxi_bake_time))); + ufbxi_macro_stable_sort(ufbxi_bake_time, 32, times, bc->tmp_arr, count, ( ufbxi_cmp_bake_time(*a, *b) < 0 )); + return 1; +} + ufbxi_nodiscard static ufbxi_noinline int ufbxi_finalize_bake_times(ufbxi_bake_context *bc, ufbxi_bake_time_list *p_dst) { if (bc->layer_weight_times.count > 0) { @@ -25624,8 +26115,7 @@ ufbxi_nodiscard static ufbxi_noinline int ufbxi_finalize_bake_times(ufbxi_bake_c ufbxi_bake_time *times = ufbxi_push_pop(&bc->tmp_prop, &bc->tmp_times, ufbxi_bake_time, num_times); ufbxi_check_err(&bc->error, times); - // TODO: Something better - qsort(times, num_times, sizeof(ufbxi_bake_time), &ufbxi_cmp_bake_time_fn); + ufbxi_check_err(&bc->error, ufbxi_sort_bake_times(bc, times, num_times)); // Deduplicate times if (num_times > 0) { @@ -25736,13 +26226,13 @@ ufbxi_nodiscard static ufbxi_noinline int ufbxi_finalize_bake_times(ufbxi_bake_c #define ufbxi_add_epsilon(a, epsilon) ((a)>0 ? (a)*(epsilon) : (a)/(epsilon)) #define ufbxi_sub_epsilon(a, epsilon) ((a)>0 ? (a)/(epsilon) : (a)*(epsilon)) -static ufbxi_noinline bool ufbxi_postprocess_step(ufbxi_bake_context *bc, double prev_time, double next_time, double *p_time, uint32_t flags) +static ufbxi_noinline bool ufbxi_postprocess_step(ufbxi_bake_context *bc, double prev_time, double next_time, double *p_time, ufbx_baked_key_flags flags) { ufbxi_dev_assert((flags & (UFBX_BAKED_KEY_STEP_LEFT|UFBX_BAKED_KEY_STEP_RIGHT)) != 0); bool left = (flags & UFBX_BAKED_KEY_STEP_LEFT) != 0; double step = 0.001; - double epsilon = 1.0 + FLT_EPSILON * 4.0f; + double epsilon = 1.0 + UFBX_FLT_EPSILON * 4.0f; double time = *p_time; switch (bc->opts.step_handling) { @@ -25982,7 +26472,7 @@ ufbxi_nodiscard static ufbxi_noinline int ufbxi_push_resampled_times(ufbxi_bake_ ufbxi_bake_time *times = ufbxi_push(&bc->tmp_times, ufbxi_bake_time, keys.count); ufbxi_check_err(&bc->error, times); for (size_t i = 0; i < keys.count; i++) { - uint32_t flags = keys.data[i].flags; + ufbx_baked_key_flags flags = keys.data[i].flags; double time = keys.data[i].time; if ((flags & UFBX_BAKED_KEY_STEP_LEFT) != 0 && i + 1 < keys.count && (keys.data[i + 1].flags & UFBX_BAKED_KEY_STEP_KEY) != 0) { time = keys.data[i + 1].time; @@ -26402,8 +26892,7 @@ ufbxi_nodiscard static ufbxi_noinline int ufbxi_bake_anim(ufbxi_bake_context *bc ufbxi_bake_prop *props = ufbxi_push_pop(&bc->tmp, &bc->tmp_bake_props, ufbxi_bake_prop, num_props); ufbxi_check_err(&bc->error, props); - // TODO: Macro unstable/non allocating sort - qsort(props, num_props, sizeof(ufbxi_bake_prop), &ufbxi_cmp_bake_prop); + ufbxi_unstable_sort(props, num_props, sizeof(ufbxi_bake_prop), &ufbxi_bake_prop_less, NULL); // Pre-bake layer weight times if (!bc->opts.ignore_layer_weight_animation) { @@ -27453,19 +27942,19 @@ ufbxi_noinline static uint32_t ufbxi_triangulate_ngon(ufbxi_ngon_context *nc, ui #endif -static int ufbxi_cmp_topo_index_prev_next(const void *va, const void *vb) +static bool ufbxi_topo_less_index_prev_next(void *user, const void *va, const void *vb) { + (void)user; const ufbx_topo_edge *a = (const ufbx_topo_edge*)va, *b = (const ufbx_topo_edge*)vb; - if ((int32_t)a->prev != (int32_t)b->prev) return (int32_t)a->prev < (int32_t)b->prev ? -1 : +1; - if ((int32_t)a->next != (int32_t)b->next) return (int32_t)a->next < (int32_t)b->next ? -1 : +1; - return 0; + if ((int32_t)a->prev != (int32_t)b->prev) return (int32_t)a->prev < (int32_t)b->prev; + return (int32_t)a->next < (int32_t)b->next; } -static int ufbxi_cmp_topo_index_index(const void *va, const void *vb) +static bool ufbxi_topo_less_index_index(void *user, const void *va, const void *vb) { + (void)user; const ufbx_topo_edge *a = (const ufbx_topo_edge*)va, *b = (const ufbx_topo_edge*)vb; - if ((int32_t)a->index != (int32_t)b->index) return (int32_t)a->index < (int32_t)b->index ? -1 : +1; - return 0; + return (int32_t)a->index < (int32_t)b->index; } ufbxi_noinline static void ufbxi_compute_topology(const ufbx_mesh *mesh, ufbx_topo_edge *topo) @@ -27494,8 +27983,7 @@ ufbxi_noinline static void ufbxi_compute_topology(const ufbx_mesh *mesh, ufbx_to } } - // TODO: Macro unstable/non allocating sort - qsort(topo, num_indices, sizeof(ufbx_topo_edge), &ufbxi_cmp_topo_index_prev_next); + ufbxi_unstable_sort(topo, num_indices, sizeof(ufbx_topo_edge), &ufbxi_topo_less_index_prev_next, NULL); if (mesh->edges.data) { for (uint32_t ei = 0; ei < mesh->num_edges; ei++) { @@ -27535,8 +28023,7 @@ ufbxi_noinline static void ufbxi_compute_topology(const ufbx_mesh *mesh, ufbx_to i0 = i1 + 1; } - // TODO: Macro unstable/non allocating sort - qsort(topo, num_indices, sizeof(ufbx_topo_edge), &ufbxi_cmp_topo_index_index); + ufbxi_unstable_sort(topo, num_indices, sizeof(ufbx_topo_edge), &ufbxi_topo_less_index_index, NULL); // Fix `prev` and `next` to the actual index values for (uint32_t fi = 0; fi < mesh->num_faces; fi++) { @@ -27702,12 +28189,13 @@ static int ufbxi_subdivide_sum_vec4(void *user, void *output, const ufbxi_subdiv return 1; } -static ufbxi_noinline int ufbxi_cmp_subdivision_weight(const void *va, const void *vb) +static ufbxi_noinline bool ufbxi_subdivision_weight_less(void *user, const void *va, const void *vb) { + (void)user; ufbx_subdivision_weight a = *(const ufbx_subdivision_weight*)va, b = *(const ufbx_subdivision_weight*)vb; ufbxi_dev_assert(a.index != b.index); - if (a.weight != b.weight) return a.weight > b.weight ? -1 : +1; - return a.index < b.index ? -1 : +1; + if (a.weight != b.weight) return a.weight > b.weight; + return a.index < b.index; } static int ufbxi_subdivide_sum_vertex_weights(void *user, void *output, const ufbxi_subdivide_input *inputs, size_t num_inputs) @@ -27743,7 +28231,7 @@ static int ufbxi_subdivide_sum_vertex_weights(void *user, void *output, const uf vertex_weights[vx] = 0.0f; } - qsort(tmp_weights, num_weights, sizeof(ufbx_subdivision_weight), ufbxi_cmp_subdivision_weight); + ufbxi_unstable_sort(tmp_weights, num_weights, sizeof(ufbx_subdivision_weight), ufbxi_subdivision_weight_less, NULL); if (sc->max_vertex_weights != SIZE_MAX) { num_weights = ufbxi_min_sz(sc->max_vertex_weights, num_weights); @@ -28823,8 +29311,7 @@ ufbxi_noinline static ufbx_mesh *ufbxi_subdivide_mesh(const ufbx_mesh *mesh, siz ufbxi_mesh_imp *imp = sc.imp; return &imp->mesh; } else { - ufbxi_fix_error_type(&sc.error, "Failed to subdivide"); - if (p_error) *p_error = sc.error; + ufbxi_fix_error_type(&sc.error, "Failed to subdivide", p_error); ufbxi_buf_free(&sc.result); ufbxi_free_ator(&sc.ator_tmp); ufbxi_free_ator(&sc.ator_result); @@ -28976,7 +29463,7 @@ static ufbxi_noinline size_t ufbxi_generate_indices(const ufbx_vertex_stream *us ufbxi_clear_error(error); } else { - ufbxi_fix_error_type(error, "Failed to generate indices"); + ufbxi_fix_error_type(error, "Failed to generate indices", NULL); } if (streams && streams != local_streams) { @@ -29169,29 +29656,43 @@ ufbx_abi_data_def const size_t ufbx_element_type_size[UFBX_ELEMENT_TYPE_COUNT] = sizeof(ufbx_metadata_object), }; -ufbx_abi bool ufbx_open_file(ufbx_stream *stream, const char *path, size_t path_len) +ufbx_abi bool ufbx_default_open_file(void *user, ufbx_stream *stream, const char *path, size_t path_len, const ufbx_open_file_info *info) { - ufbxi_allocator tmp_ator = { 0 }; - ufbx_error tmp_error = { UFBX_ERROR_NONE }; - ufbxi_init_ator(&tmp_error, &tmp_ator, NULL, "filename"); - FILE *f = ufbxi_fopen(path, path_len, &tmp_ator); - if (!f) return false; + (void)user; + return ufbx_open_file_ctx(stream, info->context, path, path_len, NULL, NULL); +} - stream->read_fn = &ufbxi_file_read; - stream->skip_fn = &ufbxi_file_skip; - stream->close_fn = &ufbxi_file_close; - stream->user = f; - return true; +ufbx_abi bool ufbx_open_file(ufbx_stream *stream, const char *path, size_t path_len, const ufbx_open_file_opts *opts, ufbx_error *error) +{ + return ufbx_open_file_ctx(stream, (ufbx_open_file_context)NULL, path, path_len, opts, error); } -ufbx_abi bool ufbx_default_open_file(void *user, ufbx_stream *stream, const char *path, size_t path_len, const ufbx_open_file_info *info) +ufbx_abi bool ufbx_open_file_ctx(ufbx_stream *stream, ufbx_open_file_context ctx, const char *path, size_t path_len, const ufbx_open_file_opts *opts, ufbx_error *error) { - (void)user; - (void)info; - return ufbx_open_file(stream, path, path_len); + bool ok = false; + ufbxi_file_context fc; // ufbxi_uninit + ufbxi_begin_file_context(&fc, ctx, NULL); + if (path_len == SIZE_MAX) path_len = strlen(path); +#if !defined(UFBX_NO_STDIO) + ok = ufbxi_stdio_open(&fc, stream, path, path_len, opts ? opts->filename_null_terminated : false); +#else + (void)stream; + (void)path; + (void)path_len; + (void)opts; + ufbxi_fmt_err_info(&fc.error, "UFBX_NO_STDIO"); + ufbxi_report_err_msg(&fc.error, "UFBX_NO_STDIO", "Feature disabled"); +#endif + ufbxi_end_file_context(&fc, error, ok); + return ok; } ufbx_abi bool ufbx_open_memory(ufbx_stream *stream, const void *data, size_t data_size, const ufbx_open_memory_opts *opts, ufbx_error *error) +{ + return ufbx_open_memory_ctx(stream, (ufbx_open_file_context)NULL, data, data_size, opts, error); +} + +ufbx_abi bool ufbx_open_memory_ctx(ufbx_stream *stream, ufbx_open_file_context ctx, const void *data, size_t data_size, const ufbx_open_memory_opts *opts, ufbx_error *error) { ufbx_open_memory_opts local_opts; // ufbxi_uninit if (!opts) { @@ -29200,22 +29701,17 @@ ufbx_abi bool ufbx_open_memory(ufbx_stream *stream, const void *data, size_t dat } ufbx_assert(opts->_begin_zero == 0 && opts->_end_zero == 0); - ufbx_error local_error = { UFBX_ERROR_NONE }; - if (!error) error = &local_error; - ufbxi_clear_error(error); - - ufbxi_allocator ator = { 0 }; - ufbxi_init_ator(error, &ator, &opts->allocator, "memory"); + ufbxi_file_context fc; // ufbxi_uninit + ufbxi_begin_file_context(&fc, ctx, &opts->allocator); size_t copy_size = opts->no_copy ? 0 : data_size; // Align the allocation size to 8 bytes to make sure the header is aligned. size_t self_size = ufbxi_align_to_mask(sizeof(ufbxi_memory_stream) + copy_size, 7); - void *memory = ufbxi_alloc(&ator, char, self_size); + void *memory = ufbxi_alloc(&fc.ator, char, self_size); if (!memory) { - ufbxi_free_ator(&ator); - ufbxi_fix_error_type(error, "Failed to open memory"); + ufbxi_end_file_context(&fc, error, false); return false; } @@ -29234,14 +29730,20 @@ ufbx_abi bool ufbx_open_memory(ufbx_stream *stream, const void *data, size_t dat } // Transplant the allocator in the result blob - mem->ator = ator; - mem->ator.error = &mem->error; + if (fc.parent_ator) { + mem->parent_ator = fc.parent_ator; + } else { + fc.parent_ator = &mem->local_ator; + } stream->read_fn = ufbxi_memory_read; stream->skip_fn = ufbxi_memory_skip; + stream->size_fn = ufbxi_memory_size; stream->close_fn = ufbxi_memory_close; stream->user = mem; + ufbxi_end_file_context(&fc, error, true); + return true; } @@ -29253,7 +29755,8 @@ ufbx_abi bool ufbx_is_thread_safe(void) ufbx_abi ufbx_scene *ufbx_load_memory(const void *data, size_t size, const ufbx_load_opts *opts, ufbx_error *error) { ufbxi_check_opts_ptr(ufbx_scene, opts, error); - ufbxi_context uc = { UFBX_ERROR_NONE }; + ufbxi_context uc; // ufbxi_uninit + memset(&uc, 0, sizeof(ufbxi_context)); uc.data_begin = uc.data = (const char *)data; uc.data_size = size; uc.progress_bytes_total = size; @@ -29268,42 +29771,12 @@ ufbx_abi ufbx_scene *ufbx_load_file(const char *filename, const ufbx_load_opts * ufbx_abi ufbx_scene *ufbx_load_file_len(const char *filename, size_t filename_len, const ufbx_load_opts *opts, ufbx_error *error) { ufbxi_check_opts_ptr(ufbx_scene, opts, error); - ufbx_load_opts opts_copy; - if (opts) { - opts_copy = *opts; - } else { - memset(&opts_copy, 0, sizeof(opts_copy)); - opts = &opts_copy; - } - if (opts_copy.filename.length == 0 || opts_copy.filename.data == NULL) { - opts_copy.filename.data = filename; - opts_copy.filename.length = filename_len; - } - - // Defer to `ufbx_load_stream()` if the user so prefers. - if (!opts->open_main_file_with_default && opts->open_file_cb.fn) { - ufbx_stream stream = { 0 }; - if (ufbxi_open_file(&opts->open_file_cb, &stream, filename, filename_len, NULL, NULL, UFBX_OPEN_FILE_MAIN_MODEL)) { - return ufbx_load_stream_prefix(&stream, NULL, 0, &opts_copy, error); - } else { - return ufbxi_load_not_found(filename, filename_len, error); - } - } - - ufbxi_allocator tmp_ator = { 0 }; - ufbx_error tmp_error = { UFBX_ERROR_NONE }; - ufbxi_init_ator(&tmp_error, &tmp_ator, opts ? &opts->temp_allocator : NULL, "filename"); - - FILE *file = ufbxi_fopen(filename, filename_len, &tmp_ator); - if (!file) { - return ufbxi_load_not_found(filename, filename_len, error); - } - - ufbx_scene *scene = ufbx_load_stdio(file, &opts_copy, error); - - fclose(file); - - return scene; + ufbxi_context uc; // ufbxi_uninit + memset(&uc, 0, sizeof(ufbxi_context)); + uc.deferred_load = true; + uc.load_filename = filename; + uc.load_filename_len = filename_len; + return ufbxi_load(&uc, opts, error); } ufbx_abi ufbx_scene *ufbx_load_stdio(void *file_void, const ufbx_load_opts *opts, ufbx_error *error) @@ -29313,37 +29786,24 @@ ufbx_abi ufbx_scene *ufbx_load_stdio(void *file_void, const ufbx_load_opts *opts ufbx_abi ufbx_scene *ufbx_load_stdio_prefix(void *file_void, const void *prefix, size_t prefix_size, const ufbx_load_opts *opts, ufbx_error *error) { - ufbxi_check_opts_ptr(ufbx_scene, opts, error); - FILE *file = (FILE*)file_void; - - ufbxi_context uc = { UFBX_ERROR_NONE }; - uc.data_begin = uc.data = (const char *)prefix; - uc.data_size = prefix_size; - uc.read_fn = &ufbxi_file_read; - uc.skip_fn = &ufbxi_file_skip; - uc.read_user = file; - - if (opts && opts->progress_cb.fn && opts->file_size_estimate == 0) { - uint64_t begin = ufbxi_ftell(file); - if (begin < UINT64_MAX) { - fpos_t pos; // ufbxi_uninit - if (fgetpos(file, &pos) == 0) { - if (fseek(file, 0, SEEK_END) == 0) { - uint64_t end = ufbxi_ftell(file); - if (end != UINT64_MAX && begin < end) { - uc.progress_bytes_total = end - begin; - } - - // Both `rewind()` and `fsetpos()` to reset error and EOF - rewind(file); - fsetpos(file, &pos); - } - } - } - } - - ufbx_scene *scene = ufbxi_load(&uc, opts, error); - return scene; +#if !defined(UFBX_NO_STDIO) + if (!file_void) return NULL; + ufbx_stream stream = { 0 }; + ufbxi_stdio_init(&stream, file_void, false); + return ufbx_load_stream_prefix(&stream, prefix, prefix_size, opts, error); +#else + (void)file_void; + (void)prefix; + (void)prefix_size; + (void)opts; + + ufbxi_context uc; // ufbxi_uninit + memset(&uc, 0, sizeof(ufbxi_context)); + ufbxi_fmt_err_info(&uc.error, "UFBX_NO_STDIO"); + ufbxi_report_err_msg(&uc.error, "UFBX_NO_STDIO", "Feature disabled"); + uc.deferred_failure = true; + return ufbxi_load(&uc, NULL, error); +#endif } ufbx_abi ufbx_scene *ufbx_load_stream(const ufbx_stream *stream, const ufbx_load_opts *opts, ufbx_error *error) @@ -29354,13 +29814,16 @@ ufbx_abi ufbx_scene *ufbx_load_stream(const ufbx_stream *stream, const ufbx_load ufbx_abi ufbx_scene *ufbx_load_stream_prefix(const ufbx_stream *stream, const void *prefix, size_t prefix_size, const ufbx_load_opts *opts, ufbx_error *error) { ufbxi_check_opts_ptr(ufbx_scene, opts, error); - ufbxi_context uc = { UFBX_ERROR_NONE }; + ufbxi_context uc; // ufbxi_uninit + memset(&uc, 0, sizeof(ufbxi_context)); uc.data_begin = uc.data = (const char *)prefix; uc.data_size = prefix_size; uc.read_fn = stream->read_fn; uc.skip_fn = stream->skip_fn; + uc.size_fn = stream->size_fn; uc.close_fn = stream->close_fn; uc.read_user = stream->user; + ufbx_scene *scene = ufbxi_load(&uc, opts, error); return scene; } @@ -29412,9 +29875,10 @@ ufbx_abi ufbxi_noinline size_t ufbx_format_error(char *dst, size_t dst_size, con } size_t stack_size = ufbxi_min_sz(error->stack_size, UFBX_ERROR_STACK_MAX_DEPTH); + int line_width = 6; for (size_t i = 0; i < stack_size; i++) { const ufbx_error_frame *frame = &error->stack[i]; - int num = ufbxi_snprintf(dst + offset, dst_size - offset, "%6u:%s: %s\n", frame->source_line, frame->function.data, frame->description.data); + int num = ufbxi_snprintf(dst + offset, dst_size - offset, "%*u:%s: %s\n", line_width, frame->source_line, frame->function.data, frame->description.data); if (num > 0) offset = ufbxi_min_sz(offset + (size_t)num, dst_size - 1); } @@ -29958,8 +30422,7 @@ ufbx_abi ufbx_anim *ufbx_create_anim(const ufbx_scene *scene, const ufbx_anim_op ufbxi_anim_imp *imp = ac.imp; return &imp->anim; } else { - ufbxi_fix_error_type(&ac.error, "Failed to create anim"); - if (error) *error = ac.error; + ufbxi_fix_error_type(&ac.error, "Failed to create anim", error); ufbxi_buf_free(&ac.result); ufbxi_free_ator(&ac.ator_result); return NULL; @@ -30014,6 +30477,7 @@ ufbx_abi ufbx_baked_anim *ufbx_bake_anim(const ufbx_scene *scene, const ufbx_ani ufbxi_buf_free(&bc.tmp_elements); ufbxi_buf_free(&bc.tmp_props); ufbxi_buf_free(&bc.tmp_bake_stack); + ufbxi_free(&bc.ator_tmp, char, bc.tmp_arr, bc.tmp_arr_size); ufbxi_free_ator(&bc.ator_tmp); if (ok) { @@ -30021,8 +30485,7 @@ ufbx_abi ufbx_baked_anim *ufbx_bake_anim(const ufbx_scene *scene, const ufbx_ani ufbxi_baked_anim_imp *imp = bc.imp; return &imp->bake; } else { - ufbxi_fix_error_type(&bc.error, "Failed to bake anim"); - if (error) *error = bc.error; + ufbxi_fix_error_type(&bc.error, "Failed to bake anim", error); ufbxi_buf_free(&bc.result); ufbxi_free_ator(&bc.ator_result); return NULL; @@ -31046,8 +31509,7 @@ ufbx_abi ufbx_line_curve *ufbx_tessellate_nurbs_curve(const ufbx_nurbs_curve *cu ufbxi_line_curve_imp *imp = tc.imp; return &imp->curve; } else { - ufbxi_fix_error_type(&tc.error, "Failed to tessellate"); - if (error) *error = tc.error; + ufbxi_fix_error_type(&tc.error, "Failed to tessellate", error); ufbxi_buf_free(&tc.result); ufbxi_free_ator(&tc.ator_result); return NULL; @@ -31087,8 +31549,7 @@ ufbx_abi ufbx_mesh *ufbx_tessellate_nurbs_surface(const ufbx_nurbs_surface *surf ufbxi_mesh_imp *imp = tc.imp; return &imp->mesh; } else { - ufbxi_fix_error_type(&tc.error, "Failed to tessellate"); - if (error) *error = tc.error; + ufbxi_fix_error_type(&tc.error, "Failed to tessellate", error); ufbxi_buf_free(&tc.result); ufbxi_free_ator(&tc.ator_result); return NULL; @@ -31230,7 +31691,7 @@ ufbx_abi void ufbx_catch_compute_topology(ufbx_panic *panic, const ufbx_mesh *me ufbx_abi uint32_t ufbx_catch_topo_next_vertex_edge(ufbx_panic *panic, const ufbx_topo_edge *topo, size_t num_topo, uint32_t index) { if (index == UFBX_NO_INDEX) return UFBX_NO_INDEX; - if (ufbxi_panicf(panic, (size_t)index < num_topo, "index (%d) out of bounds (%zu)", index, num_topo)) return UFBX_NO_INDEX; + if (ufbxi_panicf(panic, (size_t)index < num_topo, "index (%u) out of bounds (%zu)", index, num_topo)) return UFBX_NO_INDEX; uint32_t twin = topo[index].twin; if (twin == UFBX_NO_INDEX) return UFBX_NO_INDEX; if (ufbxi_panicf(panic, (size_t)twin < num_topo, "Corrupted topology structure")) return UFBX_NO_INDEX; @@ -31240,7 +31701,7 @@ ufbx_abi uint32_t ufbx_catch_topo_next_vertex_edge(ufbx_panic *panic, const ufbx ufbx_abi uint32_t ufbx_catch_topo_prev_vertex_edge(ufbx_panic *panic, const ufbx_topo_edge *topo, size_t num_topo, uint32_t index) { if (index == UFBX_NO_INDEX) return UFBX_NO_INDEX; - if (ufbxi_panicf(panic, (size_t)index < num_topo, "index (%d) out of bounds (%zu)", index, num_topo)) return UFBX_NO_INDEX; + if (ufbxi_panicf(panic, (size_t)index < num_topo, "index (%u) out of bounds (%zu)", index, num_topo)) return UFBX_NO_INDEX; return topo[topo[index].prev].twin; } @@ -31340,7 +31801,7 @@ ufbx_abi void ufbx_catch_compute_normals(ufbx_panic *panic, const ufbx_mesh *mes for (size_t ix = 0; ix < face.num_indices; ix++) { uint32_t index = normal_indices[face.index_begin + ix]; - if (ufbxi_panicf(panic, index < num_normals, "Normal index (%d) out of bounds (%zu) at %zu", index, num_normals, ix)) return; + if (ufbxi_panicf(panic, index < num_normals, "Normal index (%u) out of bounds (%zu) at %zu", index, num_normals, ix)) return; ufbx_vec3 *n = &normals[index]; *n = ufbxi_add3(*n, normal); @@ -31820,12 +32281,63 @@ ufbx_abi ufbx_audio_clip *ufbx_as_audio_clip(const ufbx_element *element) { retu ufbx_abi ufbx_pose *ufbx_as_pose(const ufbx_element *element) { return element && element->type == UFBX_ELEMENT_POSE ? (ufbx_pose*)element : NULL; } ufbx_abi ufbx_metadata_object *ufbx_as_metadata_object(const ufbx_element *element) { return element && element->type == UFBX_ELEMENT_METADATA_OBJECT ? (ufbx_metadata_object*)element : NULL; } +// -- String API + +ufbx_abi ufbx_prop *ufbx_find_prop(const ufbx_props *props, const char *name) { return ufbx_find_prop_len(props, name, strlen(name)); } +ufbx_abi ufbx_real ufbx_find_real(const ufbx_props *props, const char *name, ufbx_real def) { return ufbx_find_real_len(props, name, strlen(name), def); } +ufbx_abi ufbx_vec3 ufbx_find_vec3(const ufbx_props *props, const char *name, ufbx_vec3 def) { return ufbx_find_vec3_len(props, name, strlen(name), def); } +ufbx_abi int64_t ufbx_find_int(const ufbx_props *props, const char *name, int64_t def) { return ufbx_find_int_len(props, name, strlen(name), def); } +ufbx_abi bool ufbx_find_bool(const ufbx_props *props, const char *name, bool def) { return ufbx_find_bool_len(props, name, strlen(name), def); } +ufbx_abi ufbx_string ufbx_find_string(const ufbx_props *props, const char *name, ufbx_string def) { return ufbx_find_string_len(props, name, strlen(name), def); } +ufbx_abi ufbx_blob ufbx_find_blob(const ufbx_props *props, const char *name, ufbx_blob def) { return ufbx_find_blob_len(props, name, strlen(name), def); } +ufbx_abi ufbx_element *ufbx_find_prop_element(const ufbx_element *element, const char *name, ufbx_element_type type) { return ufbx_find_prop_element_len(element, name, strlen(name), type); } +ufbx_abi ufbx_element *ufbx_find_element(const ufbx_scene *scene, ufbx_element_type type, const char *name) { return ufbx_find_element_len(scene, type, name, strlen(name)); } +ufbx_abi ufbx_node *ufbx_find_node(const ufbx_scene *scene, const char *name) { return ufbx_find_node_len(scene, name, strlen(name)); } +ufbx_abi ufbx_anim_stack *ufbx_find_anim_stack(const ufbx_scene *scene, const char *name) { return ufbx_find_anim_stack_len(scene, name, strlen(name)); } +ufbx_abi ufbx_material *ufbx_find_material(const ufbx_scene *scene, const char *name) { return ufbx_find_material_len(scene, name, strlen(name)); } +ufbx_abi ufbx_anim_prop *ufbx_find_anim_prop(const ufbx_anim_layer *layer, const ufbx_element *element, const char *prop) { return ufbx_find_anim_prop_len(layer, element, prop, strlen(prop)); } +ufbx_abi ufbx_prop ufbx_evaluate_prop(const ufbx_anim *anim, const ufbx_element *element, const char *name, double time) { return ufbx_evaluate_prop_len(anim, element, name, strlen(name), time); } +ufbx_abi ufbx_texture *ufbx_find_prop_texture(const ufbx_material *material, const char *name) { return ufbx_find_prop_texture_len(material, name, strlen(name)); } +ufbx_abi ufbx_string ufbx_find_shader_prop(const ufbx_shader *shader, const char *name) { return ufbx_find_shader_prop_len(shader, name, strlen(name)); } +ufbx_abi ufbx_shader_prop_binding_list ufbx_find_shader_prop_bindings(const ufbx_shader *shader, const char *name) { return ufbx_find_shader_prop_bindings_len(shader, name, strlen(name)); } +ufbx_abi ufbx_shader_texture_input *ufbx_find_shader_texture_input(const ufbx_shader_texture *shader, const char *name) { return ufbx_find_shader_texture_input_len(shader, name, strlen(name)); } +ufbx_abi ufbx_dom_node *ufbx_dom_find(const ufbx_dom_node *parent, const char *name) { return ufbx_dom_find_len(parent, name, strlen(name)); } + +// -- Catch API + +ufbx_abi uint32_t ufbx_triangulate_face(uint32_t *indices, size_t num_indices, const ufbx_mesh *mesh, ufbx_face face) { + return ufbx_catch_triangulate_face(NULL, indices, num_indices, mesh, face); +} +ufbx_abi void ufbx_compute_topology(const ufbx_mesh *mesh, ufbx_topo_edge *topo, size_t num_topo) { + ufbx_catch_compute_topology(NULL, mesh, topo, num_topo); +} +ufbx_abi uint32_t ufbx_topo_next_vertex_edge(const ufbx_topo_edge *topo, size_t num_topo, uint32_t index) { + return ufbx_catch_topo_next_vertex_edge(NULL, topo, num_topo, index); +} +ufbx_abi uint32_t ufbx_topo_prev_vertex_edge(const ufbx_topo_edge *topo, size_t num_topo, uint32_t index) { + return ufbx_catch_topo_prev_vertex_edge(NULL, topo, num_topo, index); +} +ufbx_abi ufbx_vec3 ufbx_get_weighted_face_normal(const ufbx_vertex_vec3 *positions, ufbx_face face) { + return ufbx_catch_get_weighted_face_normal(NULL, positions, face); +} + #ifdef __cplusplus } #endif #endif +#if defined(UFBX_STRING_PREFIX) + #undef strlen + #undef memcpy + #undef memmove + #undef memset + #undef memchr + #undef memcmp + #undef strcmp + #undef strncmp +#endif + #if defined(_MSC_VER) #pragma warning(pop) #elif defined(__clang__) diff --git a/thirdparty/ufbx/ufbx.h b/thirdparty/ufbx/ufbx.h index 8d856edaad09..1fc9a103655b 100644 --- a/thirdparty/ufbx/ufbx.h +++ b/thirdparty/ufbx/ufbx.h @@ -9,10 +9,11 @@ // -- Headers -#include -#include -#include -#include +#if !defined(UFBX_NO_LIBC_TYPES) + #include + #include + #include +#endif // -- Platform @@ -99,7 +100,7 @@ // make sure that it is also used within `ufbx.c`. // Defining `UFBX_NO_ASSERT` to any value disables assertions. #ifndef ufbx_assert - #if defined(UFBX_NO_ASSERT) + #if defined(UFBX_NO_ASSERT) || defined(UFBX_NO_LIBC) #define ufbx_assert(cond) (void)0 #else #include @@ -266,7 +267,7 @@ struct ufbx_converter { }; // `ufbx_source_version` contains the version of the corresponding source file. // HINT: The version can be compared numerically to the result of `ufbx_pack_version()`, // for example `#if UFBX_VERSION >= ufbx_pack_version(0, 12, 0)`. -#define UFBX_HEADER_VERSION ufbx_pack_version(0, 14, 3) +#define UFBX_HEADER_VERSION ufbx_pack_version(0, 15, 0) #define UFBX_VERSION UFBX_HEADER_VERSION // -- Basic types @@ -3984,12 +3985,17 @@ typedef size_t ufbx_read_fn(void *user, void *data, size_t size); // Skip `size` bytes in the file. typedef bool ufbx_skip_fn(void *user, size_t size); +// Get the size of the file. +// Return `0` if unknown, `UINT64_MAX` if error. +typedef uint64_t ufbx_size_fn(void *user); + // Close the file typedef void ufbx_close_fn(void *user); typedef struct ufbx_stream { ufbx_read_fn *read_fn; // < Required ufbx_skip_fn *skip_fn; // < Optional: Will use `read_fn()` if missing + ufbx_size_fn *size_fn; // < Optional ufbx_close_fn *close_fn; // < Optional // Context passed to other functions @@ -4006,13 +4012,17 @@ typedef enum ufbx_open_file_type UFBX_ENUM_REPR { UFBX_ENUM_TYPE(ufbx_open_file_type, UFBX_OPEN_FILE_TYPE, UFBX_OPEN_FILE_OBJ_MTL); +typedef uintptr_t ufbx_open_file_context; + typedef struct ufbx_open_file_info { + // Context that can be passed to the following functions to use a shared allocator: + // ufbx_open_file_ctx() + // ufbx_open_memory_ctx() + ufbx_open_file_context context; + // Kind of file to load. ufbx_open_file_type type; - // Temporary allocator to use. - ufbx_allocator temp_allocator; - // Original filename in the file, not resolved or UTF-8 encoded. // NOTE: Not necessarily NULL-terminated! ufbx_blob original_filename; @@ -4030,6 +4040,19 @@ typedef struct ufbx_open_file_cb { (stream, path, path_len, info)) } ufbx_open_file_cb; +// Options for `ufbx_open_file()`. +typedef struct ufbx_open_file_opts { + uint32_t _begin_zero; + + // Allocator to allocate the memory with. + ufbx_allocator_opts allocator; + + // The filename is guaranteed to be NULL-terminated. + ufbx_unsafe bool filename_null_terminated; + + uint32_t _end_zero; +} ufbx_open_file_opts; + // Memory stream options typedef void ufbx_close_memory_fn(void *user, void *data, size_t data_size); @@ -5092,6 +5115,7 @@ ufbx_abi_data const size_t ufbx_element_type_size[UFBX_ELEMENT_TYPE_COUNT]; // Version of the source file, comparable to `UFBX_HEADER_VERSION` ufbx_abi_data const uint32_t ufbx_source_version; + // Practically always `true` (see below), if not you need to be careful with threads. // // Guaranteed to be `true` in _any_ of the following conditions: @@ -5160,23 +5184,23 @@ ufbx_abi size_t ufbx_format_error(char *dst, size_t dst_size, const ufbx_error * // Find a property `name` from `props`, returns `NULL` if not found. // Searches through `ufbx_props.defaults` as well. ufbx_abi ufbx_prop *ufbx_find_prop_len(const ufbx_props *props, const char *name, size_t name_len); -ufbx_inline ufbx_prop *ufbx_find_prop(const ufbx_props *props, const char *name) { return ufbx_find_prop_len(props, name, strlen(name));} +ufbx_abi ufbx_prop *ufbx_find_prop(const ufbx_props *props, const char *name); // Utility functions for finding the value of a property, returns `def` if not found. // NOTE: For `ufbx_string` you need to ensure the lifetime of the default is // sufficient as no copy is made. ufbx_abi ufbx_real ufbx_find_real_len(const ufbx_props *props, const char *name, size_t name_len, ufbx_real def); -ufbx_inline ufbx_real ufbx_find_real(const ufbx_props *props, const char *name, ufbx_real def) { return ufbx_find_real_len(props, name, strlen(name), def); } +ufbx_abi ufbx_real ufbx_find_real(const ufbx_props *props, const char *name, ufbx_real def); ufbx_abi ufbx_vec3 ufbx_find_vec3_len(const ufbx_props *props, const char *name, size_t name_len, ufbx_vec3 def); -ufbx_inline ufbx_vec3 ufbx_find_vec3(const ufbx_props *props, const char *name, ufbx_vec3 def) { return ufbx_find_vec3_len(props, name, strlen(name), def); } +ufbx_abi ufbx_vec3 ufbx_find_vec3(const ufbx_props *props, const char *name, ufbx_vec3 def); ufbx_abi int64_t ufbx_find_int_len(const ufbx_props *props, const char *name, size_t name_len, int64_t def); -ufbx_inline int64_t ufbx_find_int(const ufbx_props *props, const char *name, int64_t def) { return ufbx_find_int_len(props, name, strlen(name), def); } +ufbx_abi int64_t ufbx_find_int(const ufbx_props *props, const char *name, int64_t def); ufbx_abi bool ufbx_find_bool_len(const ufbx_props *props, const char *name, size_t name_len, bool def); -ufbx_inline bool ufbx_find_bool(const ufbx_props *props, const char *name, bool def) { return ufbx_find_bool_len(props, name, strlen(name), def); } +ufbx_abi bool ufbx_find_bool(const ufbx_props *props, const char *name, bool def); ufbx_abi ufbx_string ufbx_find_string_len(const ufbx_props *props, const char *name, size_t name_len, ufbx_string def); -ufbx_inline ufbx_string ufbx_find_string(const ufbx_props *props, const char *name, ufbx_string def) { return ufbx_find_string_len(props, name, strlen(name), def); } +ufbx_abi ufbx_string ufbx_find_string(const ufbx_props *props, const char *name, ufbx_string def); ufbx_abi ufbx_blob ufbx_find_blob_len(const ufbx_props *props, const char *name, size_t name_len, ufbx_blob def); -ufbx_inline ufbx_blob ufbx_find_blob(const ufbx_props *props, const char *name, ufbx_blob def) { return ufbx_find_blob_len(props, name, strlen(name), def); } +ufbx_abi ufbx_blob ufbx_find_blob(const ufbx_props *props, const char *name, ufbx_blob def); // Find property in `props` with concatendated `parts[num_parts]`. ufbx_abi ufbx_prop *ufbx_find_prop_concat(const ufbx_props *props, const ufbx_string *parts, size_t num_parts); @@ -5186,30 +5210,30 @@ ufbx_abi ufbx_element *ufbx_get_prop_element(const ufbx_element *element, const // Find an element connected to a property by name. ufbx_abi ufbx_element *ufbx_find_prop_element_len(const ufbx_element *element, const char *name, size_t name_len, ufbx_element_type type); -ufbx_inline ufbx_element *ufbx_find_prop_element(const ufbx_element *element, const char *name, ufbx_element_type type) { return ufbx_find_prop_element_len(element, name, strlen(name), type); } +ufbx_abi ufbx_element *ufbx_find_prop_element(const ufbx_element *element, const char *name, ufbx_element_type type); // Find any element of type `type` in `scene` by `name`. // For example if you want to find `ufbx_material` named `Mat`: // (ufbx_material*)ufbx_find_element(scene, UFBX_ELEMENT_MATERIAL, "Mat"); ufbx_abi ufbx_element *ufbx_find_element_len(const ufbx_scene *scene, ufbx_element_type type, const char *name, size_t name_len); -ufbx_inline ufbx_element *ufbx_find_element(const ufbx_scene *scene, ufbx_element_type type, const char *name) { return ufbx_find_element_len(scene, type, name, strlen(name)); } +ufbx_abi ufbx_element *ufbx_find_element(const ufbx_scene *scene, ufbx_element_type type, const char *name); // Find node in `scene` by `name` (shorthand for `ufbx_find_element(UFBX_ELEMENT_NODE)`). ufbx_abi ufbx_node *ufbx_find_node_len(const ufbx_scene *scene, const char *name, size_t name_len); -ufbx_inline ufbx_node *ufbx_find_node(const ufbx_scene *scene, const char *name) { return ufbx_find_node_len(scene, name, strlen(name)); } +ufbx_abi ufbx_node *ufbx_find_node(const ufbx_scene *scene, const char *name); // Find an animation stack in `scene` by `name` (shorthand for `ufbx_find_element(UFBX_ELEMENT_ANIM_STACK)`) ufbx_abi ufbx_anim_stack *ufbx_find_anim_stack_len(const ufbx_scene *scene, const char *name, size_t name_len); -ufbx_inline ufbx_anim_stack *ufbx_find_anim_stack(const ufbx_scene *scene, const char *name) { return ufbx_find_anim_stack_len(scene, name, strlen(name)); } +ufbx_abi ufbx_anim_stack *ufbx_find_anim_stack(const ufbx_scene *scene, const char *name); // Find a material in `scene` by `name` (shorthand for `ufbx_find_element(UFBX_ELEMENT_MATERIAL)`). ufbx_abi ufbx_material *ufbx_find_material_len(const ufbx_scene *scene, const char *name, size_t name_len); -ufbx_inline ufbx_material *ufbx_find_material(const ufbx_scene *scene, const char *name) { return ufbx_find_material_len(scene, name, strlen(name)); } +ufbx_abi ufbx_material *ufbx_find_material(const ufbx_scene *scene, const char *name); // Find a single animated property `prop` of `element` in `layer`. // Returns `NULL` if not found. ufbx_abi ufbx_anim_prop *ufbx_find_anim_prop_len(const ufbx_anim_layer *layer, const ufbx_element *element, const char *prop, size_t prop_len); -ufbx_inline ufbx_anim_prop *ufbx_find_anim_prop(const ufbx_anim_layer *layer, const ufbx_element *element, const char *prop) { return ufbx_find_anim_prop_len(layer, element, prop, strlen(prop)); } +ufbx_abi ufbx_anim_prop *ufbx_find_anim_prop(const ufbx_anim_layer *layer, const ufbx_element *element, const char *prop); // Find all animated properties of `element` in `layer`. ufbx_abi ufbx_anim_prop_list ufbx_find_anim_props(const ufbx_anim_layer *layer, const ufbx_element *element); @@ -5228,16 +5252,18 @@ ufbx_abi ufbx_matrix ufbx_get_compatible_matrix_for_normals(const ufbx_node *nod // but the rest can be uninitialized. ufbx_abi ptrdiff_t ufbx_inflate(void *dst, size_t dst_size, const ufbx_inflate_input *input, ufbx_inflate_retain *retain); -// Open a `ufbx_stream` from a file. -// Use `path_len == SIZE_MAX` for NULL terminated string. -ufbx_abi bool ufbx_open_file(ufbx_stream *stream, const char *path, size_t path_len); - // Same as `ufbx_open_file()` but compatible with the callback in `ufbx_open_file_fn`. // The `user` parameter is actually not used here. ufbx_abi bool ufbx_default_open_file(void *user, ufbx_stream *stream, const char *path, size_t path_len, const ufbx_open_file_info *info); +// Open a `ufbx_stream` from a file. +// Use `path_len == SIZE_MAX` for NULL terminated string. +ufbx_abi bool ufbx_open_file(ufbx_stream *stream, const char *path, size_t path_len, const ufbx_open_file_opts *opts, ufbx_error *error); +ufbx_unsafe ufbx_abi bool ufbx_open_file_ctx(ufbx_stream *stream, ufbx_open_file_context ctx, const char *path, size_t path_len, const ufbx_open_file_opts *opts, ufbx_error *error); + // NOTE: Uses the default ufbx allocator! ufbx_abi bool ufbx_open_memory(ufbx_stream *stream, const void *data, size_t data_size, const ufbx_open_memory_opts *opts, ufbx_error *error); +ufbx_unsafe ufbx_abi bool ufbx_open_memory_ctx(ufbx_stream *stream, ufbx_open_file_context ctx, const void *data, size_t data_size, const ufbx_open_memory_opts *opts, ufbx_error *error); // Animation evaluation @@ -5252,9 +5278,7 @@ ufbx_abi ufbx_vec3 ufbx_evaluate_anim_value_vec3(const ufbx_anim_value *anim_val // Evaluate an animated property `name` from `element` at `time`. // NOTE: If the property is not found it will have the flag `UFBX_PROP_FLAG_NOT_FOUND`. ufbx_abi ufbx_prop ufbx_evaluate_prop_len(const ufbx_anim *anim, const ufbx_element *element, const char *name, size_t name_len, double time); -ufbx_inline ufbx_prop ufbx_evaluate_prop(const ufbx_anim *anim, const ufbx_element *element, const char *name, double time) { - return ufbx_evaluate_prop_len(anim, element, name, strlen(name), time); -} +ufbx_abi ufbx_prop ufbx_evaluate_prop(const ufbx_anim *anim, const ufbx_element *element, const char *name, double time); // Evaluate all _animated_ properties of `element`. // HINT: This function returns an `ufbx_props` structure with the original properties as @@ -5351,27 +5375,19 @@ ufbx_abi ufbx_bone_pose *ufbx_get_bone_pose(const ufbx_pose *pose, const ufbx_no // Find a texture for a given material FBX property. ufbx_abi ufbx_texture *ufbx_find_prop_texture_len(const ufbx_material *material, const char *name, size_t name_len); -ufbx_inline ufbx_texture *ufbx_find_prop_texture(const ufbx_material *material, const char *name) { - return ufbx_find_prop_texture_len(material, name, strlen(name)); -} +ufbx_abi ufbx_texture *ufbx_find_prop_texture(const ufbx_material *material, const char *name); // Find a texture for a given shader property. ufbx_abi ufbx_string ufbx_find_shader_prop_len(const ufbx_shader *shader, const char *name, size_t name_len); -ufbx_inline ufbx_string ufbx_find_shader_prop(const ufbx_shader *shader, const char *name) { - return ufbx_find_shader_prop_len(shader, name, strlen(name)); -} +ufbx_abi ufbx_string ufbx_find_shader_prop(const ufbx_shader *shader, const char *name); // Map from a shader property to material property. ufbx_abi ufbx_shader_prop_binding_list ufbx_find_shader_prop_bindings_len(const ufbx_shader *shader, const char *name, size_t name_len); -ufbx_inline ufbx_shader_prop_binding_list ufbx_find_shader_prop_bindings(const ufbx_shader *shader, const char *name) { - return ufbx_find_shader_prop_bindings_len(shader, name, strlen(name)); -} +ufbx_abi ufbx_shader_prop_binding_list ufbx_find_shader_prop_bindings(const ufbx_shader *shader, const char *name); // Find an input in a shader texture. ufbx_abi ufbx_shader_texture_input *ufbx_find_shader_texture_input_len(const ufbx_shader_texture *shader, const char *name, size_t name_len); -ufbx_inline ufbx_shader_texture_input *ufbx_find_shader_texture_input(const ufbx_shader_texture *shader, const char *name) { - return ufbx_find_shader_texture_input_len(shader, name, strlen(name)); -} +ufbx_abi ufbx_shader_texture_input *ufbx_find_shader_texture_input(const ufbx_shader_texture *shader, const char *name); // Math @@ -5471,37 +5487,27 @@ ufbx_abi uint32_t ufbx_find_face_index(ufbx_mesh *mesh, size_t index); // NOTE: You need to space for `(face.num_indices - 2) * 3 - 1` indices! // HINT: Using `ufbx_mesh.max_face_triangles * 3` is always safe. ufbx_abi uint32_t ufbx_catch_triangulate_face(ufbx_panic *panic, uint32_t *indices, size_t num_indices, const ufbx_mesh *mesh, ufbx_face face); -ufbx_inline uint32_t ufbx_triangulate_face(uint32_t *indices, size_t num_indices, const ufbx_mesh *mesh, ufbx_face face) { - return ufbx_catch_triangulate_face(NULL, indices, num_indices, mesh, face); -} +ufbx_abi uint32_t ufbx_triangulate_face(uint32_t *indices, size_t num_indices, const ufbx_mesh *mesh, ufbx_face face); // Generate the half-edge representation of `mesh` to `topo[mesh->num_indices]` ufbx_abi void ufbx_catch_compute_topology(ufbx_panic *panic, const ufbx_mesh *mesh, ufbx_topo_edge *topo, size_t num_topo); -ufbx_inline void ufbx_compute_topology(const ufbx_mesh *mesh, ufbx_topo_edge *topo, size_t num_topo) { - ufbx_catch_compute_topology(NULL, mesh, topo, num_topo); -} +ufbx_abi void ufbx_compute_topology(const ufbx_mesh *mesh, ufbx_topo_edge *topo, size_t num_topo); // Get the next/previous edge around a vertex // NOTE: Does not return the half-edge on the opposite side (ie. `topo[index].twin`) // Get the next half-edge in `topo`. ufbx_abi uint32_t ufbx_catch_topo_next_vertex_edge(ufbx_panic *panic, const ufbx_topo_edge *topo, size_t num_topo, uint32_t index); -ufbx_inline uint32_t ufbx_topo_next_vertex_edge(const ufbx_topo_edge *topo, size_t num_topo, uint32_t index) { - return ufbx_catch_topo_next_vertex_edge(NULL, topo, num_topo, index); -} +ufbx_abi uint32_t ufbx_topo_next_vertex_edge(const ufbx_topo_edge *topo, size_t num_topo, uint32_t index); // Get the previous half-edge in `topo`. ufbx_abi uint32_t ufbx_catch_topo_prev_vertex_edge(ufbx_panic *panic, const ufbx_topo_edge *topo, size_t num_topo, uint32_t index); -ufbx_inline uint32_t ufbx_topo_prev_vertex_edge(const ufbx_topo_edge *topo, size_t num_topo, uint32_t index) { - return ufbx_catch_topo_prev_vertex_edge(NULL, topo, num_topo, index); -} +ufbx_abi uint32_t ufbx_topo_prev_vertex_edge(const ufbx_topo_edge *topo, size_t num_topo, uint32_t index); // Calculate a normal for a given face. // The returned normal is weighted by face area. ufbx_abi ufbx_vec3 ufbx_catch_get_weighted_face_normal(ufbx_panic *panic, const ufbx_vertex_vec3 *positions, ufbx_face face); -ufbx_inline ufbx_vec3 ufbx_get_weighted_face_normal(const ufbx_vertex_vec3 *positions, ufbx_face face) { - return ufbx_catch_get_weighted_face_normal(NULL, positions, face); -} +ufbx_abi ufbx_vec3 ufbx_get_weighted_face_normal(const ufbx_vertex_vec3 *positions, ufbx_face face); // Generate indices for normals from the topology. // Respects smoothing groups. @@ -5558,7 +5564,7 @@ ufbx_abi size_t ufbx_sample_geometry_cache_vec3(const ufbx_cache_channel *channe // Find a DOM node given a name. ufbx_abi ufbx_dom_node *ufbx_dom_find_len(const ufbx_dom_node *parent, const char *name, size_t name_len); -ufbx_inline ufbx_dom_node *ufbx_dom_find(const ufbx_dom_node *parent, const char *name) { return ufbx_dom_find_len(parent, name, strlen(name)); } +ufbx_abi ufbx_dom_node *ufbx_dom_find(const ufbx_dom_node *parent, const char *name); // Utility From 6cf1d3c13e4ecc0f50f689e9001be954b7ae6eff Mon Sep 17 00:00:00 2001 From: "K. S. Ernest (iFire) Lee" Date: Tue, 3 Dec 2024 04:12:36 -0800 Subject: [PATCH 14/33] Print better manifold errors and avoid crash on non manifold csg input. * Manifold does not have a snap property. * Tolerance means simplification amount. * CSG snap has been removed * Add better error messages. * Verbose print manifold meshgl64 properties as json. * Update manifold for error fixes --- modules/csg/csg_shape.cpp | 134 +++- modules/csg/csg_shape.h | 2 + modules/csg/doc_classes/CSGShape3D.xml | 4 +- thirdparty/README.md | 2 +- thirdparty/manifold/src/constructors.cpp | 2 +- thirdparty/manifold/src/csg_tree.cpp | 573 +++++++++--------- thirdparty/manifold/src/csg_tree.h | 25 +- thirdparty/manifold/src/impl.cpp | 11 +- .../{include/manifold => src}/iters.h | 0 thirdparty/manifold/src/parallel.h | 2 +- thirdparty/manifold/src/properties.cpp | 9 +- 11 files changed, 443 insertions(+), 321 deletions(-) rename thirdparty/manifold/{include/manifold => src}/iters.h (100%) diff --git a/modules/csg/csg_shape.cpp b/modules/csg/csg_shape.cpp index cc4595929f31..eed5a7673792 100644 --- a/modules/csg/csg_shape.cpp +++ b/modules/csg/csg_shape.cpp @@ -30,6 +30,9 @@ #include "csg_shape.h" +#ifdef DEV_ENABLED +#include "core/io/json.h" +#endif // DEV_ENABLED #include "core/math/geometry_2d.h" #include @@ -142,6 +145,7 @@ bool CSGShape3D::is_root_shape() const { return !parent_shape; } +#ifndef DISABLE_DEPRECATED void CSGShape3D::set_snap(float p_snap) { if (snap == p_snap) { return; @@ -154,6 +158,7 @@ void CSGShape3D::set_snap(float p_snap) { float CSGShape3D::get_snap() const { return snap; } +#endif // DISABLE_DEPRECATED void CSGShape3D::_make_dirty(bool p_parent_removing) { if ((p_parent_removing || is_root_shape()) && !dirty) { @@ -232,13 +237,111 @@ static void _unpack_manifold( r_mesh_merge->_regen_face_aabbs(); } +// Errors matching `thirdparty/manifold/include/manifold/manifold.h`. +static String manifold_error_to_string(const manifold::Manifold::Error &p_error) { + switch (p_error) { + case manifold::Manifold::Error::NoError: + return "No Error"; + case manifold::Manifold::Error::NonFiniteVertex: + return "Non Finite Vertex"; + case manifold::Manifold::Error::NotManifold: + return "Not Manifold"; + case manifold::Manifold::Error::VertexOutOfBounds: + return "Vertex Out Of Bounds"; + case manifold::Manifold::Error::PropertiesWrongLength: + return "Properties Wrong Length"; + case manifold::Manifold::Error::MissingPositionProperties: + return "Missing Position Properties"; + case manifold::Manifold::Error::MergeVectorsDifferentLengths: + return "Merge Vectors Different Lengths"; + case manifold::Manifold::Error::MergeIndexOutOfBounds: + return "Merge Index Out Of Bounds"; + case manifold::Manifold::Error::TransformWrongLength: + return "Transform Wrong Length"; + case manifold::Manifold::Error::RunIndexWrongLength: + return "Run Index Wrong Length"; + case manifold::Manifold::Error::FaceIDWrongLength: + return "Face ID Wrong Length"; + case manifold::Manifold::Error::InvalidConstruction: + return "Invalid Construction"; + default: + return "Unknown Error"; + } +} + +#ifdef DEV_ENABLED +static String _export_meshgl_as_json(const manifold::MeshGL64 &p_mesh) { + Dictionary mesh_dict; + mesh_dict["numProp"] = p_mesh.numProp; + + Array vert_properties; + for (const double &val : p_mesh.vertProperties) { + vert_properties.append(val); + } + mesh_dict["vertProperties"] = vert_properties; + + Array tri_verts; + for (const uint64_t &val : p_mesh.triVerts) { + tri_verts.append(val); + } + mesh_dict["triVerts"] = tri_verts; + + Array merge_from_vert; + for (const uint64_t &val : p_mesh.mergeFromVert) { + merge_from_vert.append(val); + } + mesh_dict["mergeFromVert"] = merge_from_vert; + + Array merge_to_vert; + for (const uint64_t &val : p_mesh.mergeToVert) { + merge_to_vert.append(val); + } + mesh_dict["mergeToVert"] = merge_to_vert; + + Array run_index; + for (const uint64_t &val : p_mesh.runIndex) { + run_index.append(val); + } + mesh_dict["runIndex"] = run_index; + + Array run_original_id; + for (const uint32_t &val : p_mesh.runOriginalID) { + run_original_id.append(val); + } + mesh_dict["runOriginalID"] = run_original_id; + + Array run_transform; + for (const double &val : p_mesh.runTransform) { + run_transform.append(val); + } + mesh_dict["runTransform"] = run_transform; + + Array face_id; + for (const uint64_t &val : p_mesh.faceID) { + face_id.append(val); + } + mesh_dict["faceID"] = face_id; + + Array halfedge_tangent; + for (const double &val : p_mesh.halfedgeTangent) { + halfedge_tangent.append(val); + } + mesh_dict["halfedgeTangent"] = halfedge_tangent; + + mesh_dict["tolerance"] = p_mesh.tolerance; + + String json_string = JSON::stringify(mesh_dict); + return json_string; +} +#endif // DEV_ENABLED + static void _pack_manifold( const CSGBrush *const p_mesh_merge, manifold::Manifold &r_manifold, HashMap> &p_mesh_materials, - float p_snap) { + CSGShape3D *p_csg_shape) { ERR_FAIL_NULL_MSG(p_mesh_merge, "p_mesh_merge is null"); - + ERR_FAIL_NULL_MSG(p_csg_shape, "p_shape is null"); HashMap> faces_by_material; for (int face_i = 0; face_i < p_mesh_merge->faces.size(); face_i++) { const CSGBrush::Face &face = p_mesh_merge->faces[face_i]; @@ -246,7 +349,6 @@ static void _pack_manifold( } manifold::MeshGL64 mesh; - mesh.tolerance = p_snap; mesh.numProp = MANIFOLD_PROPERTY_MAX; mesh.runOriginalID.reserve(faces_by_material.size()); mesh.runIndex.reserve(faces_by_material.size() + 1); @@ -291,12 +393,22 @@ static void _pack_manifold( } // runIndex needs an explicit end value. mesh.runIndex.push_back(mesh.triVerts.size()); + mesh.tolerance = 2 * FLT_EPSILON; ERR_FAIL_COND_MSG(mesh.vertProperties.size() % mesh.numProp != 0, "Invalid vertex properties size."); mesh.Merge(); +#ifdef DEV_ENABLED + print_verbose(_export_meshgl_as_json(mesh)); +#endif // DEV_ENABLED r_manifold = manifold::Manifold(mesh); - manifold::Manifold::Error err = r_manifold.Status(); - if (err != manifold::Manifold::Error::NoError) { - print_error(String("Manifold creation from mesh failed:" + itos((int)err))); + manifold::Manifold::Error error = r_manifold.Status(); + if (error == manifold::Manifold::Error::NoError) { + return; + } + if (p_csg_shape->get_owner()) { + NodePath path = p_csg_shape->get_owner()->get_path_to(p_csg_shape, true); + print_error(vformat("CSGShape3D manifold creation from mesh failed at %s: %s.", path, manifold_error_to_string(error))); + } else { + print_error(vformat("CSGShape3D manifold creation from mesh failed at .: %s.", manifold_error_to_string(error))); } } @@ -330,7 +442,7 @@ CSGBrush *CSGShape3D::_get_brush() { CSGBrush *n = _build_brush(); HashMap> mesh_materials; manifold::Manifold root_manifold; - _pack_manifold(n, root_manifold, mesh_materials, get_snap()); + _pack_manifold(n, root_manifold, mesh_materials, this); manifold::OpType current_op = ManifoldOperation::convert_csg_op(get_operation()); std::vector manifolds; manifolds.push_back(root_manifold); @@ -346,7 +458,7 @@ CSGBrush *CSGShape3D::_get_brush() { CSGBrush transformed_brush; transformed_brush.copy_from(*child_brush, child->get_transform()); manifold::Manifold child_manifold; - _pack_manifold(&transformed_brush, child_manifold, mesh_materials, get_snap()); + _pack_manifold(&transformed_brush, child_manifold, mesh_materials, child); manifold::OpType child_operation = ManifoldOperation::convert_csg_op(child->get_operation()); if (child_operation != current_op) { manifold::Manifold result = manifold::Manifold::BatchBoolean(manifolds, current_op); @@ -834,8 +946,10 @@ void CSGShape3D::_bind_methods() { ClassDB::bind_method(D_METHOD("set_operation", "operation"), &CSGShape3D::set_operation); ClassDB::bind_method(D_METHOD("get_operation"), &CSGShape3D::get_operation); +#ifndef DISABLE_DEPRECATED ClassDB::bind_method(D_METHOD("set_snap", "snap"), &CSGShape3D::set_snap); ClassDB::bind_method(D_METHOD("get_snap"), &CSGShape3D::get_snap); +#endif // DISABLE_DEPRECATED ClassDB::bind_method(D_METHOD("set_use_collision", "operation"), &CSGShape3D::set_use_collision); ClassDB::bind_method(D_METHOD("is_using_collision"), &CSGShape3D::is_using_collision); @@ -864,7 +978,9 @@ void CSGShape3D::_bind_methods() { ClassDB::bind_method(D_METHOD("bake_collision_shape"), &CSGShape3D::bake_collision_shape); ADD_PROPERTY(PropertyInfo(Variant::INT, "operation", PROPERTY_HINT_ENUM, "Union,Intersection,Subtraction"), "set_operation", "get_operation"); - ADD_PROPERTY(PropertyInfo(Variant::FLOAT, "snap", PROPERTY_HINT_RANGE, "0.000001,1,0.000001,suffix:m"), "set_snap", "get_snap"); +#ifndef DISABLE_DEPRECATED + ADD_PROPERTY(PropertyInfo(Variant::FLOAT, "snap", PROPERTY_HINT_RANGE, "0.000001,1,0.000001,suffix:m", PROPERTY_USAGE_NONE), "set_snap", "get_snap"); +#endif // DISABLE_DEPRECATED ADD_PROPERTY(PropertyInfo(Variant::BOOL, "calculate_tangents"), "set_calculate_tangents", "is_calculating_tangents"); ADD_GROUP("Collision", "collision_"); diff --git a/modules/csg/csg_shape.h b/modules/csg/csg_shape.h index 8f23ae2f9e2c..9aa612885aa8 100644 --- a/modules/csg/csg_shape.h +++ b/modules/csg/csg_shape.h @@ -155,8 +155,10 @@ class CSGShape3D : public GeometryInstance3D { void set_collision_priority(real_t p_priority); real_t get_collision_priority() const; +#ifndef DISABLE_DEPRECATED void set_snap(float p_snap); float get_snap() const; +#endif // DISABLE_DEPRECATED void set_calculate_tangents(bool p_calculate_tangents); bool is_calculating_tangents() const; diff --git a/modules/csg/doc_classes/CSGShape3D.xml b/modules/csg/doc_classes/CSGShape3D.xml index ac62d8dd83dc..8ea471e62d83 100644 --- a/modules/csg/doc_classes/CSGShape3D.xml +++ b/modules/csg/doc_classes/CSGShape3D.xml @@ -89,8 +89,8 @@ The operation that is performed on this shape. This is ignored for the first CSG child node as the operation is between this node and the previous child of this nodes parent. - - Snap makes the mesh vertices snap to a given distance so that the faces of two meshes can be perfectly aligned. A lower value results in greater precision but may be harder to adjust. The top-level CSG shape's snap value is used for the entire CSG tree. + + This property does nothing. Adds a collision shape to the physics engine for our CSG shape. This will always act like a static body. Note that the collision shape is still active even if the CSG shape itself is hidden. See also [member collision_mask] and [member collision_priority]. diff --git a/thirdparty/README.md b/thirdparty/README.md index 79d7e5b7b6ad..b70738c21a80 100644 --- a/thirdparty/README.md +++ b/thirdparty/README.md @@ -549,7 +549,7 @@ in the MSVC debugger. ## manifold - Upstream: https://github.com/elalish/manifold -- Version: 3.0.0 (5d127e57fbfb89225a8e905d0d914ccc86c139c8, 2024) +- Version: master (36035428bc32302a9d7c9ee1ecc833fb8394a2a3, 2024) - License: Apache 2.0 File extracted from upstream source: diff --git a/thirdparty/manifold/src/constructors.cpp b/thirdparty/manifold/src/constructors.cpp index 17df744ee965..2b3e528c6bbe 100644 --- a/thirdparty/manifold/src/constructors.cpp +++ b/thirdparty/manifold/src/constructors.cpp @@ -436,7 +436,7 @@ Manifold Manifold::Compose(const std::vector& manifolds) { for (const auto& manifold : manifolds) { children.push_back(manifold.pNode_->ToLeafNode()); } - return Manifold(std::make_shared(CsgLeafNode::Compose(children))); + return Manifold(CsgLeafNode::Compose(children)); } /** diff --git a/thirdparty/manifold/src/csg_tree.cpp b/thirdparty/manifold/src/csg_tree.cpp index 8f98fe29895c..a05fce3aa6b6 100644 --- a/thirdparty/manifold/src/csg_tree.cpp +++ b/thirdparty/manifold/src/csg_tree.cpp @@ -19,7 +19,6 @@ #endif #include -#include #include "./boolean3.h" #include "./csg_tree.h" @@ -31,66 +30,10 @@ constexpr int kParallelThreshold = 4096; namespace { using namespace manifold; -struct Transform4x3 { - mat3x4 transform; - - vec3 operator()(vec3 position) const { - return transform * vec4(position, 1.0); - } -}; - -struct UpdateHalfedge { - const int nextVert; - const int nextEdge; - const int nextFace; - - Halfedge operator()(Halfedge edge) { - edge.startVert += nextVert; - edge.endVert += nextVert; - edge.pairedHalfedge += nextEdge; - return edge; - } -}; - -struct UpdateTriProp { - const int nextProp; - - ivec3 operator()(ivec3 tri) { - tri += nextProp; - return tri; - } -}; - -struct UpdateMeshIDs { - const int offset; - - TriRef operator()(TriRef ref) { - ref.meshID += offset; - return ref; - } -}; - -struct CheckOverlap { - VecView boxes; - const size_t i; - bool operator()(size_t j) { return boxes[i].DoesOverlap(boxes[j]); } -}; - -using SharedImpl = std::variant, - std::shared_ptr>; -struct GetImplPtr { - const Manifold::Impl *operator()(const SharedImpl &p) { - if (std::holds_alternative>(p)) { - return std::get_if>(&p)->get(); - } else { - return std::get_if>(&p)->get(); - } - }; -}; - struct MeshCompare { - bool operator()(const SharedImpl &a, const SharedImpl &b) { - return GetImplPtr()(a)->NumVert() < GetImplPtr()(b)->NumVert(); + bool operator()(const std::shared_ptr &a, + const std::shared_ptr &b) { + return a->GetImpl()->NumVert() < b->GetImpl()->NumVert(); } }; @@ -99,11 +42,12 @@ namespace manifold { std::shared_ptr CsgNode::Boolean( const std::shared_ptr &second, OpType op) { - if (auto opNode = std::dynamic_pointer_cast(second)) { + if (second->GetNodeType() != CsgNodeType::Leaf) { // "this" is not a CsgOpNode (which overrides Boolean), but if "second" is // and the operation is commutative, we let it built the tree. if ((op == OpType::Add || op == OpType::Intersect)) { - return opNode->Boolean(shared_from_this(), op); + return std::static_pointer_cast(second)->Boolean( + shared_from_this(), op); } } std::vector> children({shared_from_this(), second}); @@ -154,8 +98,6 @@ std::shared_ptr CsgLeafNode::GetImpl() const { return pImpl_; } -mat3x4 CsgLeafNode::GetTransform() const { return transform_; } - std::shared_ptr CsgLeafNode::ToLeafNode() const { return std::make_shared(*this); } @@ -166,10 +108,14 @@ std::shared_ptr CsgLeafNode::Transform(const mat3x4 &m) const { CsgNodeType CsgLeafNode::GetNodeType() const { return CsgNodeType::Leaf; } +std::shared_ptr ImplToLeaf(Manifold::Impl &&impl) { + return std::make_shared(std::make_shared(impl)); +} + /** * Efficient union of a set of pairwise disjoint meshes. */ -Manifold::Impl CsgLeafNode::Compose( +std::shared_ptr CsgLeafNode::Compose( const std::vector> &nodes) { ZoneScoped; double epsilon = -1; @@ -187,7 +133,7 @@ Manifold::Impl CsgLeafNode::Compose( if (node->pImpl_->status_ != Manifold::Error::NoError) { Manifold::Impl impl; impl.status_ = node->pImpl_->status_; - return impl; + return ImplToLeaf(std::move(impl)); } double nodeOldScale = node->pImpl_->bBox_.Scale(); double nodeNewScale = @@ -242,18 +188,30 @@ Manifold::Impl CsgLeafNode::Compose( copy(node->pImpl_->halfedgeTangent_.begin(), node->pImpl_->halfedgeTangent_.end(), combined.halfedgeTangent_.begin() + edgeIndices[i]); - transform( - node->pImpl_->halfedge_.begin(), node->pImpl_->halfedge_.end(), - combined.halfedge_.begin() + edgeIndices[i], - UpdateHalfedge({vertIndices[i], edgeIndices[i], triIndices[i]})); + const int nextVert = vertIndices[i]; + const int nextEdge = edgeIndices[i]; + const int nextFace = triIndices[i]; + transform(node->pImpl_->halfedge_.begin(), + node->pImpl_->halfedge_.end(), + combined.halfedge_.begin() + edgeIndices[i], + [nextVert, nextEdge, nextFace](Halfedge edge) { + edge.startVert += nextVert; + edge.endVert += nextVert; + edge.pairedHalfedge += nextEdge; + return edge; + }); if (numPropOut > 0) { auto start = combined.meshRelation_.triProperties.begin() + triIndices[i]; if (node->pImpl_->NumProp() > 0) { auto &triProp = node->pImpl_->meshRelation_.triProperties; + const int nextProp = propVertIndices[i]; transform(triProp.begin(), triProp.end(), start, - UpdateTriProp({propVertIndices[i]})); + [nextProp](ivec3 tri) { + tri += nextProp; + return tri; + }); const int numProp = node->pImpl_->NumProp(); auto &oldProp = node->pImpl_->meshRelation_.properties; @@ -282,8 +240,11 @@ Manifold::Impl CsgLeafNode::Compose( } else { // no need to apply the transform to the node, just copy the vertices // and face normals and apply transform on the fly + const mat3x4 transform = node->transform_; auto vertPosBegin = TransformIterator( - node->pImpl_->vertPos_.begin(), Transform4x3({node->transform_})); + node->pImpl_->vertPos_.begin(), [&transform](vec3 position) { + return transform * vec4(position, 1.0); + }); mat3 normalTransform = la::inverse(la::transpose(mat3(node->transform_))); auto faceNormalBegin = @@ -311,7 +272,10 @@ Manifold::Impl CsgLeafNode::Compose( transform(node->pImpl_->meshRelation_.triRef.begin(), node->pImpl_->meshRelation_.triRef.end(), combined.meshRelation_.triRef.begin() + triIndices[i], - UpdateMeshIDs({offset})); + [offset](TriRef ref) { + ref.meshID += offset; + return ref; + }); }); for (size_t i = 0; i < nodes.size(); i++) { @@ -327,173 +291,52 @@ Manifold::Impl CsgLeafNode::Compose( combined.SimplifyTopology(); combined.Finish(); combined.IncrementMeshIDs(); - return combined; -} - -CsgOpNode::CsgOpNode() {} - -CsgOpNode::CsgOpNode(const std::vector> &children, - OpType op) - : impl_(Impl{}) { - auto impl = impl_.GetGuard(); - impl->children_ = children; - SetOp(op); -} - -CsgOpNode::CsgOpNode(std::vector> &&children, - OpType op) - : impl_(Impl{}) { - auto impl = impl_.GetGuard(); - impl->children_ = children; - SetOp(op); -} - -std::shared_ptr CsgOpNode::Boolean( - const std::shared_ptr &second, OpType op) { - std::vector> children; - - auto isReused = [](const auto &node) { return node->impl_.UseCount() > 1; }; - - auto copyChildren = [&](const auto &list, const mat3x4 &transform) { - for (const auto &child : list) { - children.push_back(child->Transform(transform)); - } - }; - - auto self = std::dynamic_pointer_cast(shared_from_this()); - if (IsOp(op) && !isReused(self)) { - auto impl = impl_.GetGuard(); - copyChildren(impl->children_, transform_); - } else { - children.push_back(self); - } - - auto secondOp = std::dynamic_pointer_cast(second); - auto canInlineSecondOp = [&]() { - switch (op) { - case OpType::Add: - case OpType::Intersect: - return secondOp->IsOp(op); - case OpType::Subtract: - return secondOp->IsOp(OpType::Add); - default: - return false; - } - }; - - if (secondOp && canInlineSecondOp() && !isReused(secondOp)) { - auto secondImpl = secondOp->impl_.GetGuard(); - copyChildren(secondImpl->children_, secondOp->transform_); - } else { - children.push_back(second); - } - - return std::make_shared(children, op); -} - -std::shared_ptr CsgOpNode::Transform(const mat3x4 &m) const { - auto node = std::make_shared(); - node->impl_ = impl_; - node->transform_ = m * Mat4(transform_); - node->op_ = op_; - return node; -} - -std::shared_ptr CsgOpNode::ToLeafNode() const { - if (cache_ != nullptr) return cache_; - // turn the children into leaf nodes - GetChildren(); - auto impl = impl_.GetGuard(); - auto &children_ = impl->children_; - if (children_.size() > 1) { - switch (op_) { - case CsgNodeType::Union: - BatchUnion(); - break; - case CsgNodeType::Intersection: { - std::vector> impls; - for (auto &child : children_) { - impls.push_back( - std::dynamic_pointer_cast(child)->GetImpl()); - } - children_.clear(); - children_.push_back(std::make_shared( - BatchBoolean(OpType::Intersect, impls))); - break; - }; - case CsgNodeType::Difference: { - // take the lhs out and treat the remaining nodes as the rhs, perform - // union optimization for them - auto lhs = std::dynamic_pointer_cast(children_.front()); - children_.erase(children_.begin()); - BatchUnion(); - auto rhs = std::dynamic_pointer_cast(children_.front()); - children_.clear(); - Boolean3 boolean(*lhs->GetImpl(), *rhs->GetImpl(), OpType::Subtract); - children_.push_back( - std::make_shared(std::make_shared( - boolean.Result(OpType::Subtract)))); - }; - case CsgNodeType::Leaf: - // unreachable - break; - } - } else if (children_.size() == 0) { - return nullptr; - } - // children_ must contain only one CsgLeafNode now, and its Transform will - // give CsgLeafNode as well - cache_ = std::dynamic_pointer_cast( - children_.front()->Transform(transform_)); - return cache_; + return ImplToLeaf(std::move(combined)); } /** * Efficient boolean operation on a set of nodes utilizing commutativity of the * operation. Only supports union and intersection. */ -std::shared_ptr CsgOpNode::BatchBoolean( - OpType operation, - std::vector> &results) { +std::shared_ptr BatchBoolean( + OpType operation, std::vector> &results) { ZoneScoped; - auto getImplPtr = GetImplPtr(); DEBUG_ASSERT(operation != OpType::Subtract, logicErr, "BatchBoolean doesn't support Difference."); // common cases - if (results.size() == 0) return std::make_shared(); - if (results.size() == 1) - return std::make_shared(*results.front()); + if (results.size() == 0) return std::make_shared(); + if (results.size() == 1) return results.front(); if (results.size() == 2) { - Boolean3 boolean(*results[0], *results[1], operation); - return std::make_shared(boolean.Result(operation)); + Boolean3 boolean(*results[0]->GetImpl(), *results[1]->GetImpl(), operation); + return ImplToLeaf(boolean.Result(operation)); } #if (MANIFOLD_PAR == 1) && __has_include() tbb::task_group group; - tbb::concurrent_priority_queue queue(results.size()); + tbb::concurrent_priority_queue, MeshCompare> + queue(results.size()); for (auto result : results) { queue.emplace(result); } results.clear(); std::function process = [&]() { while (queue.size() > 1) { - SharedImpl a, b; + std::shared_ptr a, b; if (!queue.try_pop(a)) continue; if (!queue.try_pop(b)) { queue.push(a); continue; } group.run([&, a, b]() { - Boolean3 boolean(*getImplPtr(a), *getImplPtr(b), operation); - queue.emplace( - std::make_shared(boolean.Result(operation))); + Boolean3 boolean(*a->GetImpl(), *b->GetImpl(), operation); + queue.emplace(ImplToLeaf(boolean.Result(operation))); return group.run(process); }); } }; group.run_and_wait(process); - SharedImpl r; + std::shared_ptr r; queue.try_pop(r); - return *std::get_if>(&r); + return r; #endif // apply boolean operations starting from smaller meshes // the assumption is that boolean operations on smaller meshes is faster, @@ -508,24 +351,23 @@ std::shared_ptr CsgOpNode::BatchBoolean( auto b = std::move(results.back()); results.pop_back(); // boolean operation - Boolean3 boolean(*a, *b, operation); - auto result = std::make_shared(boolean.Result(operation)); + Boolean3 boolean(*a->GetImpl(), *b->GetImpl(), operation); + auto result = ImplToLeaf(boolean.Result(operation)); if (results.size() == 0) { return result; } results.push_back(result); std::push_heap(results.begin(), results.end(), cmpFn); } - return std::make_shared(*results.front()); + return results.front(); } /** * Efficient union operation on a set of nodes by doing Compose as much as * possible. - * Note: Due to some unknown issues with `Compose`, we are now doing - * `BatchBoolean` instead of using `Compose` for non-intersecting manifolds. */ -void CsgOpNode::BatchUnion() const { +std::shared_ptr BatchUnion( + std::vector> &children) { ZoneScoped; // INVARIANT: children_ is a vector of leaf nodes // this kMaxUnionSize is a heuristic to avoid the pairwise disjoint check @@ -533,26 +375,25 @@ void CsgOpNode::BatchUnion() const { // If the number of children exceeded this limit, we will operate on chunks // with size kMaxUnionSize. constexpr size_t kMaxUnionSize = 1000; - auto impl = impl_.GetGuard(); - auto &children_ = impl->children_; - while (children_.size() > 1) { - const size_t start = (children_.size() > kMaxUnionSize) - ? (children_.size() - kMaxUnionSize) + DEBUG_ASSERT(!children.empty(), logicErr, + "BatchUnion should not have empty children"); + while (children.size() > 1) { + const size_t start = (children.size() > kMaxUnionSize) + ? (children.size() - kMaxUnionSize) : 0; Vec boxes; - boxes.reserve(children_.size() - start); - for (size_t i = start; i < children_.size(); i++) { - boxes.push_back(std::dynamic_pointer_cast(children_[i]) - ->GetImpl() - ->bBox_); + boxes.reserve(children.size() - start); + for (size_t i = start; i < children.size(); i++) { + boxes.push_back(children[i]->GetImpl()->bBox_); } // partition the children into a set of disjoint sets // each set contains a set of children that are pairwise disjoint std::vector> disjointSets; for (size_t i = 0; i < boxes.size(); i++) { auto lambda = [&boxes, i](const Vec &set) { - return std::find_if(set.begin(), set.end(), CheckOverlap({boxes, i})) == - set.end(); + return std::find_if(set.begin(), set.end(), [&boxes, i](size_t j) { + return boxes[i].DoesOverlap(boxes[j]); + }) == set.end(); }; auto it = std::find_if(disjointSets.begin(), disjointSets.end(), lambda); if (it == disjointSets.end()) { @@ -562,82 +403,260 @@ void CsgOpNode::BatchUnion() const { } } // compose each set of disjoint children - std::vector> impls; + std::vector> impls; for (auto &set : disjointSets) { if (set.size() == 1) { - impls.push_back( - std::dynamic_pointer_cast(children_[start + set[0]]) - ->GetImpl()); + impls.push_back(children[start + set[0]]); } else { std::vector> tmp; for (size_t j : set) { - tmp.push_back( - std::dynamic_pointer_cast(children_[start + j])); + tmp.push_back(children[start + j]); } - impls.push_back( - std::make_shared(CsgLeafNode::Compose(tmp))); + impls.push_back(CsgLeafNode::Compose(tmp)); } } - children_.erase(children_.begin() + start, children_.end()); - children_.push_back( - std::make_shared(BatchBoolean(OpType::Add, impls))); + children.erase(children.begin() + start, children.end()); + children.push_back(BatchBoolean(OpType::Add, impls)); // move it to the front as we process from the back, and the newly added // child should be quite complicated - std::swap(children_.front(), children_.back()); + std::swap(children.front(), children.back()); } + return children.front(); } -/** - * Flatten the children to a list of leaf nodes and return them. - * If forceToLeafNodes is true, the list will be guaranteed to be a list of leaf - * nodes (i.e. no ops). Otherwise, the list may contain ops. Note that this - * function will not apply the transform to children, as they may be shared with - * other nodes. - */ -std::vector> &CsgOpNode::GetChildren( - bool forceToLeafNodes) const { +CsgOpNode::CsgOpNode() {} + +CsgOpNode::CsgOpNode(const std::vector> &children, + OpType op) + : impl_(Impl{}), op_(op) { auto impl = impl_.GetGuard(); + impl->children_ = children; +} - if (forceToLeafNodes && !impl->forcedToLeafNodes_) { - impl->forcedToLeafNodes_ = true; - for_each(ExecutionPolicy::Par, impl->children_.begin(), - impl->children_.end(), [](auto &child) { - if (child->GetNodeType() != CsgNodeType::Leaf) { - child = child->ToLeafNode(); - } - }); - } - return impl->children_; +std::shared_ptr CsgOpNode::Boolean( + const std::shared_ptr &second, OpType op) { + std::vector> children; + children.push_back(shared_from_this()); + children.push_back(second); + + return std::make_shared(children, op); } -void CsgOpNode::SetOp(OpType op) { - switch (op) { - case OpType::Add: - op_ = CsgNodeType::Union; - break; - case OpType::Subtract: - op_ = CsgNodeType::Difference; - break; - case OpType::Intersect: - op_ = CsgNodeType::Intersection; - break; +std::shared_ptr CsgOpNode::Transform(const mat3x4 &m) const { + auto node = std::make_shared(); + node->impl_ = impl_; + node->transform_ = m * Mat4(transform_); + node->op_ = op_; + return node; +} + +struct CsgStackFrame { + bool finalize; + OpType parent_op; + mat3x4 transform; + std::vector> *destination; + std::shared_ptr op_node; + std::vector> positive_children; + std::vector> negative_children; + + CsgStackFrame(bool finalize, OpType parent_op, mat3x4 transform, + std::vector> *parent, + std::shared_ptr op_node) + : finalize(finalize), + parent_op(parent_op), + transform(transform), + destination(parent), + op_node(op_node) {} +}; + +std::shared_ptr CsgOpNode::ToLeafNode() const { + if (cache_ != nullptr) return cache_; + + // Note: We do need a pointer here to avoid vector pointers from being + // invalidated after pushing elements into the explicit stack. + // It is a `shared_ptr` because we may want to drop the stack frame while + // still referring to some of the elements inside the old frame. + // It is possible to use `unique_ptr`, extending the lifetime of the frame + // when we remove it from the stack, but it is a bit more complicated and + // there is no measurable overhead from using `shared_ptr` here... + std::vector> stack; + // initial node, destination is a nullptr because we don't need to put the + // result anywhere else (except in the cache_). + stack.push_back(std::make_shared( + false, op_, la::identity, nullptr, + std::static_pointer_cast(shared_from_this()))); + + // Instead of actually using recursion in the algorithm, we use an explicit + // stack, do DFS and store the intermediate states into `CsgStackFrame` to + // avoid stack overflow. + // + // Before performing boolean operations, we should make sure that all children + // are `CsgLeafNodes`, i.e. are actual meshes that can be operated on. Hence, + // we do it in two steps: + // 1. Populate `children` (`left_children` and `right_children`, see below) + // If the child is a `CsgOpNode`, we either collapse it or compute its + // boolean operation result. + // 2. Performs boolean after populating the `children` set. + // After a boolean operation is completed, we put the result back to its + // parent's `children` set. + // + // When we populate `children`, we perform collapsing on-the-fly. + // For example, we want to turn `(Union a (Union b c))` into `(Union a b c)`. + // This allows more efficient `BatchBoolean`/`BatchUnion` calls. + // We can do this when the child operation is the same as the parent + // operation, except when the operation is `Subtract` (see below). + // Note that to avoid repeating work, we will not collapse nodes that are + // reused. And in the special case where the children set only contains one + // element, we don't need any operation, so we can collapse that as well. + // Instead of moving `b` and `c` into the parent, and running this collapsing + // check until a fixed point, we remember the `destination` where we should + // put the `CsgLeafNode` into. Normally, the `destination` pointer point to + // the parent `children` set. However, when a child is being collapsed, we + // keep using the old `destination` pointer for the grandchildren. Hence, + // removing a node by collapsing takes O(1) time. We also need to store the + // parent operation type for checking if the node is eligible for collapsing, + // and transform matrix because we need to re-apply the transformation to the + // children. + // + // `Subtract` is handled differently from `Add` and `Intersect`. It is treated + // as two `Add` nodes, `positive_children` and `negative_children`, that + // should be subtracted later. This allows collapsing children `Add` nodes. + // For normal `Add` and `Intersect`, we only use `positive_children`. + // + // `impl->children_` should always contain either the raw set of children or + // the NOT transformed result, while `cache_` should contain the transformed + // result. This is because `impl` can be shared between `CsgOpNode` that + // differ in `transform_`, so we want it to be able to share the result. + // =========================================================================== + // Recursive version (pseudocode only): + // + // void f(CsgOpNode node, OpType parent_op, mat3x4 transform, + // std::vector *destination) { + // auto impl = node->impl_.GetGuard(); + // // can collapse when we have the same operation as the parent and is + // // unique, or when we have only one children. + // const bool canCollapse = (node->op_ == parent_op && IsUnique(node)) || + // impl->children_.size() == 1; + // const mat3x4 transform2 = canCollapse ? transform * node->transform_ + // : la::identity; + // std::vector positive_children, negative_children; + // // for subtract, we pretend the operation is Add for our children. + // auto op = node->op_ == OpType::Subtract ? OpType::Add : node->op_; + // for (size_t i = 0; i < impl->children_.size(); i++) { + // auto child = impl->children_[i]; + // // negative when it is the remaining operands for Subtract + // auto dest = node->op_ == OpType::Subtract && i != 0 ? + // negative_children : positive_children; + // if (canCollapse) dest = destination; + // if (child->GetNodeType() == CsgNodeType::Leaf) + // dest.push_back(child); + // else + // f(child, op, transform2, dest); + // } + // if (canCollapse) return; + // if (node->op_ == OpType::Add) + // impl->children_ = {BatchUnion(positive_children)}; + // else if (node->op_ == OpType::Intersect) + // impl->children_ = {BatchBoolean(Intersect, positive_children)}; + // else // subtract + // impl->children_ = { BatchUnion(positive_children) - + // BatchUnion(negative_children)}; + // // node local transform + // node->cache_ = impl->children_[0].Transform(node.transform); + // // collapsed node transforms + // if (destination) + // destination->push_back(node->cache_->Transform(transform)); + // } + while (!stack.empty()) { + std::shared_ptr frame = stack.back(); + auto impl = frame->op_node->impl_.GetGuard(); + if (frame->finalize) { + switch (frame->op_node->op_) { + case OpType::Add: + impl->children_ = {BatchUnion(frame->positive_children)}; + break; + case OpType::Intersect: { + impl->children_ = { + BatchBoolean(OpType::Intersect, frame->positive_children)}; + break; + }; + case OpType::Subtract: + if (frame->positive_children.empty()) { + // nothing to subtract from, so the result is empty. + impl->children_ = {std::make_shared()}; + } else { + auto positive = BatchUnion(frame->positive_children); + if (frame->negative_children.empty()) { + // nothing to subtract, result equal to the LHS. + impl->children_ = {frame->positive_children[0]}; + } else { + Boolean3 boolean(*positive->GetImpl(), + *BatchUnion(frame->negative_children)->GetImpl(), + OpType::Subtract); + impl->children_ = {ImplToLeaf(boolean.Result(OpType::Subtract))}; + } + } + break; + } + frame->op_node->cache_ = std::static_pointer_cast( + impl->children_[0]->Transform(frame->op_node->transform_)); + if (frame->destination != nullptr) + frame->destination->push_back(std::static_pointer_cast( + frame->op_node->cache_->Transform(frame->transform))); + stack.pop_back(); + } else { + auto add_children = [&stack](std::shared_ptr &node, OpType op, + mat3x4 transform, auto *destination) { + if (node->GetNodeType() == CsgNodeType::Leaf) + destination->push_back(std::static_pointer_cast( + node->Transform(transform))); + else + stack.push_back(std::make_shared( + false, op, transform, destination, + std::static_pointer_cast(node))); + }; + // op_node use_count == 2 because it is both inside one CsgOpNode + // and in our stack. + // if there is only one child, we can also collapse. + const bool canCollapse = frame->destination != nullptr && + ((frame->op_node->op_ == frame->parent_op && + frame->op_node.use_count() <= 2 && + frame->op_node->impl_.UseCount() == 1) || + impl->children_.size() == 1); + if (canCollapse) + stack.pop_back(); + else + frame->finalize = true; + + const mat3x4 transform = + canCollapse ? (frame->transform * Mat4(frame->op_node->transform_)) + : la::identity; + OpType op = frame->op_node->op_ == OpType::Subtract ? OpType::Add + : frame->op_node->op_; + for (size_t i = 0; i < impl->children_.size(); i++) { + auto dest = canCollapse ? frame->destination + : (frame->op_node->op_ == OpType::Subtract && i != 0) + ? &frame->negative_children + : &frame->positive_children; + add_children(impl->children_[i], op, transform, dest); + } + } } + return cache_; } -bool CsgOpNode::IsOp(OpType op) { - switch (op) { +CsgNodeType CsgOpNode::GetNodeType() const { + switch (op_) { case OpType::Add: - return op_ == CsgNodeType::Union; + return CsgNodeType::Union; case OpType::Subtract: - return op_ == CsgNodeType::Difference; + return CsgNodeType::Difference; case OpType::Intersect: - return op_ == CsgNodeType::Intersection; - default: - return false; + return CsgNodeType::Intersection; } + // unreachable... + return CsgNodeType::Leaf; } -mat3x4 CsgOpNode::GetTransform() const { return transform_; } - } // namespace manifold diff --git a/thirdparty/manifold/src/csg_tree.h b/thirdparty/manifold/src/csg_tree.h index f2a2f692afed..5c24dc5a2050 100644 --- a/thirdparty/manifold/src/csg_tree.h +++ b/thirdparty/manifold/src/csg_tree.h @@ -27,7 +27,6 @@ class CsgNode : public std::enable_shared_from_this { virtual std::shared_ptr ToLeafNode() const = 0; virtual std::shared_ptr Transform(const mat3x4 &m) const = 0; virtual CsgNodeType GetNodeType() const = 0; - virtual mat3x4 GetTransform() const = 0; virtual std::shared_ptr Boolean( const std::shared_ptr &second, OpType op); @@ -52,9 +51,7 @@ class CsgLeafNode final : public CsgNode { CsgNodeType GetNodeType() const override; - mat3x4 GetTransform() const override; - - static Manifold::Impl Compose( + static std::shared_ptr Compose( const std::vector> &nodes); private: @@ -68,8 +65,6 @@ class CsgOpNode final : public CsgNode { CsgOpNode(const std::vector> &children, OpType op); - CsgOpNode(std::vector> &&children, OpType op); - std::shared_ptr Boolean(const std::shared_ptr &second, OpType op) override; @@ -77,9 +72,7 @@ class CsgOpNode final : public CsgNode { std::shared_ptr ToLeafNode() const override; - CsgNodeType GetNodeType() const override { return op_; } - - mat3x4 GetTransform() const override; + CsgNodeType GetNodeType() const override; private: struct Impl { @@ -87,22 +80,10 @@ class CsgOpNode final : public CsgNode { bool forcedToLeafNodes_ = false; }; mutable ConcurrentSharedPtr impl_ = ConcurrentSharedPtr(Impl{}); - CsgNodeType op_; + OpType op_; mat3x4 transform_ = la::identity; // the following fields are for lazy evaluation, so they are mutable mutable std::shared_ptr cache_ = nullptr; - - void SetOp(OpType); - bool IsOp(OpType op); - - static std::shared_ptr BatchBoolean( - OpType operation, - std::vector> &results); - - void BatchUnion() const; - - std::vector> &GetChildren( - bool forceToLeafNodes = true) const; }; } // namespace manifold diff --git a/thirdparty/manifold/src/impl.cpp b/thirdparty/manifold/src/impl.cpp index 9582b2c12638..59cc0293d6da 100644 --- a/thirdparty/manifold/src/impl.cpp +++ b/thirdparty/manifold/src/impl.cpp @@ -194,13 +194,14 @@ int GetLabels(std::vector& components, } void DedupePropVerts(manifold::Vec& triProp, - const Vec>& vert2vert) { + const Vec>& vert2vert, + size_t numPropVert) { ZoneScoped; std::vector vertLabels; - const int numLabels = GetLabels(vertLabels, vert2vert, vert2vert.size()); + const int numLabels = GetLabels(vertLabels, vert2vert, numPropVert); std::vector label2vert(numLabels); - for (size_t v = 0; v < vert2vert.size(); ++v) label2vert[vertLabels[v]] = v; + for (size_t v = 0; v < numPropVert; ++v) label2vert[vertLabels[v]] = v; for (auto& prop : triProp) for (int i : {0, 1, 2}) prop[i] = label2vert[vertLabels[prop[i]]]; } @@ -343,6 +344,8 @@ void Manifold::Impl::CreateFaces() { const int prop1 = meshRelation_ .triProperties[pairFace][jointNum == 2 ? 0 : jointNum + 1]; + if (prop0 == prop1) return; + bool propEqual = true; for (size_t p = 0; p < numProp; ++p) { if (meshRelation_.properties[numProp * prop0 + p] != @@ -355,7 +358,7 @@ void Manifold::Impl::CreateFaces() { vert2vert[edgeIdx] = std::make_pair(prop0, prop1); } }); - DedupePropVerts(meshRelation_.triProperties, vert2vert); + DedupePropVerts(meshRelation_.triProperties, vert2vert, NumPropVert()); } for_each_n(autoPolicy(halfedge_.size(), 1e4), countAt(0), halfedge_.size(), diff --git a/thirdparty/manifold/include/manifold/iters.h b/thirdparty/manifold/src/iters.h similarity index 100% rename from thirdparty/manifold/include/manifold/iters.h rename to thirdparty/manifold/src/iters.h diff --git a/thirdparty/manifold/src/parallel.h b/thirdparty/manifold/src/parallel.h index 221cefac1bb7..a434939494c6 100644 --- a/thirdparty/manifold/src/parallel.h +++ b/thirdparty/manifold/src/parallel.h @@ -17,6 +17,7 @@ #pragma once +#include "./iters.h" #if (MANIFOLD_PAR == 1) #include #include @@ -27,7 +28,6 @@ #include #include -#include "manifold/iters.h" namespace manifold { enum class ExecutionPolicy { diff --git a/thirdparty/manifold/src/properties.cpp b/thirdparty/manifold/src/properties.cpp index 32f79e5b601f..911133776ab3 100644 --- a/thirdparty/manifold/src/properties.cpp +++ b/thirdparty/manifold/src/properties.cpp @@ -157,10 +157,11 @@ struct CheckCCW { "tol = %g, area2 = %g, base2*tol2 = %g\n" "normal = %g, %g, %g\n" "norm = %g, %g, %g\nverts: %d, %d, %d\n", - face, area / base, base, tol, area * area, base2 * tol * tol, - triNormal[face].x, triNormal[face].y, triNormal[face].z, norm.x, - norm.y, norm.z, halfedges[3 * face].startVert, - halfedges[3 * face + 1].startVert, halfedges[3 * face + 2].startVert); + static_cast(face), area / base, base, tol, area * area, + base2 * tol * tol, triNormal[face].x, triNormal[face].y, + triNormal[face].z, norm.x, norm.y, norm.z, + halfedges[3 * face].startVert, halfedges[3 * face + 1].startVert, + halfedges[3 * face + 2].startVert); } #endif return check; From 113621dd8998cc3f33281c8c420bcde678cfdb99 Mon Sep 17 00:00:00 2001 From: kobewi Date: Wed, 4 Dec 2024 15:10:05 +0100 Subject: [PATCH 15/33] Don't rebuild tree when navigating to path --- editor/filesystem_dock.cpp | 52 +++++++++++++++++++++++++++++++++++--- editor/filesystem_dock.h | 1 + 2 files changed, 49 insertions(+), 4 deletions(-) diff --git a/editor/filesystem_dock.cpp b/editor/filesystem_dock.cpp index 93bfe9511090..873ad2a32599 100644 --- a/editor/filesystem_dock.cpp +++ b/editor/filesystem_dock.cpp @@ -245,6 +245,8 @@ void FileSystemDock::_create_tree(TreeItem *p_parent, EditorFileSystemDirectory } subdirectory_item->set_selectable(0, true); subdirectory_item->set_metadata(0, lpath); + folder_map[lpath] = subdirectory_item; + if (!p_select_in_favorites && (current_path == lpath || ((display_mode != DISPLAY_MODE_TREE_ONLY) && current_path.get_base_dir() == lpath))) { subdirectory_item->select(0); // Keep select an item when re-created a tree @@ -371,6 +373,7 @@ void FileSystemDock::_update_tree(const Vector &p_uncollapsed_paths, boo tree_update_id++; updating_tree = true; TreeItem *root = tree->create_item(); + folder_map.clear(); // Handles the favorites. favorites_item = tree->create_item(root); @@ -698,19 +701,21 @@ void FileSystemDock::_set_current_path_line_edit_text(const String &p_path) { } void FileSystemDock::_navigate_to_path(const String &p_path, bool p_select_in_favorites) { + bool is_directory = false; if (p_path == "Favorites") { current_path = p_path; } else { String target_path = p_path; // If the path is a file, do not only go to the directory in the tree, also select the file in the file list. if (target_path.ends_with("/")) { - target_path = target_path.substr(0, target_path.length() - 1); + target_path = target_path.trim_suffix("/"); } Ref da = DirAccess::create(DirAccess::ACCESS_RESOURCES); if (da->file_exists(p_path)) { current_path = target_path; } else if (da->dir_exists(p_path)) { current_path = target_path + "/"; + is_directory = true; } else { ERR_FAIL_MSG(vformat("Cannot navigate to '%s' as it has not been found in the file system!", p_path)); } @@ -719,17 +724,56 @@ void FileSystemDock::_navigate_to_path(const String &p_path, bool p_select_in_fa _set_current_path_line_edit_text(current_path); _push_to_history(); - _update_tree(get_uncollapsed_paths(), false, p_select_in_favorites, true); + const String file_name = is_directory ? p_path.trim_suffix("/").get_file() + "/" : p_path.get_file(); + bool found = false; + + TreeItem **base_dir_ptr; + { + const String base_dir = current_path.get_base_dir(); + if (base_dir == "res://") { + base_dir_ptr = folder_map.getptr(base_dir); + } else if (is_directory) { + base_dir_ptr = folder_map.getptr(base_dir.get_base_dir() + "/"); + } else { + base_dir_ptr = folder_map.getptr(base_dir + "/"); + } + } + + if (base_dir_ptr) { + TreeItem *directory = *base_dir_ptr; + { + TreeItem *entry = directory->get_first_child(); + while (entry) { + if (entry->get_metadata(0).operator String().ends_with(file_name)) { + tree->deselect_all(); + entry->select(0); + found = true; + break; + } + entry = entry->get_next(); + } + } + + while (directory) { + directory->set_collapsed(false); + directory = directory->get_parent(); + } + } + + if (!found) { + return; + } + + tree->ensure_cursor_is_visible(); if (display_mode != DISPLAY_MODE_TREE_ONLY) { _update_file_list(false); // Reset the scroll for a directory. - if (p_path.ends_with("/")) { + if (is_directory) { files->get_v_scroll_bar()->set_value(0); } } - String file_name = p_path.get_file(); if (!file_name.is_empty()) { for (int i = 0; i < files->get_item_count(); i++) { if (files->get_item_text(i) == file_name) { diff --git a/editor/filesystem_dock.h b/editor/filesystem_dock.h index d2e403a8aff3..7dd98fefab5b 100644 --- a/editor/filesystem_dock.h +++ b/editor/filesystem_dock.h @@ -133,6 +133,7 @@ class FileSystemDock : public VBoxContainer { CONVERT_BASE_ID = 1000, }; + HashMap folder_map; HashMap folder_colors; Dictionary assigned_folder_colors; From 5318008ce6a52fa3af7f466189a263d4905c76fa Mon Sep 17 00:00:00 2001 From: Jakub Marcowski Date: Wed, 4 Dec 2024 13:47:20 +0100 Subject: [PATCH 16/33] thorvg: Update to 0.15.5 --- modules/svg/SCsub | 2 +- thirdparty/README.md | 2 +- thirdparty/thorvg/AUTHORS | 6 +- thirdparty/thorvg/inc/config.h | 2 +- thirdparty/thorvg/inc/thorvg.h | 292 +++++++---- .../thorvg/src/common/tvgCompressor.cpp | 2 +- thirdparty/thorvg/src/common/tvgLines.cpp | 245 ---------- thirdparty/thorvg/src/common/tvgLines.h | 61 --- thirdparty/thorvg/src/common/tvgMath.cpp | 255 +++++++++- thirdparty/thorvg/src/common/tvgMath.h | 119 +++-- thirdparty/thorvg/src/common/tvgStr.cpp | 10 - thirdparty/thorvg/src/common/tvgStr.h | 1 - .../src/loaders/external_png/tvgPngLoader.cpp | 4 +- .../loaders/external_webp/tvgWebpLoader.cpp | 2 +- .../src/loaders/external_webp/tvgWebpLoader.h | 2 +- .../thorvg/src/loaders/jpg/tvgJpgLoader.cpp | 2 +- .../thorvg/src/loaders/jpg/tvgJpgLoader.h | 2 +- thirdparty/thorvg/src/loaders/jpg/tvgJpgd.cpp | 17 +- .../thorvg/src/loaders/svg/tvgSvgLoader.cpp | 49 +- .../src/loaders/svg/tvgSvgLoaderCommon.h | 1 + .../thorvg/src/loaders/svg/tvgSvgPath.cpp | 12 +- .../src/loaders/svg/tvgSvgSceneBuilder.cpp | 36 +- .../thorvg/src/loaders/svg/tvgXmlParser.cpp | 7 +- .../src/renderer/sw_engine/tvgSwCommon.h | 49 +- .../src/renderer/sw_engine/tvgSwFill.cpp | 40 +- .../src/renderer/sw_engine/tvgSwImage.cpp | 4 +- .../src/renderer/sw_engine/tvgSwMath.cpp | 39 +- .../renderer/sw_engine/tvgSwPostEffect.cpp | 410 ++++++++++++++++ .../src/renderer/sw_engine/tvgSwRaster.cpp | 456 +++++------------- .../src/renderer/sw_engine/tvgSwRasterAvx.h | 4 +- .../src/renderer/sw_engine/tvgSwRasterC.h | 38 +- .../src/renderer/sw_engine/tvgSwRasterNeon.h | 2 +- .../renderer/sw_engine/tvgSwRasterTexmap.h | 21 +- .../src/renderer/sw_engine/tvgSwRenderer.cpp | 203 +++++--- .../src/renderer/sw_engine/tvgSwRenderer.h | 14 +- .../src/renderer/sw_engine/tvgSwRle.cpp | 257 +++++----- .../src/renderer/sw_engine/tvgSwShape.cpp | 30 +- .../src/renderer/sw_engine/tvgSwStroke.cpp | 40 +- .../thorvg/src/renderer/tvgBinaryDesc.h | 2 +- thirdparty/thorvg/src/renderer/tvgCanvas.h | 16 +- thirdparty/thorvg/src/renderer/tvgCommon.h | 9 - thirdparty/thorvg/src/renderer/tvgFill.cpp | 25 +- thirdparty/thorvg/src/renderer/tvgFill.h | 1 - .../thorvg/src/renderer/tvgGlCanvas.cpp | 4 +- .../thorvg/src/renderer/tvgLoadModule.h | 4 +- thirdparty/thorvg/src/renderer/tvgLoader.cpp | 8 +- thirdparty/thorvg/src/renderer/tvgPaint.cpp | 135 +++--- thirdparty/thorvg/src/renderer/tvgPaint.h | 23 +- thirdparty/thorvg/src/renderer/tvgPicture.cpp | 16 +- thirdparty/thorvg/src/renderer/tvgPicture.h | 2 +- thirdparty/thorvg/src/renderer/tvgRender.h | 98 +++- thirdparty/thorvg/src/renderer/tvgSaver.cpp | 7 +- thirdparty/thorvg/src/renderer/tvgScene.cpp | 66 ++- thirdparty/thorvg/src/renderer/tvgScene.h | 60 ++- thirdparty/thorvg/src/renderer/tvgShape.cpp | 21 +- thirdparty/thorvg/src/renderer/tvgShape.h | 10 +- .../thorvg/src/renderer/tvgSwCanvas.cpp | 4 +- thirdparty/thorvg/src/renderer/tvgText.cpp | 5 +- thirdparty/thorvg/src/renderer/tvgText.h | 4 +- .../thorvg/src/renderer/tvgWgCanvas.cpp | 15 +- thirdparty/thorvg/update-thorvg.sh | 2 +- 61 files changed, 1936 insertions(+), 1339 deletions(-) delete mode 100644 thirdparty/thorvg/src/common/tvgLines.cpp delete mode 100644 thirdparty/thorvg/src/common/tvgLines.h create mode 100644 thirdparty/thorvg/src/renderer/sw_engine/tvgSwPostEffect.cpp diff --git a/modules/svg/SCsub b/modules/svg/SCsub index af8f6c14f4ed..83321cf4493b 100644 --- a/modules/svg/SCsub +++ b/modules/svg/SCsub @@ -14,7 +14,6 @@ thirdparty_dir = "#thirdparty/thorvg/" thirdparty_sources = [ # common "src/common/tvgCompressor.cpp", - "src/common/tvgLines.cpp", "src/common/tvgMath.cpp", "src/common/tvgStr.cpp", # SVG parser @@ -52,6 +51,7 @@ thirdparty_sources = [ "src/renderer/sw_engine/tvgSwImage.cpp", "src/renderer/sw_engine/tvgSwMath.cpp", "src/renderer/sw_engine/tvgSwMemPool.cpp", + "src/renderer/sw_engine/tvgSwPostEffect.cpp", "src/renderer/sw_engine/tvgSwRaster.cpp", "src/renderer/sw_engine/tvgSwRenderer.cpp", "src/renderer/sw_engine/tvgSwRle.cpp", diff --git a/thirdparty/README.md b/thirdparty/README.md index 1e11c6c9725b..117845dd0adb 100644 --- a/thirdparty/README.md +++ b/thirdparty/README.md @@ -916,7 +916,7 @@ instead of `miniz.h` as an external dependency. ## thorvg - Upstream: https://github.com/thorvg/thorvg -- Version: 0.14.10 (366dcd72850c360b49e841e568fc5a154d7cce9e, 2024) +- Version: 0.15.5 (89ab573acb253567975b2494069c7ee9abc9267c, 2024) - License: MIT Files extracted from upstream source: diff --git a/thirdparty/thorvg/AUTHORS b/thirdparty/thorvg/AUTHORS index e00e91a6967b..a15f3262a82b 100644 --- a/thirdparty/thorvg/AUTHORS +++ b/thirdparty/thorvg/AUTHORS @@ -28,6 +28,10 @@ Nattu Adnan Gabor Kiss-Vamosi Lorcán Mc Donagh Lucas Niu -Francisco Ramírez +Francisco Ramírez Abdelrahman Ashraf Neo Xu +Thaddeus Crews +Josh Soref +Elliott Sales de Andrade +Łukasz Pomietło diff --git a/thirdparty/thorvg/inc/config.h b/thirdparty/thorvg/inc/config.h index fc2faca29f95..6df6f52d0476 100644 --- a/thirdparty/thorvg/inc/config.h +++ b/thirdparty/thorvg/inc/config.h @@ -15,5 +15,5 @@ // For internal debugging: //#define THORVG_LOG_ENABLED -#define THORVG_VERSION_STRING "0.14.10" +#define THORVG_VERSION_STRING "0.15.5" #endif diff --git a/thirdparty/thorvg/inc/thorvg.h b/thirdparty/thorvg/inc/thorvg.h index 4303092a5ea7..1ee898ca6ff2 100644 --- a/thirdparty/thorvg/inc/thorvg.h +++ b/thirdparty/thorvg/inc/thorvg.h @@ -157,7 +157,7 @@ enum class FillRule enum class CompositeMethod { None = 0, ///< No composition is applied. - ClipPath, ///< The intersection of the source and the target is determined and only the resulting pixels from the source are rendered. Note that ClipPath only supports the Shape type. + ClipPath, ///< The intersection of the source and the target is determined and only the resulting pixels from the source are rendered. Note that ClipPath only supports the Shape type. @deprecated Use Paint::clip() instead. AlphaMask, ///< Alpha Masking using the compositing target's pixels as an alpha value. InvAlphaMask, ///< Alpha Masking using the complement to the compositing target's pixels as an alpha value. LumaMask, ///< Alpha Masking using the grayscale (0.2125R + 0.7154G + 0.0721*B) of the compositing target's pixels. @since 0.9 @@ -178,24 +178,46 @@ enum class CompositeMethod * * @see Paint::blend() * - * @note Experimental API + * @since 0.15 */ enum class BlendMethod : uint8_t { Normal = 0, ///< Perform the alpha blending(default). S if (Sa == 255), otherwise (Sa * S) + (255 - Sa) * D - Add, ///< Simply adds pixel values of one layer with the other. (S + D) - Screen, ///< The values of the pixels in the two layers are inverted, multiplied, and then inverted again. (S + D) - (S * D) Multiply, ///< Takes the RGB channel values from 0 to 255 of each pixel in the top layer and multiples them with the values for the corresponding pixel from the bottom layer. (S * D) + Screen, ///< The values of the pixels in the two layers are inverted, multiplied, and then inverted again. (S + D) - (S * D) Overlay, ///< Combines Multiply and Screen blend modes. (2 * S * D) if (2 * D < Da), otherwise (Sa * Da) - 2 * (Da - S) * (Sa - D) - Difference, ///< Subtracts the bottom layer from the top layer or the other way around, to always get a non-negative value. (S - D) if (S > D), otherwise (D - S) - Exclusion, ///< The result is twice the product of the top and bottom layers, subtracted from their sum. s + d - (2 * s * d) - SrcOver, ///< Replace the bottom layer with the top layer. Darken, ///< Creates a pixel that retains the smallest components of the top and bottom layer pixels. min(S, D) Lighten, ///< Only has the opposite action of Darken Only. max(S, D) ColorDodge, ///< Divides the bottom layer by the inverted top layer. D / (255 - S) ColorBurn, ///< Divides the inverted bottom layer by the top layer, and then inverts the result. 255 - (255 - D) / S HardLight, ///< The same as Overlay but with the color roles reversed. (2 * S * D) if (S < Sa), otherwise (Sa * Da) - 2 * (Da - S) * (Sa - D) - SoftLight ///< The same as Overlay but with applying pure black or white does not result in pure black or white. (1 - 2 * S) * (D ^ 2) + (2 * S * D) + SoftLight, ///< The same as Overlay but with applying pure black or white does not result in pure black or white. (1 - 2 * S) * (D ^ 2) + (2 * S * D) + Difference, ///< Subtracts the bottom layer from the top layer or the other way around, to always get a non-negative value. (S - D) if (S > D), otherwise (D - S) + Exclusion, ///< The result is twice the product of the top and bottom layers, subtracted from their sum. s + d - (2 * s * d) + Hue, ///< Reserved. Not supported. + Saturation, ///< Reserved. Not supported. + Color, ///< Reserved. Not supported. + Luminosity, ///< Reserved. Not supported. + Add, ///< Simply adds pixel values of one layer with the other. (S + D) + HardMix ///< Reserved. Not supported. +}; + + +/** + * @brief Enumeration that defines methods used for Scene Effects. + * + * This enum provides options to apply various post-processing effects to a scene. + * Scene effects are typically applied to modify the final appearance of a rendered scene, such as blurring. + * + * @see Scene::push(SceneEffect effect, ...) + * + * @note Experimental API + */ +enum class SceneEffect : uint8_t +{ + ClearAll = 0, ///< Reset all previously applied scene effects, restoring the scene to its original state. + GaussianBlur, ///< Apply a blur effect with a Gaussian filter. Param(3) = {sigma(float)[> 0], direction(int)[both: 0 / horizontal: 1 / vertical: 2], border(int)[duplicate: 0 / wrap: 1], quality(int)[0 - 100]} + DropShadow ///< Apply a drop shadow effect with a Gaussian Blur filter. Param(8) = {color_R(int)[0 - 255], color_G(int)[0 - 255], color_B(int)[0 - 255], opacity(int)[0 - 255], angle(float)[0 - 360], distance(float), blur_sigma(float)[> 0], quality(int)[0 - 100]} }; @@ -206,7 +228,29 @@ enum class CanvasEngine { Sw = (1 << 1), ///< CPU rasterizer. Gl = (1 << 2), ///< OpenGL rasterizer. - Wg = (1 << 3), ///< WebGPU rasterizer. (Experimental API) + Wg = (1 << 3), ///< WebGPU rasterizer. @since 0.15 +}; + + +/** + * @brief Enumeration specifying the ThorVG class type value. + * + * ThorVG's drawing objects can return class type values, allowing you to identify the specific class of each object. + * + * @see Paint::type() + * @see Fill::type() + * + * @note Experimental API + */ +enum class Type : uint8_t +{ + Undefined = 0, ///< Unkown class + Shape, ///< Shape class + Scene, ///< Scene class + Picture, ///< Picture class + Text, ///< Text class + LinearGradient = 10, ///< LinearGradient class + RadialGradient ///< RadialGradient class }; @@ -274,7 +318,7 @@ class TVG_API Paint /** * @brief Sets the values by which the object is moved in a two-dimensional space. * - * The origin of the coordinate system is in the upper left corner of the canvas. + * The origin of the coordinate system is in the upper-left corner of the canvas. * The horizontal and vertical axes point to the right and down, respectively. * * @param[in] x The value of the horizontal shift. @@ -312,7 +356,6 @@ class TVG_API Paint * @param[in] o The opacity value in the range [0 ~ 255], where 0 is completely transparent and 255 is opaque. * * @note Setting the opacity with this API may require multiple render pass for composition. It is recommended to avoid changing the opacity if possible. - * @note ClipPath won't use the opacity value. (see: enum class CompositeMethod::ClipPath) */ Result opacity(uint8_t o) noexcept; @@ -324,6 +367,20 @@ class TVG_API Paint */ Result composite(std::unique_ptr target, CompositeMethod method) noexcept; + /** + * @brief Clip the drawing region of the paint object. + * + * This function restricts the drawing area of the paint object to the specified shape's paths. + * + * @param[in] clipper The shape object as the clipper. + * + * @retval Result::NonSupport If the @p clipper type is not Shape. + * + * @note @p clipper only supports the Shape type. + * @note Experimental API + */ + Result clip(std::unique_ptr clipper) noexcept; + /** * @brief Sets the blending method for the paint object. * @@ -386,22 +443,15 @@ class TVG_API Paint CompositeMethod composite(const Paint** target) const noexcept; /** - * @brief Retrieves the current blending method applied to the paint object. + * @brief Returns the ID value of this class. * - * @return The currently set blending method. + * This method can be used to check the current concrete instance type. * - * @note Experimental API - */ - BlendMethod blend() const noexcept; - - /** - * @brief Return the unique id value of the paint instance. - * - * This method can be called for checking the current concrete instance type. + * @return The class type ID of the Paint instance. * - * @return The type id of the Paint instance. + * @since Experimental API */ - uint32_t identifier() const noexcept; + virtual Type type() const noexcept = 0; /** * @brief Unique ID of this instance. @@ -412,6 +462,11 @@ class TVG_API Paint */ uint32_t id = 0; + /** + * @see Paint::type() + */ + TVG_DEPRECATED uint32_t identifier() const noexcept; + _TVG_DECLARE_PRIVATE(Paint); }; @@ -503,13 +558,20 @@ class TVG_API Fill Fill* duplicate() const noexcept; /** - * @brief Return the unique id value of the Fill instance. + * @brief Returns the ID value of this class. + * + * This method can be used to check the current concrete instance type. * - * This method can be called for checking the current concrete instance type. + * @return The class type ID of the Fill instance. * - * @return The type id of the Fill instance. + * @since Experimental API + */ + virtual Type type() const noexcept = 0; + + /** + * @see Fill::type() */ - uint32_t identifier() const noexcept; + TVG_DEPRECATED uint32_t identifier() const noexcept; _TVG_DECLARE_PRIVATE(Fill); }; @@ -538,7 +600,7 @@ class TVG_API Canvas * * This function provides the list of paint nodes, allowing users a direct opportunity to modify the scene tree. * - * @warning Please avoid accessing the paints during Canvas update/draw. You can access them after calling sync(). + * @warning Please avoid accessing the paints during Canvas update/draw. You can access them after calling sync(). * @see Canvas::sync() * * @note Experimental API @@ -614,7 +676,7 @@ class TVG_API Canvas * @warning It's not allowed to change the viewport during Canvas::push() - Canvas::sync() or Canvas::update() - Canvas::sync(). * * @note When resetting the target, the viewport will also be reset to the target size. - * @note Experimental API + * @since 0.15 */ virtual Result viewport(int32_t x, int32_t y, int32_t w, int32_t h) noexcept; @@ -686,13 +748,20 @@ class TVG_API LinearGradient final : public Fill static std::unique_ptr gen() noexcept; /** - * @brief Return the unique id value of this class. + * @brief Returns the ID value of this class. * - * This method can be referred for identifying the LinearGradient class type. + * This method can be used to check the current concrete instance type. * - * @return The type id of the LinearGradient class. + * @return The class type ID of the LinearGradient instance. + * + * @since Experimental API + */ + Type type() const noexcept override; + + /** + * @see LinearGradient::type() */ - static uint32_t identifier() noexcept; + TVG_DEPRECATED static uint32_t identifier() noexcept; _TVG_DECLARE_PRIVATE(LinearGradient); }; @@ -744,13 +813,20 @@ class TVG_API RadialGradient final : public Fill static std::unique_ptr gen() noexcept; /** - * @brief Return the unique id value of this class. + * @brief Returns the ID value of this class. * - * This method can be referred for identifying the RadialGradient class type. + * This method can be used to check the current concrete instance type. * - * @return The type id of the RadialGradient class. + * @return The class type ID of the LinearGradient instance. + * + * @since Experimental API + */ + Type type() const noexcept override; + + /** + * @see RadialGradient::type() */ - static uint32_t identifier() noexcept; + TVG_DEPRECATED static uint32_t identifier() noexcept; _TVG_DECLARE_PRIVATE(RadialGradient); }; @@ -774,11 +850,11 @@ class TVG_API Shape final : public Paint ~Shape(); /** - * @brief Resets the properties of the shape path. + * @brief Resets the shape path. * - * The transformation matrix, the color, the fill and the stroke properties are retained. + * The transformation matrix, color, fill, and stroke properties are retained. * - * @note The memory, where the path data is stored, is not deallocated at this stage for caching effect. + * @note The memory where the path data is stored is not deallocated at this stage to allow for caching. */ Result reset() noexcept; @@ -836,15 +912,15 @@ class TVG_API Shape final : public Paint * The rectangle with rounded corners can be achieved by setting non-zero values to @p rx and @p ry arguments. * The @p rx and @p ry values specify the radii of the ellipse defining the rounding of the corners. * - * The position of the rectangle is specified by the coordinates of its upper left corner - @p x and @p y arguments. + * The position of the rectangle is specified by the coordinates of its upper-left corner - @p x and @p y arguments. * * The rectangle is treated as a new sub-path - it is not connected with the previous sub-path. * * The value of the current point is set to (@p x + @p rx, @p y) - in case @p rx is greater * than @p w/2 the current point is set to (@p x + @p w/2, @p y) * - * @param[in] x The horizontal coordinate of the upper left corner of the rectangle. - * @param[in] y The vertical coordinate of the upper left corner of the rectangle. + * @param[in] x The horizontal coordinate of the upper-left corner of the rectangle. + * @param[in] y The vertical coordinate of the upper-left corner of the rectangle. * @param[in] w The width of the rectangle. * @param[in] h The height of the rectangle. * @param[in] rx The x-axis radius of the ellipse defining the rounded corners of the rectangle. @@ -886,7 +962,7 @@ class TVG_API Shape final : public Paint * * @note Setting @p sweep value greater than 360 degrees, is equivalent to calling appendCircle(cx, cy, radius, radius). */ - Result appendArc(float cx, float cy, float radius, float startAngle, float sweep, bool pie) noexcept; + TVG_DEPRECATED Result appendArc(float cx, float cy, float radius, float startAngle, float sweep, bool pie) noexcept; /** * @brief Appends a given sub-path to the path. @@ -999,7 +1075,6 @@ class TVG_API Shape final : public Paint * @param[in] a The alpha channel value in the range [0 ~ 255], where 0 is completely transparent and 255 is opaque. The default value is 0. * * @note Either a solid color or a gradient fill is applied, depending on what was set as last. - * @note ClipPath won't use the fill values. (see: enum class CompositeMethod::ClipPath) */ Result fill(uint8_t r, uint8_t g, uint8_t b, uint8_t a = 255) noexcept; @@ -1130,18 +1205,6 @@ class TVG_API Shape final : public Paint */ float strokeMiterlimit() const noexcept; - /** - * @brief Gets the trim of the stroke along the defined path segment. - * - * @param[out] begin The starting point of the segment to display along the path. - * @param[out] end Specifies the end of the segment to display along the path. - * - * @return @c true if trimming is applied simultaneously to all paths of the shape, @c false otherwise. - * - * @note Experimental API - */ - bool strokeTrim(float* begin, float* end) const noexcept; - /** * @brief Creates a new Shape object. * @@ -1150,13 +1213,20 @@ class TVG_API Shape final : public Paint static std::unique_ptr gen() noexcept; /** - * @brief Return the unique id value of this class. + * @brief Returns the ID value of this class. * - * This method can be referred for identifying the Shape class type. + * This method can be used to check the current concrete instance type. * - * @return The type id of the Shape class. + * @return The class type ID of the Shape instance. + * + * @since Experimental API */ - static uint32_t identifier() noexcept; + Type type() const noexcept override; + + /** + * @see Shape::type() + */ + TVG_DEPRECATED static uint32_t identifier() noexcept; _TVG_DECLARE_PRIVATE(Shape); }; @@ -1213,7 +1283,7 @@ class TVG_API Picture final : public Paint * @retval Result::InvalidArguments In case no data are provided or the @p size is zero or less. * @retval Result::NonSupport When trying to load a file with an unknown extension. * - * @warning: It's the user responsibility to release the @p data memory. + * @warning It's the user responsibility to release the @p data memory. * * @note If you are unsure about the MIME type, you can provide an empty value like @c "", and thorvg will attempt to figure it out. * @since 0.5 @@ -1251,9 +1321,9 @@ class TVG_API Picture final : public Paint * @param[in] data A pointer to a memory location where the content of the picture raw data is stored. * @param[in] w The width of the image @p data in pixels. * @param[in] h The height of the image @p data in pixels. + * @param[in] premultiplied If @c true, the given image data is alpha-premultiplied. * @param[in] copy If @c true the data are copied into the engine local buffer, otherwise they are not. * - * @note It expects premultiplied alpha data. * @since 0.9 */ Result load(uint32_t* data, uint32_t w, uint32_t h, bool copy) noexcept; @@ -1281,13 +1351,20 @@ class TVG_API Picture final : public Paint static std::unique_ptr gen() noexcept; /** - * @brief Return the unique id value of this class. + * @brief Returns the ID value of this class. * - * This method can be referred for identifying the Picture class type. + * This method can be used to check the current concrete instance type. * - * @return The type id of the Picture class. + * @return The class type ID of the Picture instance. + * + * @since Experimental API + */ + Type type() const noexcept override; + + /** + * @see Picture::type() */ - static uint32_t identifier() noexcept; + TVG_DEPRECATED static uint32_t identifier() noexcept; _TVG_DECLARE_ACCESSOR(Animation); _TVG_DECLARE_PRIVATE(Picture); @@ -1331,9 +1408,9 @@ class TVG_API Scene final : public Paint * * This function provides the list of paint nodes, allowing users a direct opportunity to modify the scene tree. * - * @warning Please avoid accessing the paints during Scene update/draw. You can access them after calling Canvas::sync(). + * @warning Please avoid accessing the paints during Scene update/draw. You can access them after calling Canvas::sync(). * @see Canvas::sync() - * @see Scene::push() + * @see Scene::push(std::unique_ptr paint) * @see Scene::clear() * * @note Experimental API @@ -1352,6 +1429,20 @@ class TVG_API Scene final : public Paint */ Result clear(bool free = true) noexcept; + /** + * @brief Apply a post-processing effect to the scene. + * + * This function adds a specified scene effect, such as clearing all effects or applying a Gaussian blur, + * to the scene after it has been rendered. Multiple effects can be applied in sequence. + * + * @param[in] effect The scene effect to apply. Options are defined in the SceneEffect enum. + * For example, use SceneEffect::GaussianBlur to apply a blur with specific parameters. + * @param[in] ... Additional variadic parameters required for certain effects (e.g., sigma and direction for GaussianBlur). + * + * @note Experimental API + */ + Result push(SceneEffect effect, ...) noexcept; + /** * @brief Creates a new Scene object. * @@ -1360,13 +1451,20 @@ class TVG_API Scene final : public Paint static std::unique_ptr gen() noexcept; /** - * @brief Return the unique id value of this class. + * @brief Returns the ID value of this class. * - * This method can be referred for identifying the Scene class type. + * This method can be used to check the current concrete instance type. * - * @return The type id of the Scene class. + * @return The class type ID of the Scene instance. + * + * @since Experimental API */ - static uint32_t identifier() noexcept; + Type type() const noexcept override; + + /** + * @see Scene::type() + */ + TVG_DEPRECATED static uint32_t identifier() noexcept; _TVG_DECLARE_PRIVATE(Scene); }; @@ -1377,7 +1475,7 @@ class TVG_API Scene final : public Paint * * @brief A class to represent text objects in a graphical context, allowing for rendering and manipulation of unicode text. * - * @note Experimental API + * @since 0.15 */ class TVG_API Text final : public Paint { @@ -1422,7 +1520,7 @@ class TVG_API Text final : public Paint * * @see Text::font() * - * @note Experimental API + * @since 0.15 */ Result fill(uint8_t r, uint8_t g, uint8_t b) noexcept; @@ -1434,9 +1532,9 @@ class TVG_API Text final : public Paint * @param[in] f The unique pointer to the gradient fill. * * @note Either a solid color or a gradient fill is applied, depending on what was set as last. - * @note Experimental API - * * @see Text::font() + * + * @since 0.15 */ Result fill(std::unique_ptr f) noexcept; @@ -1452,9 +1550,9 @@ class TVG_API Text final : public Paint * @retval Result::InvalidArguments In case the @p path is invalid. * @retval Result::NonSupport When trying to load a file with an unknown extension. * - * @note Experimental API - * * @see Text::unload(const std::string& path) + * + * @since 0.15 */ static Result load(const std::string& path) noexcept; @@ -1475,13 +1573,13 @@ class TVG_API Text final : public Paint * @retval Result::NonSupport When trying to load a file with an unsupported extension. * @retval Result::InsufficientCondition If attempting to unload the font data that has not been previously loaded. * - * @warning: It's the user responsibility to release the @p data memory. + * @warning It's the user responsibility to release the @p data memory. * * @note To unload the font data loaded using this API, pass the proper @p name and @c nullptr as @p data. * @note If you are unsure about the MIME type, you can provide an empty value like @c "", and thorvg will attempt to figure it out. - * @note Experimental API - * * @see Text::font(const char* name, float size, const char* style) + * + * @note 0.15 */ static Result load(const char* name, const char* data, uint32_t size, const std::string& mimeType = "ttf", bool copy = false) noexcept; @@ -1495,9 +1593,9 @@ class TVG_API Text final : public Paint * @retval Result::InsufficientCondition Fails if the loader is not initialized. * * @note If the font data is currently in use, it will not be immediately unloaded. - * @note Experimental API - * * @see Text::load(const std::string& path) + * + * @since 0.15 */ static Result unload(const std::string& path) noexcept; @@ -1506,18 +1604,20 @@ class TVG_API Text final : public Paint * * @return A new Text object. * - * @note Experimental API + * @since 0.15 */ static std::unique_ptr gen() noexcept; /** - * @brief Return the unique id value of this class. + * @brief Returns the ID value of this class. + * + * This method can be used to check the current concrete instance type. * - * This method can be referred for identifying the Text class type. + * @return The class type ID of the Text instance. * - * @return The type id of the Text class. + * @since Experimental API */ - static uint32_t identifier() noexcept; + Type type() const noexcept override; _TVG_DECLARE_PRIVATE(Text); }; @@ -1616,8 +1716,6 @@ class TVG_API SwCanvas final : public Canvas * * @brief A class for the rendering graphic elements with a GL raster engine. * - * @warning Please do not use it. This class is not fully supported yet. - * * @since 0.14 */ class TVG_API GlCanvas final : public Canvas @@ -1666,7 +1764,7 @@ class TVG_API GlCanvas final : public Canvas * * @warning Please do not use it. This class is not fully supported yet. * - * @note Experimental API + * @since 0.15 */ class TVG_API WgCanvas final : public Canvas { @@ -1680,6 +1778,7 @@ class TVG_API WgCanvas final : public Canvas * @param[in] surface WGPUSurface, handle to a presentable surface. * @param[in] w The width of the surface. * @param[in] h The height of the surface. + * @param[in] device WGPUDevice, a desired handle for the wgpu device. If it is @c nullptr, ThorVG will assign an appropriate device internally. * * @retval Result::InsufficientCondition if the canvas is performing rendering. Please ensure the canvas is synced. * @retval Result::NonSupport In case the wg engine is not supported. @@ -1689,14 +1788,14 @@ class TVG_API WgCanvas final : public Canvas * @see Canvas::viewport() * @see Canvas::sync() */ - Result target(void* instance, void* surface, uint32_t w, uint32_t h) noexcept; + Result target(void* instance, void* surface, uint32_t w, uint32_t h, void* device = nullptr) noexcept; /** * @brief Creates a new WgCanvas object. * * @return A new WgCanvas object. * - * @note Experimental API + * @since 0.15 */ static std::unique_ptr gen() noexcept; @@ -1752,7 +1851,7 @@ class TVG_API Initializer final * * @return The version of the engine in the format major.minor.micro, or a @p nullptr in case of an internal error. * - * @note Experimental API + * @since 0.15 */ static const char* version(uint32_t* major, uint32_t* minor, uint32_t* micro) noexcept; @@ -1857,6 +1956,7 @@ class TVG_API Animation * @note Animation allows a range from 0.0 to 1.0. @p end should not be higher than @p begin. * @note If a marker has been specified, its range will be disregarded. * @see LottieAnimation::segment(const char* marker) + * * @note Experimental API */ Result segment(float begin, float end) noexcept; @@ -1895,7 +1995,7 @@ class TVG_API Animation * It's useful when you need to save the composed scene or image from a paint object and recreate it later. * * The file format is decided by the extension name(i.e. "*.tvg") while the supported formats depend on the TVG packaging environment. - * If it doesn't support the file format, the save() method returns the @c Result::NonSuppport result. + * If it doesn't support the file format, the save() method returns the @c Result::NonSupport result. * * Once you export a paint to the file successfully, you can recreate it using the Picture class. * diff --git a/thirdparty/thorvg/src/common/tvgCompressor.cpp b/thirdparty/thorvg/src/common/tvgCompressor.cpp index aebe9a4ef164..714f21e07c9d 100644 --- a/thirdparty/thorvg/src/common/tvgCompressor.cpp +++ b/thirdparty/thorvg/src/common/tvgCompressor.cpp @@ -468,7 +468,7 @@ size_t b64Decode(const char* encoded, const size_t len, char** decoded) encoded += 4; } *decoded = output; - return reserved; + return idx; } diff --git a/thirdparty/thorvg/src/common/tvgLines.cpp b/thirdparty/thorvg/src/common/tvgLines.cpp deleted file mode 100644 index 9d704900a50e..000000000000 --- a/thirdparty/thorvg/src/common/tvgLines.cpp +++ /dev/null @@ -1,245 +0,0 @@ -/* - * Copyright (c) 2020 - 2024 the ThorVG project. All rights reserved. - - * 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 "tvgMath.h" -#include "tvgLines.h" - -#define BEZIER_EPSILON 1e-2f - -/************************************************************************/ -/* Internal Class Implementation */ -/************************************************************************/ - -static float _lineLengthApprox(const Point& pt1, const Point& pt2) -{ - /* approximate sqrt(x*x + y*y) using alpha max plus beta min algorithm. - With alpha = 1, beta = 3/8, giving results with the largest error less - than 7% compared to the exact value. */ - Point diff = {pt2.x - pt1.x, pt2.y - pt1.y}; - if (diff.x < 0) diff.x = -diff.x; - if (diff.y < 0) diff.y = -diff.y; - return (diff.x > diff.y) ? (diff.x + diff.y * 0.375f) : (diff.y + diff.x * 0.375f); -} - - -static float _lineLength(const Point& pt1, const Point& pt2) -{ - Point diff = {pt2.x - pt1.x, pt2.y - pt1.y}; - return sqrtf(diff.x * diff.x + diff.y * diff.y); -} - - -template -float _bezLength(const Bezier& cur, LengthFunc lineLengthFunc) -{ - Bezier left, right; - auto len = lineLengthFunc(cur.start, cur.ctrl1) + lineLengthFunc(cur.ctrl1, cur.ctrl2) + lineLengthFunc(cur.ctrl2, cur.end); - auto chord = lineLengthFunc(cur.start, cur.end); - - if (fabsf(len - chord) > BEZIER_EPSILON) { - tvg::bezSplit(cur, left, right); - return _bezLength(left, lineLengthFunc) + _bezLength(right, lineLengthFunc); - } - return len; -} - - -template -float _bezAt(const Bezier& bz, float at, float length, LengthFunc lineLengthFunc) -{ - auto biggest = 1.0f; - auto smallest = 0.0f; - auto t = 0.5f; - - //just in case to prevent an infinite loop - if (at <= 0) return 0.0f; - if (at >= length) return 1.0f; - - while (true) { - auto right = bz; - Bezier left; - bezSplitLeft(right, t, left); - length = _bezLength(left, lineLengthFunc); - if (fabsf(length - at) < BEZIER_EPSILON || fabsf(smallest - biggest) < BEZIER_EPSILON) { - break; - } - if (length < at) { - smallest = t; - t = (t + biggest) * 0.5f; - } else { - biggest = t; - t = (smallest + t) * 0.5f; - } - } - return t; -} - - -/************************************************************************/ -/* External Class Implementation */ -/************************************************************************/ - -namespace tvg -{ - -float lineLength(const Point& pt1, const Point& pt2) -{ - return _lineLength(pt1, pt2); -} - - -void lineSplitAt(const Line& cur, float at, Line& left, Line& right) -{ - auto len = lineLength(cur.pt1, cur.pt2); - auto dx = ((cur.pt2.x - cur.pt1.x) / len) * at; - auto dy = ((cur.pt2.y - cur.pt1.y) / len) * at; - left.pt1 = cur.pt1; - left.pt2.x = left.pt1.x + dx; - left.pt2.y = left.pt1.y + dy; - right.pt1 = left.pt2; - right.pt2 = cur.pt2; -} - - -void bezSplit(const Bezier& cur, Bezier& left, Bezier& right) -{ - auto c = (cur.ctrl1.x + cur.ctrl2.x) * 0.5f; - left.ctrl1.x = (cur.start.x + cur.ctrl1.x) * 0.5f; - right.ctrl2.x = (cur.ctrl2.x + cur.end.x) * 0.5f; - left.start.x = cur.start.x; - right.end.x = cur.end.x; - left.ctrl2.x = (left.ctrl1.x + c) * 0.5f; - right.ctrl1.x = (right.ctrl2.x + c) * 0.5f; - left.end.x = right.start.x = (left.ctrl2.x + right.ctrl1.x) * 0.5f; - - c = (cur.ctrl1.y + cur.ctrl2.y) * 0.5f; - left.ctrl1.y = (cur.start.y + cur.ctrl1.y) * 0.5f; - right.ctrl2.y = (cur.ctrl2.y + cur.end.y) * 0.5f; - left.start.y = cur.start.y; - right.end.y = cur.end.y; - left.ctrl2.y = (left.ctrl1.y + c) * 0.5f; - right.ctrl1.y = (right.ctrl2.y + c) * 0.5f; - left.end.y = right.start.y = (left.ctrl2.y + right.ctrl1.y) * 0.5f; -} - - -float bezLength(const Bezier& cur) -{ - return _bezLength(cur, _lineLength); -} - - -float bezLengthApprox(const Bezier& cur) -{ - return _bezLength(cur, _lineLengthApprox); -} - - -void bezSplitLeft(Bezier& cur, float at, Bezier& left) -{ - left.start = cur.start; - - left.ctrl1.x = cur.start.x + at * (cur.ctrl1.x - cur.start.x); - left.ctrl1.y = cur.start.y + at * (cur.ctrl1.y - cur.start.y); - - left.ctrl2.x = cur.ctrl1.x + at * (cur.ctrl2.x - cur.ctrl1.x); //temporary holding spot - left.ctrl2.y = cur.ctrl1.y + at * (cur.ctrl2.y - cur.ctrl1.y); //temporary holding spot - - cur.ctrl2.x = cur.ctrl2.x + at * (cur.end.x - cur.ctrl2.x); - cur.ctrl2.y = cur.ctrl2.y + at * (cur.end.y - cur.ctrl2.y); - - cur.ctrl1.x = left.ctrl2.x + at * (cur.ctrl2.x - left.ctrl2.x); - cur.ctrl1.y = left.ctrl2.y + at * (cur.ctrl2.y - left.ctrl2.y); - - left.ctrl2.x = left.ctrl1.x + at * (left.ctrl2.x - left.ctrl1.x); - left.ctrl2.y = left.ctrl1.y + at * (left.ctrl2.y - left.ctrl1.y); - - left.end.x = cur.start.x = left.ctrl2.x + at * (cur.ctrl1.x - left.ctrl2.x); - left.end.y = cur.start.y = left.ctrl2.y + at * (cur.ctrl1.y - left.ctrl2.y); -} - - -float bezAt(const Bezier& bz, float at, float length) -{ - return _bezAt(bz, at, length, _lineLength); -} - - -float bezAtApprox(const Bezier& bz, float at, float length) -{ - return _bezAt(bz, at, length, _lineLengthApprox); -} - - -void bezSplitAt(const Bezier& cur, float at, Bezier& left, Bezier& right) -{ - right = cur; - auto t = bezAt(right, at, bezLength(right)); - bezSplitLeft(right, t, left); -} - - -Point bezPointAt(const Bezier& bz, float t) -{ - Point cur; - auto it = 1.0f - t; - - auto ax = bz.start.x * it + bz.ctrl1.x * t; - auto bx = bz.ctrl1.x * it + bz.ctrl2.x * t; - auto cx = bz.ctrl2.x * it + bz.end.x * t; - ax = ax * it + bx * t; - bx = bx * it + cx * t; - cur.x = ax * it + bx * t; - - float ay = bz.start.y * it + bz.ctrl1.y * t; - float by = bz.ctrl1.y * it + bz.ctrl2.y * t; - float cy = bz.ctrl2.y * it + bz.end.y * t; - ay = ay * it + by * t; - by = by * it + cy * t; - cur.y = ay * it + by * t; - - return cur; -} - - -float bezAngleAt(const Bezier& bz, float t) -{ - if (t < 0 || t > 1) return 0; - - //derivate - // p'(t) = 3 * (-(1-2t+t^2) * p0 + (1 - 4 * t + 3 * t^2) * p1 + (2 * t - 3 * - // t^2) * p2 + t^2 * p3) - float mt = 1.0f - t; - float d = t * t; - float a = -mt * mt; - float b = 1 - 4 * t + 3 * d; - float c = 2 * t - 3 * d; - - Point pt ={a * bz.start.x + b * bz.ctrl1.x + c * bz.ctrl2.x + d * bz.end.x, a * bz.start.y + b * bz.ctrl1.y + c * bz.ctrl2.y + d * bz.end.y}; - pt.x *= 3; - pt.y *= 3; - - return mathRad2Deg(mathAtan2(pt.y, pt.x)); -} - - -} diff --git a/thirdparty/thorvg/src/common/tvgLines.h b/thirdparty/thorvg/src/common/tvgLines.h deleted file mode 100644 index d900782b5627..000000000000 --- a/thirdparty/thorvg/src/common/tvgLines.h +++ /dev/null @@ -1,61 +0,0 @@ -/* - * Copyright (c) 2020 - 2024 the ThorVG project. All rights reserved. - - * 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 _TVG_LINES_H_ -#define _TVG_LINES_H_ - -#include "tvgCommon.h" - -namespace tvg -{ - -struct Line -{ - Point pt1; - Point pt2; -}; - -float lineLength(const Point& pt1, const Point& pt2); -void lineSplitAt(const Line& cur, float at, Line& left, Line& right); - - -struct Bezier -{ - Point start; - Point ctrl1; - Point ctrl2; - Point end; -}; - -void bezSplit(const Bezier&cur, Bezier& left, Bezier& right); -float bezLength(const Bezier& cur); -void bezSplitLeft(Bezier& cur, float at, Bezier& left); -float bezAt(const Bezier& bz, float at, float length); -void bezSplitAt(const Bezier& cur, float at, Bezier& left, Bezier& right); -Point bezPointAt(const Bezier& bz, float t); -float bezAngleAt(const Bezier& bz, float t); - -float bezLengthApprox(const Bezier& cur); -float bezAtApprox(const Bezier& bz, float at, float length); -} - -#endif //_TVG_LINES_H_ diff --git a/thirdparty/thorvg/src/common/tvgMath.cpp b/thirdparty/thorvg/src/common/tvgMath.cpp index c03b54e5f8b9..cb7f24ff4030 100644 --- a/thirdparty/thorvg/src/common/tvgMath.cpp +++ b/thirdparty/thorvg/src/common/tvgMath.cpp @@ -22,11 +22,88 @@ #include "tvgMath.h" -//see: https://en.wikipedia.org/wiki/Remez_algorithm -float mathAtan2(float y, float x) +#define BEZIER_EPSILON 1e-2f + + +/************************************************************************/ +/* Internal Class Implementation */ +/************************************************************************/ + +static float _lineLengthApprox(const Point& pt1, const Point& pt2) { - if (y == 0.0f && x == 0.0f) return 0.0f; + /* approximate sqrt(x*x + y*y) using alpha max plus beta min algorithm. + With alpha = 1, beta = 3/8, giving results with the largest error less + than 7% compared to the exact value. */ + Point diff = {pt2.x - pt1.x, pt2.y - pt1.y}; + if (diff.x < 0) diff.x = -diff.x; + if (diff.y < 0) diff.y = -diff.y; + return (diff.x > diff.y) ? (diff.x + diff.y * 0.375f) : (diff.y + diff.x * 0.375f); +} + + +static float _lineLength(const Point& pt1, const Point& pt2) +{ + Point diff = {pt2.x - pt1.x, pt2.y - pt1.y}; + return sqrtf(diff.x * diff.x + diff.y * diff.y); +} + + +template +float _bezLength(const Bezier& cur, LengthFunc lineLengthFunc) +{ + Bezier left, right; + auto len = lineLengthFunc(cur.start, cur.ctrl1) + lineLengthFunc(cur.ctrl1, cur.ctrl2) + lineLengthFunc(cur.ctrl2, cur.end); + auto chord = lineLengthFunc(cur.start, cur.end); + + if (fabsf(len - chord) > BEZIER_EPSILON) { + cur.split(left, right); + return _bezLength(left, lineLengthFunc) + _bezLength(right, lineLengthFunc); + } + return len; +} + + +template +float _bezAt(const Bezier& bz, float at, float length, LengthFunc lineLengthFunc) +{ + auto biggest = 1.0f; + auto smallest = 0.0f; + auto t = 0.5f; + + //just in case to prevent an infinite loop + if (at <= 0) return 0.0f; + if (at >= length) return 1.0f; + + while (true) { + auto right = bz; + Bezier left; + right.split(t, left); + length = _bezLength(left, lineLengthFunc); + if (fabsf(length - at) < BEZIER_EPSILON || fabsf(smallest - biggest) < 1e-3f) { + break; + } + if (length < at) { + smallest = t; + t = (t + biggest) * 0.5f; + } else { + biggest = t; + t = (smallest + t) * 0.5f; + } + } + return t; +} + + +/************************************************************************/ +/* External Class Implementation */ +/************************************************************************/ +namespace tvg { + +//https://en.wikipedia.org/wiki/Remez_algorithm +float atan2(float y, float x) +{ + if (y == 0.0f && x == 0.0f) return 0.0f; auto a = std::min(fabsf(x), fabsf(y)) / std::max(fabsf(x), fabsf(y)); auto s = a * a; auto r = ((-0.0464964749f * s + 0.15931422f) * s - 0.327622764f) * s * a + a; @@ -37,7 +114,7 @@ float mathAtan2(float y, float x) } -bool mathInverse(const Matrix* m, Matrix* out) +bool inverse(const Matrix* m, Matrix* out) { auto det = m->e11 * (m->e22 * m->e33 - m->e32 * m->e23) - m->e12 * (m->e21 * m->e33 - m->e23 * m->e31) + @@ -60,7 +137,7 @@ bool mathInverse(const Matrix* m, Matrix* out) } -bool mathIdentity(const Matrix* m) +bool identity(const Matrix* m) { if (m->e11 != 1.0f || m->e12 != 0.0f || m->e13 != 0.0f || m->e21 != 0.0f || m->e22 != 1.0f || m->e23 != 0.0f || @@ -71,7 +148,7 @@ bool mathIdentity(const Matrix* m) } -void mathRotate(Matrix* m, float degree) +void rotate(Matrix* m, float degree) { if (degree == 0.0f) return; @@ -108,9 +185,9 @@ Matrix operator*(const Matrix& lhs, const Matrix& rhs) bool operator==(const Matrix& lhs, const Matrix& rhs) { - if (!mathEqual(lhs.e11, rhs.e11) || !mathEqual(lhs.e12, rhs.e12) || !mathEqual(lhs.e13, rhs.e13) || - !mathEqual(lhs.e21, rhs.e21) || !mathEqual(lhs.e22, rhs.e22) || !mathEqual(lhs.e23, rhs.e23) || - !mathEqual(lhs.e31, rhs.e31) || !mathEqual(lhs.e32, rhs.e32) || !mathEqual(lhs.e33, rhs.e33)) { + if (!tvg::equal(lhs.e11, rhs.e11) || !tvg::equal(lhs.e12, rhs.e12) || !tvg::equal(lhs.e13, rhs.e13) || + !tvg::equal(lhs.e21, rhs.e21) || !tvg::equal(lhs.e22, rhs.e22) || !tvg::equal(lhs.e23, rhs.e23) || + !tvg::equal(lhs.e31, rhs.e31) || !tvg::equal(lhs.e32, rhs.e32) || !tvg::equal(lhs.e33, rhs.e33)) { return false; } return true; @@ -133,9 +210,165 @@ Point operator*(const Point& pt, const Matrix& m) return {tx, ty}; } -uint8_t mathLerp(const uint8_t &start, const uint8_t &end, float t) + +Point normal(const Point& p1, const Point& p2) +{ + auto dir = p2 - p1; + auto len = length(dir); + if (tvg::zero(len)) return {}; + + auto unitDir = dir / len; + return {-unitDir.y, unitDir.x}; +} + + +float Line::length() const +{ + return _lineLength(pt1, pt2); +} + + +void Line::split(float at, Line& left, Line& right) const +{ + auto len = length(); + auto dx = ((pt2.x - pt1.x) / len) * at; + auto dy = ((pt2.y - pt1.y) / len) * at; + left.pt1 = pt1; + left.pt2.x = left.pt1.x + dx; + left.pt2.y = left.pt1.y + dy; + right.pt1 = left.pt2; + right.pt2 = pt2; +} + + +void Bezier::split(Bezier& left, Bezier& right) const +{ + auto c = (ctrl1.x + ctrl2.x) * 0.5f; + left.ctrl1.x = (start.x + ctrl1.x) * 0.5f; + right.ctrl2.x = (ctrl2.x + end.x) * 0.5f; + left.start.x = start.x; + right.end.x = end.x; + left.ctrl2.x = (left.ctrl1.x + c) * 0.5f; + right.ctrl1.x = (right.ctrl2.x + c) * 0.5f; + left.end.x = right.start.x = (left.ctrl2.x + right.ctrl1.x) * 0.5f; + + c = (ctrl1.y + ctrl2.y) * 0.5f; + left.ctrl1.y = (start.y + ctrl1.y) * 0.5f; + right.ctrl2.y = (ctrl2.y + end.y) * 0.5f; + left.start.y = start.y; + right.end.y = end.y; + left.ctrl2.y = (left.ctrl1.y + c) * 0.5f; + right.ctrl1.y = (right.ctrl2.y + c) * 0.5f; + left.end.y = right.start.y = (left.ctrl2.y + right.ctrl1.y) * 0.5f; +} + + +void Bezier::split(float at, Bezier& left, Bezier& right) const +{ + right = *this; + auto t = right.at(at, right.length()); + right.split(t, left); +} + + +float Bezier::length() const +{ + return _bezLength(*this, _lineLength); +} + + +float Bezier::lengthApprox() const +{ + return _bezLength(*this, _lineLengthApprox); +} + + +void Bezier::split(float t, Bezier& left) +{ + left.start = start; + + left.ctrl1.x = start.x + t * (ctrl1.x - start.x); + left.ctrl1.y = start.y + t * (ctrl1.y - start.y); + + left.ctrl2.x = ctrl1.x + t * (ctrl2.x - ctrl1.x); //temporary holding spot + left.ctrl2.y = ctrl1.y + t * (ctrl2.y - ctrl1.y); //temporary holding spot + + ctrl2.x = ctrl2.x + t * (end.x - ctrl2.x); + ctrl2.y = ctrl2.y + t * (end.y - ctrl2.y); + + ctrl1.x = left.ctrl2.x + t * (ctrl2.x - left.ctrl2.x); + ctrl1.y = left.ctrl2.y + t * (ctrl2.y - left.ctrl2.y); + + left.ctrl2.x = left.ctrl1.x + t * (left.ctrl2.x - left.ctrl1.x); + left.ctrl2.y = left.ctrl1.y + t * (left.ctrl2.y - left.ctrl1.y); + + left.end.x = start.x = left.ctrl2.x + t * (ctrl1.x - left.ctrl2.x); + left.end.y = start.y = left.ctrl2.y + t * (ctrl1.y - left.ctrl2.y); +} + + +float Bezier::at(float at, float length) const +{ + return _bezAt(*this, at, length, _lineLength); +} + + +float Bezier::atApprox(float at, float length) const +{ + return _bezAt(*this, at, length, _lineLengthApprox); +} + + +Point Bezier::at(float t) const +{ + Point cur; + auto it = 1.0f - t; + + auto ax = start.x * it + ctrl1.x * t; + auto bx = ctrl1.x * it + ctrl2.x * t; + auto cx = ctrl2.x * it + end.x * t; + ax = ax * it + bx * t; + bx = bx * it + cx * t; + cur.x = ax * it + bx * t; + + float ay = start.y * it + ctrl1.y * t; + float by = ctrl1.y * it + ctrl2.y * t; + float cy = ctrl2.y * it + end.y * t; + ay = ay * it + by * t; + by = by * it + cy * t; + cur.y = ay * it + by * t; + + return cur; +} + + +float Bezier::angle(float t) const +{ + if (t < 0 || t > 1) return 0; + + //derivate + // p'(t) = 3 * (-(1-2t+t^2) * p0 + (1 - 4 * t + 3 * t^2) * p1 + (2 * t - 3 * + // t^2) * p2 + t^2 * p3) + float mt = 1.0f - t; + float d = t * t; + float a = -mt * mt; + float b = 1 - 4 * t + 3 * d; + float c = 2 * t - 3 * d; + + Point pt ={a * start.x + b * ctrl1.x + c * ctrl2.x + d * end.x, a * start.y + b * ctrl1.y + c * ctrl2.y + d * end.y}; + pt.x *= 3; + pt.y *= 3; + + return rad2deg(tvg::atan2(pt.y, pt.x)); +} + + +uint8_t lerp(const uint8_t &start, const uint8_t &end, float t) { auto result = static_cast(start + (end - start) * t); - mathClamp(result, 0, 255); + tvg::clamp(result, 0, 255); return static_cast(result); } + +} + diff --git a/thirdparty/thorvg/src/common/tvgMath.h b/thirdparty/thorvg/src/common/tvgMath.h index 50786754a159..a917998256fd 100644 --- a/thirdparty/thorvg/src/common/tvgMath.h +++ b/thirdparty/thorvg/src/common/tvgMath.h @@ -29,47 +29,47 @@ #include #include "tvgCommon.h" +namespace tvg +{ + #define MATH_PI 3.14159265358979323846f #define MATH_PI2 1.57079632679489661923f #define FLOAT_EPSILON 1.0e-06f //1.192092896e-07f #define PATH_KAPPA 0.552284f -#define mathMin(x, y) (((x) < (y)) ? (x) : (y)) -#define mathMax(x, y) (((x) > (y)) ? (x) : (y)) - - /************************************************************************/ /* General functions */ /************************************************************************/ -float mathAtan2(float y, float x); +float atan2(float y, float x); -static inline float mathDeg2Rad(float degree) + +static inline float deg2rad(float degree) { return degree * (MATH_PI / 180.0f); } -static inline float mathRad2Deg(float radian) +static inline float rad2deg(float radian) { return radian * (180.0f / MATH_PI); } -static inline bool mathZero(float a) +static inline bool zero(float a) { return (fabsf(a) <= FLOAT_EPSILON) ? true : false; } -static inline bool mathEqual(float a, float b) +static inline bool equal(float a, float b) { - return mathZero(a - b); + return tvg::zero(a - b); } template -static inline void mathClamp(T& v, const T& min, const T& max) +static inline void clamp(T& v, const T& min, const T& max) { if (v < min) v = min; else if (v > max) v = max; @@ -79,27 +79,27 @@ static inline void mathClamp(T& v, const T& min, const T& max) /* Matrix functions */ /************************************************************************/ -void mathRotate(Matrix* m, float degree); -bool mathInverse(const Matrix* m, Matrix* out); -bool mathIdentity(const Matrix* m); +void rotate(Matrix* m, float degree); +bool inverse(const Matrix* m, Matrix* out); +bool identity(const Matrix* m); Matrix operator*(const Matrix& lhs, const Matrix& rhs); bool operator==(const Matrix& lhs, const Matrix& rhs); -static inline bool mathRightAngle(const Matrix& m) +static inline bool rightAngle(const Matrix& m) { - auto radian = fabsf(mathAtan2(m.e21, m.e11)); - if (radian < FLOAT_EPSILON || mathEqual(radian, MATH_PI2) || mathEqual(radian, MATH_PI)) return true; + auto radian = fabsf(tvg::atan2(m.e21, m.e11)); + if (radian < FLOAT_EPSILON || tvg::equal(radian, MATH_PI2) || tvg::equal(radian, MATH_PI)) return true; return false; } -static inline bool mathSkewed(const Matrix& m) +static inline bool skewed(const Matrix& m) { - return !mathZero(m.e21 + m.e12); + return !tvg::zero(m.e21 + m.e12); } -static inline void mathIdentity(Matrix* m) +static inline void identity(Matrix* m) { m->e11 = 1.0f; m->e12 = 0.0f; @@ -113,14 +113,14 @@ static inline void mathIdentity(Matrix* m) } -static inline void mathScale(Matrix* m, float sx, float sy) +static inline void scale(Matrix* m, float sx, float sy) { m->e11 *= sx; m->e22 *= sy; } -static inline void mathScaleR(Matrix* m, float x, float y) +static inline void scaleR(Matrix* m, float x, float y) { if (x != 1.0f) { m->e11 *= x; @@ -133,14 +133,14 @@ static inline void mathScaleR(Matrix* m, float x, float y) } -static inline void mathTranslate(Matrix* m, float x, float y) +static inline void translate(Matrix* m, float x, float y) { m->e13 += x; m->e23 += y; } -static inline void mathTranslateR(Matrix* m, float x, float y) +static inline void translateR(Matrix* m, float x, float y) { if (x == 0.0f && y == 0.0f) return; m->e13 += (x * m->e11 + y * m->e12); @@ -160,7 +160,7 @@ static inline void operator*=(Matrix& lhs, const Matrix& rhs) } -static inline void mathLog(const Matrix& m) +static inline void log(const Matrix& m) { TVGLOG("COMMON", "Matrix: [%f %f %f] [%f %f %f] [%f %f %f]", m.e11, m.e12, m.e13, m.e21, m.e22, m.e23, m.e31, m.e32, m.e33); } @@ -172,15 +172,21 @@ static inline void mathLog(const Matrix& m) void operator*=(Point& pt, const Matrix& m); Point operator*(const Point& pt, const Matrix& m); +Point normal(const Point& p1, const Point& p2); +static inline float cross(const Point& lhs, const Point& rhs) +{ + return lhs.x * rhs.y - rhs.x * lhs.y; +} -static inline bool mathZero(const Point& p) + +static inline bool zero(const Point& p) { - return mathZero(p.x) && mathZero(p.y); + return tvg::zero(p.x) && tvg::zero(p.y); } -static inline float mathLength(const Point* a, const Point* b) +static inline float length(const Point* a, const Point* b) { auto x = b->x - a->x; auto y = b->y - a->y; @@ -192,7 +198,7 @@ static inline float mathLength(const Point* a, const Point* b) } -static inline float mathLength(const Point& a) +static inline float length(const Point& a) { return sqrtf(a.x * a.x + a.y * a.y); } @@ -200,7 +206,7 @@ static inline float mathLength(const Point& a) static inline bool operator==(const Point& lhs, const Point& rhs) { - return mathEqual(lhs.x, rhs.x) && mathEqual(lhs.y, rhs.y); + return tvg::equal(lhs.x, rhs.x) && tvg::equal(lhs.y, rhs.y); } @@ -240,32 +246,61 @@ static inline Point operator/(const Point& lhs, const float rhs) } -static inline Point mathNormal(const Point& p1, const Point& p2) +static inline void log(const Point& pt) { - auto dir = p2 - p1; - auto len = mathLength(dir); - if (mathZero(len)) return {}; - - auto unitDir = dir / len; - return {-unitDir.y, unitDir.x}; + TVGLOG("COMMON", "Point: [%f %f]", pt.x, pt.y); } -static inline void mathLog(const Point& pt) +/************************************************************************/ +/* Line functions */ +/************************************************************************/ + +struct Line { - TVGLOG("COMMON", "Point: [%f %f]", pt.x, pt.y); -} + Point pt1; + Point pt2; + + void split(float at, Line& left, Line& right) const; + float length() const; +}; + + +/************************************************************************/ +/* Bezier functions */ +/************************************************************************/ + +struct Bezier +{ + Point start; + Point ctrl1; + Point ctrl2; + Point end; + + void split(float t, Bezier& left); + void split(Bezier& left, Bezier& right) const; + void split(float at, Bezier& left, Bezier& right) const; + float length() const; + float lengthApprox() const; + float at(float at, float length) const; + float atApprox(float at, float length) const; + Point at(float t) const; + float angle(float t) const; +}; + /************************************************************************/ /* Interpolation functions */ /************************************************************************/ template -static inline T mathLerp(const T &start, const T &end, float t) +static inline T lerp(const T &start, const T &end, float t) { return static_cast(start + (end - start) * t); } -uint8_t mathLerp(const uint8_t &start, const uint8_t &end, float t); +uint8_t lerp(const uint8_t &start, const uint8_t &end, float t); + +} #endif //_TVG_MATH_H_ diff --git a/thirdparty/thorvg/src/common/tvgStr.cpp b/thirdparty/thorvg/src/common/tvgStr.cpp index 3f1896680960..1ebdd41c5e92 100644 --- a/thirdparty/thorvg/src/common/tvgStr.cpp +++ b/thirdparty/thorvg/src/common/tvgStr.cpp @@ -207,16 +207,6 @@ float strToFloat(const char *nPtr, char **endPtr) return 0.0f; } - -int str2int(const char* str, size_t n) -{ - int ret = 0; - for(size_t i = 0; i < n; ++i) { - ret = ret * 10 + (str[i] - '0'); - } - return ret; -} - char* strDuplicate(const char *str, size_t n) { auto len = strlen(str); diff --git a/thirdparty/thorvg/src/common/tvgStr.h b/thirdparty/thorvg/src/common/tvgStr.h index 56fdf86fb2d5..6b16b4c7610c 100644 --- a/thirdparty/thorvg/src/common/tvgStr.h +++ b/thirdparty/thorvg/src/common/tvgStr.h @@ -29,7 +29,6 @@ namespace tvg { float strToFloat(const char *nPtr, char **endPtr); //convert to float -int str2int(const char* str, size_t n); //convert to integer char* strDuplicate(const char *str, size_t n); //copy the string char* strAppend(char* lhs, const char* rhs, size_t n); //append the rhs to the lhs char* strDirname(const char* path); //return the full directory name diff --git a/thirdparty/thorvg/src/loaders/external_png/tvgPngLoader.cpp b/thirdparty/thorvg/src/loaders/external_png/tvgPngLoader.cpp index 6c4c8410e82f..49c9f6e8aa2e 100644 --- a/thirdparty/thorvg/src/loaders/external_png/tvgPngLoader.cpp +++ b/thirdparty/thorvg/src/loaders/external_png/tvgPngLoader.cpp @@ -86,10 +86,10 @@ bool PngLoader::read() if (cs == ColorSpace::ARGB8888 || cs == ColorSpace::ARGB8888S) { image->format = PNG_FORMAT_BGRA; - surface.cs = ColorSpace::ARGB8888; + surface.cs = ColorSpace::ARGB8888S; } else { image->format = PNG_FORMAT_RGBA; - surface.cs = ColorSpace::ABGR8888; + surface.cs = ColorSpace::ABGR8888S; } auto buffer = static_cast(malloc(PNG_IMAGE_SIZE((*image)))); diff --git a/thirdparty/thorvg/src/loaders/external_webp/tvgWebpLoader.cpp b/thirdparty/thorvg/src/loaders/external_webp/tvgWebpLoader.cpp index fbaaea874319..0db7d2d233f8 100644 --- a/thirdparty/thorvg/src/loaders/external_webp/tvgWebpLoader.cpp +++ b/thirdparty/thorvg/src/loaders/external_webp/tvgWebpLoader.cpp @@ -134,7 +134,7 @@ bool WebpLoader::read() } -Surface* WebpLoader::bitmap() +RenderSurface* WebpLoader::bitmap() { this->done(); diff --git a/thirdparty/thorvg/src/loaders/external_webp/tvgWebpLoader.h b/thirdparty/thorvg/src/loaders/external_webp/tvgWebpLoader.h index ad8fb5a69954..3d44edac3a18 100644 --- a/thirdparty/thorvg/src/loaders/external_webp/tvgWebpLoader.h +++ b/thirdparty/thorvg/src/loaders/external_webp/tvgWebpLoader.h @@ -36,7 +36,7 @@ class WebpLoader : public ImageLoader, public Task bool open(const char* data, uint32_t size, bool copy) override; bool read() override; - Surface* bitmap() override; + RenderSurface* bitmap() override; private: void run(unsigned tid) override; diff --git a/thirdparty/thorvg/src/loaders/jpg/tvgJpgLoader.cpp b/thirdparty/thorvg/src/loaders/jpg/tvgJpgLoader.cpp index 86a46245d988..cb1306b71a3e 100644 --- a/thirdparty/thorvg/src/loaders/jpg/tvgJpgLoader.cpp +++ b/thirdparty/thorvg/src/loaders/jpg/tvgJpgLoader.cpp @@ -125,7 +125,7 @@ bool JpgLoader::close() } -Surface* JpgLoader::bitmap() +RenderSurface* JpgLoader::bitmap() { this->done(); return ImageLoader::bitmap(); diff --git a/thirdparty/thorvg/src/loaders/jpg/tvgJpgLoader.h b/thirdparty/thorvg/src/loaders/jpg/tvgJpgLoader.h index 05cbb54c852f..85f6d25dc392 100644 --- a/thirdparty/thorvg/src/loaders/jpg/tvgJpgLoader.h +++ b/thirdparty/thorvg/src/loaders/jpg/tvgJpgLoader.h @@ -46,7 +46,7 @@ class JpgLoader : public ImageLoader, public Task bool read() override; bool close() override; - Surface* bitmap() override; + RenderSurface* bitmap() override; }; #endif //_TVG_JPG_LOADER_H_ diff --git a/thirdparty/thorvg/src/loaders/jpg/tvgJpgd.cpp b/thirdparty/thorvg/src/loaders/jpg/tvgJpgd.cpp index 6f2bd916d5ea..92e23698af2c 100644 --- a/thirdparty/thorvg/src/loaders/jpg/tvgJpgd.cpp +++ b/thirdparty/thorvg/src/loaders/jpg/tvgJpgd.cpp @@ -36,6 +36,8 @@ #include #include #include + +#include "tvgCommon.h" #include "tvgJpgd.h" #ifdef _MSC_VER @@ -70,7 +72,7 @@ enum jpgd_status JPGD_BAD_DHT_COUNTS = -256, JPGD_BAD_DHT_INDEX, JPGD_BAD_DHT_MARKER, JPGD_BAD_DQT_MARKER, JPGD_BAD_DQT_TABLE, JPGD_BAD_PRECISION, JPGD_BAD_HEIGHT, JPGD_BAD_WIDTH, JPGD_TOO_MANY_COMPONENTS, JPGD_BAD_SOF_LENGTH, JPGD_BAD_VARIABLE_MARKER, JPGD_BAD_DRI_LENGTH, JPGD_BAD_SOS_LENGTH, - JPGD_BAD_SOS_COMP_ID, JPGD_W_EXTRA_BYTES_BEFORE_MARKER, JPGD_NO_ARITHMITIC_SUPPORT, JPGD_UNEXPECTED_MARKER, + JPGD_BAD_SOS_COMP_ID, JPGD_W_EXTRA_BYTES_BEFORE_MARKER, JPGD_NO_ARITHMETIC_SUPPORT, JPGD_UNEXPECTED_MARKER, JPGD_NOT_JPEG, JPGD_UNSUPPORTED_MARKER, JPGD_BAD_DQT_LENGTH, JPGD_TOO_MANY_BLOCKS, JPGD_UNDEFINED_QUANT_TABLE, JPGD_UNDEFINED_HUFF_TABLE, JPGD_NOT_SINGLE_SCAN, JPGD_UNSUPPORTED_COLORSPACE, JPGD_UNSUPPORTED_SAMP_FACTORS, JPGD_DECODE_ERROR, JPGD_BAD_RESTART_MARKER, JPGD_ASSERTION_ERROR, @@ -1382,9 +1384,9 @@ int jpeg_decoder::process_markers() read_dht_marker(); break; } - // No arithmitic support - dumb patents! + // No arithmetic support - dumb patents! case M_DAC: { - stop_decoding(JPGD_NO_ARITHMITIC_SUPPORT); + stop_decoding(JPGD_NO_ARITHMETIC_SUPPORT); break; } case M_DQT: { @@ -1466,8 +1468,8 @@ void jpeg_decoder::locate_sof_marker() read_sof_marker(); break; } - case M_SOF9: { /* Arithmitic coding */ - stop_decoding(JPGD_NO_ARITHMITIC_SUPPORT); + case M_SOF9: { /* Arithmetic coding */ + stop_decoding(JPGD_NO_ARITHMETIC_SUPPORT); break; } default: { @@ -1738,7 +1740,8 @@ void jpeg_decoder::transform_mcu_expand(int mcu_row) DCT_Upsample::R_S<8, 8>::calc(R, S, pSrc_ptr); break; default: - JPGD_ASSERT(false); + TVGERR("JPG", "invalid transform_mcu_expand"); + return; } DCT_Upsample::Matrix44 a(P + Q); P -= Q; DCT_Upsample::Matrix44& b = P; @@ -1831,7 +1834,7 @@ void jpeg_decoder::process_restart() int i; int c = 0; - // Align to a byte boundry + // Align to a byte boundary // FIXME: Is this really necessary? get_bits_no_markers() never reads in markers! //get_bits_no_markers(m_bits_left & 7); diff --git a/thirdparty/thorvg/src/loaders/svg/tvgSvgLoader.cpp b/thirdparty/thorvg/src/loaders/svg/tvgSvgLoader.cpp index 492cebe3a319..1ab3043c24e8 100644 --- a/thirdparty/thorvg/src/loaders/svg/tvgSvgLoader.cpp +++ b/thirdparty/thorvg/src/loaders/svg/tvgSvgLoader.cpp @@ -177,6 +177,7 @@ static float _toFloat(const SvgParser* svgParse, const char* str, SvgParserLengt else if (strstr(str, "%")) { if (type == SvgParserLengthType::Vertical) parsedValue = (parsedValue / 100.0f) * svgParse->global.h; else if (type == SvgParserLengthType::Horizontal) parsedValue = (parsedValue / 100.0f) * svgParse->global.w; + else if (type == SvgParserLengthType::Diagonal) parsedValue = (sqrtf(powf(svgParse->global.w, 2) + powf(svgParse->global.h, 2)) / sqrtf(2.0f)) * (parsedValue / 100.0f); else //if other than it's radius { float max = svgParse->global.w; @@ -593,15 +594,15 @@ static bool _hslToRgb(float hue, float saturation, float brightness, uint8_t* re saturation = saturation > 0 ? std::min(saturation, 1.0f) : 0.0f; brightness = brightness > 0 ? std::min(brightness, 1.0f) : 0.0f; - if (mathZero(saturation)) _red = _green = _blue = brightness; + if (tvg::zero(saturation)) _red = _green = _blue = brightness; else { - if (mathEqual(hue, 360.0)) hue = 0.0f; + if (tvg::equal(hue, 360.0)) hue = 0.0f; hue /= 60.0f; v = (brightness <= 0.5f) ? (brightness * (1.0f + saturation)) : (brightness + saturation - (brightness * saturation)); p = brightness + brightness - v; - if (!mathZero(v)) sv = (v - p) / v; + if (!tvg::zero(v)) sv = (v - p) / v; else sv = 0; i = static_cast(hue); @@ -858,8 +859,8 @@ static Matrix* _parseTransformationMatrix(const char* value) //Transform to signed. points[0] = fmodf(points[0], 360.0f); if (points[0] < 0) points[0] += 360.0f; - auto c = cosf(mathDeg2Rad(points[0])); - auto s = sinf(mathDeg2Rad(points[0])); + auto c = cosf(deg2rad(points[0])); + auto s = sinf(deg2rad(points[0])); if (ptCount == 1) { Matrix tmp = { c, -s, 0, s, c, 0, 0, 0, 1 }; *matrix *= tmp; @@ -882,12 +883,12 @@ static Matrix* _parseTransformationMatrix(const char* value) *matrix *= tmp; } else if (state == MatrixState::SkewX) { if (ptCount != 1) goto error; - auto deg = tanf(mathDeg2Rad(points[0])); + auto deg = tanf(deg2rad(points[0])); Matrix tmp = { 1, deg, 0, 0, 1, 0, 0, 0, 1 }; *matrix *= tmp; } else if (state == MatrixState::SkewY) { if (ptCount != 1) goto error; - auto deg = tanf(mathDeg2Rad(points[0])); + auto deg = tanf(deg2rad(points[0])); Matrix tmp = { 1, 0, 0, deg, 1, 0, 0, 0, 1 }; *matrix *= tmp; } @@ -1066,7 +1067,7 @@ static void _handleStrokeDashOffsetAttr(SvgLoaderData* loader, SvgNode* node, co static void _handleStrokeWidthAttr(SvgLoaderData* loader, SvgNode* node, const char* value) { node->style->stroke.flags = (node->style->stroke.flags | SvgStrokeFlags::Width); - node->style->stroke.width = _toFloat(loader->svgParse, value, SvgParserLengthType::Horizontal); + node->style->stroke.width = _toFloat(loader->svgParse, value, SvgParserLengthType::Diagonal); } @@ -1614,7 +1615,7 @@ static constexpr struct } circleTags[] = { {"cx", SvgParserLengthType::Horizontal, sizeof("cx"), offsetof(SvgCircleNode, cx)}, {"cy", SvgParserLengthType::Vertical, sizeof("cy"), offsetof(SvgCircleNode, cy)}, - {"r", SvgParserLengthType::Other, sizeof("r"), offsetof(SvgCircleNode, r)} + {"r", SvgParserLengthType::Diagonal, sizeof("r"), offsetof(SvgCircleNode, r)} }; @@ -2554,6 +2555,18 @@ static SvgStyleGradient* _createRadialGradient(SvgLoaderData* loader, const char } +static SvgColor* _findLatestColor(const SvgLoaderData* loader) +{ + auto parent = loader->stack.count > 0 ? loader->stack.last() : loader->doc; + + while (parent != nullptr) { + if (parent->style->curColorSet) return &parent->style->color; + parent = parent->parent; + } + return nullptr; +} + + static bool _attrParseStopsStyle(void* data, const char* key, const char* value) { SvgLoaderData* loader = (SvgLoaderData*)data; @@ -2563,7 +2576,13 @@ static bool _attrParseStopsStyle(void* data, const char* key, const char* value) stop->a = _toOpacity(value); loader->svgParse->flags = (loader->svgParse->flags | SvgStopStyleFlags::StopOpacity); } else if (!strcmp(key, "stop-color")) { - if (_toColor(value, &stop->r, &stop->g, &stop->b, nullptr)) { + if (!strcmp(value, "currentColor")) { + if (auto latestColor = _findLatestColor(loader)) { + stop->r = latestColor->r; + stop->g = latestColor->g; + stop->b = latestColor->b; + } + } else if (_toColor(value, &stop->r, &stop->g, &stop->b, nullptr)) { loader->svgParse->flags = (loader->svgParse->flags | SvgStopStyleFlags::StopColor); } } else { @@ -2586,7 +2605,13 @@ static bool _attrParseStops(void* data, const char* key, const char* value) stop->a = _toOpacity(value); } } else if (!strcmp(key, "stop-color")) { - if (!(loader->svgParse->flags & SvgStopStyleFlags::StopColor)) { + if (!strcmp(value, "currentColor")) { + if (auto latestColor = _findLatestColor(loader)) { + stop->r = latestColor->r; + stop->g = latestColor->g; + stop->b = latestColor->b; + } + } else if (!(loader->svgParse->flags & SvgStopStyleFlags::StopColor)) { _toColor(value, &stop->r, &stop->g, &stop->b, nullptr); } } else if (!strcmp(key, "style")) { @@ -3341,7 +3366,7 @@ static void _svgLoaderParserXmlOpen(SvgLoaderData* loader, const char* content, else parent = loader->doc; if (!strcmp(tagName, "style")) { // TODO: For now only the first style node is saved. After the css id selector - // is introduced this if condition shouldin't be necessary any more + // is introduced this if condition shouldn't be necessary any more if (!loader->cssStyle) { node = method(loader, nullptr, attrs, attrsLength, simpleXmlParseAttributes); loader->cssStyle = node; diff --git a/thirdparty/thorvg/src/loaders/svg/tvgSvgLoaderCommon.h b/thirdparty/thorvg/src/loaders/svg/tvgSvgLoaderCommon.h index 5661c8ae82ce..e64d7afb4113 100644 --- a/thirdparty/thorvg/src/loaders/svg/tvgSvgLoaderCommon.h +++ b/thirdparty/thorvg/src/loaders/svg/tvgSvgLoaderCommon.h @@ -215,6 +215,7 @@ enum class SvgParserLengthType { Vertical, Horizontal, + Diagonal, //In case of, for example, radius of radial gradient Other }; diff --git a/thirdparty/thorvg/src/loaders/svg/tvgSvgPath.cpp b/thirdparty/thorvg/src/loaders/svg/tvgSvgPath.cpp index 115e81aee122..57442139cdff 100644 --- a/thirdparty/thorvg/src/loaders/svg/tvgSvgPath.cpp +++ b/thirdparty/thorvg/src/loaders/svg/tvgSvgPath.cpp @@ -126,7 +126,7 @@ void _pathAppendArcTo(Array* cmds, Array* pts, Point* cur, P rx = fabsf(rx); ry = fabsf(ry); - angle = mathDeg2Rad(angle); + angle = deg2rad(angle); cosPhi = cosf(angle); sinPhi = sinf(angle); dx2 = (sx - x) / 2.0f; @@ -190,14 +190,14 @@ void _pathAppendArcTo(Array* cmds, Array* pts, Point* cur, P cx += (sx + x) / 2.0f; cy += (sy + y) / 2.0f; - //Sstep 4 (F6.5.4) + //Step 4 (F6.5.4) //We dont' use arccos (as per w3c doc), see //http://www.euclideanspace.com/maths/algebra/vectors/angleBetween/index.htm //Note: atan2 (0.0, 1.0) == 0.0 - at = mathAtan2(((y1p - cyp) / ry), ((x1p - cxp) / rx)); + at = tvg::atan2(((y1p - cyp) / ry), ((x1p - cxp) / rx)); theta1 = (at < 0.0f) ? 2.0f * MATH_PI + at : at; - nat = mathAtan2(((-y1p - cyp) / ry), ((-x1p - cxp) / rx)); + nat = tvg::atan2(((-y1p - cyp) / ry), ((-x1p - cxp) / rx)); deltaTheta = (nat < at) ? 2.0f * MATH_PI - at + nat : nat - at; if (sweep) { @@ -469,12 +469,12 @@ static bool _processCommand(Array* cmds, Array* pts, char cm } case 'a': case 'A': { - if (mathZero(arr[0]) || mathZero(arr[1])) { + if (tvg::zero(arr[0]) || tvg::zero(arr[1])) { Point p = {arr[5], arr[6]}; cmds->push(PathCommand::LineTo); pts->push(p); *cur = {arr[5], arr[6]}; - } else if (!mathEqual(cur->x, arr[5]) || !mathEqual(cur->y, arr[6])) { + } else if (!tvg::equal(cur->x, arr[5]) || !tvg::equal(cur->y, arr[6])) { _pathAppendArcTo(cmds, pts, cur, curCtl, arr[5], arr[6], fabsf(arr[0]), fabsf(arr[1]), arr[2], arr[3], arr[4]); *cur = *curCtl = {arr[5], arr[6]}; *isQuadratic = false; diff --git a/thirdparty/thorvg/src/loaders/svg/tvgSvgSceneBuilder.cpp b/thirdparty/thorvg/src/loaders/svg/tvgSvgSceneBuilder.cpp index b048695a234c..00c7408a7e88 100644 --- a/thirdparty/thorvg/src/loaders/svg/tvgSvgSceneBuilder.cpp +++ b/thirdparty/thorvg/src/loaders/svg/tvgSvgSceneBuilder.cpp @@ -153,7 +153,7 @@ static unique_ptr _applyRadialGradientProperty(SvgStyleGradient* if (isTransform) finalTransform = *g->transform; if (g->userSpace) { - //The radius scalling is done according to the Units section: + //The radius scaling is done according to the Units section: //https://www.w3.org/TR/2015/WD-SVG2-20150915/coords.html g->radial->cx = g->radial->cx * vBox.w; g->radial->cy = g->radial->cy * vBox.h; @@ -215,7 +215,7 @@ static bool _appendClipUseNode(SvgLoaderData& loaderData, SvgNode* node, Shape* } if (child->transform) finalTransform = *child->transform * finalTransform; - return _appendClipShape(loaderData, child, shape, vBox, svgPath, mathIdentity((const Matrix*)(&finalTransform)) ? nullptr : &finalTransform); + return _appendClipShape(loaderData, child, shape, vBox, svgPath, identity((const Matrix*)(&finalTransform)) ? nullptr : &finalTransform); } @@ -272,8 +272,7 @@ static void _applyComposition(SvgLoaderData& loaderData, Paint* paint, const Svg if (valid) { Matrix finalTransform = _compositionTransform(paint, node, compNode, SvgNodeType::ClipPath); comp->transform(finalTransform); - - paint->composite(std::move(comp), CompositeMethod::ClipPath); + paint->clip(std::move(comp)); } node->style->clipPath.applying = false; @@ -714,7 +713,6 @@ static Matrix _calculateAspectRatioMatrix(AspectRatioAlign align, AspectRatioMee static unique_ptr _useBuildHelper(SvgLoaderData& loaderData, const SvgNode* node, const Box& vBox, const string& svgPath, int depth, bool* isMaskWhite) { - unique_ptr finalScene; auto scene = _sceneBuildHelper(loaderData, node, vBox, svgPath, false, depth + 1, isMaskWhite); // mUseTransform = mUseTransform * mTranslate @@ -736,10 +734,10 @@ static unique_ptr _useBuildHelper(SvgLoaderData& loaderData, const SvgNod auto vh = (symbol.hasViewBox ? symbol.vh : height); Matrix mViewBox = {1, 0, 0, 0, 1, 0, 0, 0, 1}; - if ((!mathEqual(width, vw) || !mathEqual(height, vh)) && vw > 0 && vh > 0) { + if ((!tvg::equal(width, vw) || !tvg::equal(height, vh)) && vw > 0 && vh > 0) { Box box = {symbol.vx, symbol.vy, vw, vh}; mViewBox = _calculateAspectRatioMatrix(symbol.align, symbol.meetOrSlice, width, height, box); - } else if (!mathZero(symbol.vx) || !mathZero(symbol.vy)) { + } else if (!tvg::zero(symbol.vx) || !tvg::zero(symbol.vy)) { mViewBox = {1, 0, -symbol.vx, 0, 1, -symbol.vy, 0, 0, 1}; } @@ -751,9 +749,7 @@ static unique_ptr _useBuildHelper(SvgLoaderData& loaderData, const SvgNod mSceneTransform = mUseTransform * mSceneTransform; scene->transform(mSceneTransform); - if (node->node.use.symbol->node.symbol.overflowVisible) { - finalScene = std::move(scene); - } else { + if (!node->node.use.symbol->node.symbol.overflowVisible) { auto viewBoxClip = Shape::gen(); viewBoxClip->appendRect(0, 0, width, height, 0, 0); @@ -764,21 +760,13 @@ static unique_ptr _useBuildHelper(SvgLoaderData& loaderData, const SvgNod } viewBoxClip->transform(mClipTransform); - auto compositeLayer = Scene::gen(); - compositeLayer->composite(std::move(viewBoxClip), CompositeMethod::ClipPath); - compositeLayer->push(std::move(scene)); - - auto root = Scene::gen(); - root->push(std::move(compositeLayer)); - - finalScene = std::move(root); + scene->clip(std::move(viewBoxClip)); } } else { scene->transform(mUseTransform); - finalScene = std::move(scene); } - return finalScene; + return scene; } @@ -821,7 +809,7 @@ static unique_ptr _textBuildHelper(SvgLoaderData& loaderData, const SvgNod Matrix textTransform = {1, 0, 0, 0, 1, 0, 0, 0, 1}; if (node->transform) textTransform = *node->transform; - mathTranslateR(&textTransform, node->node.text.x, node->node.text.y - textNode->fontSize); + translateR(&textTransform, node->node.text.x, node->node.text.y - textNode->fontSize); text->transform(textTransform); //TODO: handle def values of font and size as used in a system? @@ -926,10 +914,10 @@ Scene* svgSceneBuild(SvgLoaderData& loaderData, Box vBox, float w, float h, Aspe if (!(viewFlag & SvgViewFlag::Viewbox)) _updateInvalidViewSize(docNode.get(), vBox, w, h, viewFlag); - if (!mathEqual(w, vBox.w) || !mathEqual(h, vBox.h)) { + if (!tvg::equal(w, vBox.w) || !tvg::equal(h, vBox.h)) { Matrix m = _calculateAspectRatioMatrix(align, meetOrSlice, w, h, vBox); docNode->transform(m); - } else if (!mathZero(vBox.x) || !mathZero(vBox.y)) { + } else if (!tvg::zero(vBox.x) || !tvg::zero(vBox.y)) { docNode->translate(-vBox.x, -vBox.y); } @@ -937,7 +925,7 @@ Scene* svgSceneBuild(SvgLoaderData& loaderData, Box vBox, float w, float h, Aspe viewBoxClip->appendRect(0, 0, w, h); auto compositeLayer = Scene::gen(); - compositeLayer->composite(std::move(viewBoxClip), CompositeMethod::ClipPath); + compositeLayer->clip(std::move(viewBoxClip)); compositeLayer->push(std::move(docNode)); auto root = Scene::gen(); diff --git a/thirdparty/thorvg/src/loaders/svg/tvgXmlParser.cpp b/thirdparty/thorvg/src/loaders/svg/tvgXmlParser.cpp index 09fc8aaac137..da1cdae9e0e7 100644 --- a/thirdparty/thorvg/src/loaders/svg/tvgXmlParser.cpp +++ b/thirdparty/thorvg/src/loaders/svg/tvgXmlParser.cpp @@ -492,13 +492,13 @@ bool simpleXmlParseW3CAttribute(const char* buf, unsigned bufLength, simpleXMLAt key[0] = '\0'; val[0] = '\0'; - if (next == nullptr && sep != nullptr) { + if (sep != nullptr && next == nullptr) { memcpy(key, buf, sep - buf); key[sep - buf] = '\0'; memcpy(val, sep + 1, end - sep - 1); val[end - sep - 1] = '\0'; - } else if (sep < next && sep != nullptr) { + } else if (sep != nullptr && sep < next) { memcpy(key, buf, sep - buf); key[sep - buf] = '\0'; @@ -522,8 +522,9 @@ bool simpleXmlParseW3CAttribute(const char* buf, unsigned bufLength, simpleXMLAt } } + if (!next) break; buf = next + 1; - } while (next != nullptr); + } while (true); return true; } diff --git a/thirdparty/thorvg/src/renderer/sw_engine/tvgSwCommon.h b/thirdparty/thorvg/src/renderer/sw_engine/tvgSwCommon.h index ab1fc18b58a5..9371ae6c5ad3 100644 --- a/thirdparty/thorvg/src/renderer/sw_engine/tvgSwCommon.h +++ b/thirdparty/thorvg/src/renderer/sw_engine/tvgSwCommon.h @@ -113,7 +113,7 @@ struct SwSpan uint8_t coverage; }; -struct SwRleData +struct SwRle { SwSpan *spans; uint32_t alloc; @@ -211,8 +211,8 @@ struct SwShape SwOutline* outline = nullptr; SwStroke* stroke = nullptr; SwFill* fill = nullptr; - SwRleData* rle = nullptr; - SwRleData* strokeRle = nullptr; + SwRle* rle = nullptr; + SwRle* strokeRle = nullptr; SwBBox bbox; //Keep it boundary without stroke region. Using for optimal filling. bool fastTrack = false; //Fast Track: axis-aligned rectangle without any clips? @@ -221,7 +221,7 @@ struct SwShape struct SwImage { SwOutline* outline = nullptr; - SwRleData* rle = nullptr; + SwRle* rle = nullptr; union { pixel_t* data; //system based data pointer uint32_t* buf32; //for explicit 32bits channels @@ -244,13 +244,13 @@ typedef uint8_t(*SwAlpha)(uint8_t*); //bl struct SwCompositor; -struct SwSurface : Surface +struct SwSurface : RenderSurface { SwJoin join; SwAlpha alphas[4]; //Alpha:2, InvAlpha:3, Luma:4, InvLuma:5 SwBlender blender = nullptr; //blender (optional) SwCompositor* compositor = nullptr; //compositor (optional) - BlendMethod blendMethod; //blending method (uint8_t) + BlendMethod blendMethod = BlendMethod::Normal; SwAlpha alpha(CompositeMethod method) { @@ -262,7 +262,7 @@ struct SwSurface : Surface { } - SwSurface(const SwSurface* rhs) : Surface(rhs) + SwSurface(const SwSurface* rhs) : RenderSurface(rhs) { join = rhs->join; memcpy(alphas, rhs->alphas, sizeof(alphas)); @@ -272,7 +272,7 @@ struct SwSurface : Surface } }; -struct SwCompositor : Compositor +struct SwCompositor : RenderCompositor { SwSurface* recoverSfc; //Recover surface when composition is started SwCompositor* recoverCmp; //Recover compositor when composition is done @@ -488,13 +488,14 @@ SwFixed mathAtan(const SwPoint& pt); SwFixed mathCos(SwFixed angle); SwFixed mathSin(SwFixed angle); void mathSplitCubic(SwPoint* base); +void mathSplitLine(SwPoint* base); SwFixed mathDiff(SwFixed angle1, SwFixed angle2); SwFixed mathLength(const SwPoint& pt); -bool mathSmallCubic(const SwPoint* base, SwFixed& angleIn, SwFixed& angleMid, SwFixed& angleOut); +int mathCubicAngle(const SwPoint* base, SwFixed& angleIn, SwFixed& angleMid, SwFixed& angleOut); SwFixed mathMean(SwFixed angle1, SwFixed angle2); SwPoint mathTransform(const Point* to, const Matrix& transform); bool mathUpdateOutlineBBox(const SwOutline* outline, const SwBBox& clipRegion, SwBBox& renderRegion, bool fastTrack); -bool mathClipBBox(const SwBBox& clipper, SwBBox& clipee); +bool mathClipBBox(const SwBBox& clipper, SwBBox& clippee); void shapeReset(SwShape* shape); bool shapePrepare(SwShape* shape, const RenderShape* rshape, const Matrix& transform, const SwBBox& clipRegion, SwBBox& renderRegion, SwMpool* mpool, unsigned tid, bool hasComposite); @@ -541,13 +542,13 @@ void fillRadial(const SwFill* fill, uint32_t* dst, uint32_t y, uint32_t x, uint3 void fillRadial(const SwFill* fill, uint32_t* dst, uint32_t y, uint32_t x, uint32_t len, SwBlender op, SwBlender op2, uint8_t a); //blending + BlendingMethod(op2) ver. void fillRadial(const SwFill* fill, uint32_t* dst, uint32_t y, uint32_t x, uint32_t len, uint8_t* cmp, SwAlpha alpha, uint8_t csize, uint8_t opacity); //matting ver. -SwRleData* rleRender(SwRleData* rle, const SwOutline* outline, const SwBBox& renderRegion, bool antiAlias); -SwRleData* rleRender(const SwBBox* bbox); -void rleFree(SwRleData* rle); -void rleReset(SwRleData* rle); -void rleMerge(SwRleData* rle, SwRleData* clip1, SwRleData* clip2); -void rleClipPath(SwRleData* rle, const SwRleData* clip); -void rleClipRect(SwRleData* rle, const SwBBox* clip); +SwRle* rleRender(SwRle* rle, const SwOutline* outline, const SwBBox& renderRegion, bool antiAlias); +SwRle* rleRender(const SwBBox* bbox); +void rleFree(SwRle* rle); +void rleReset(SwRle* rle); +void rleMerge(SwRle* rle, SwRle* clip1, SwRle* clip2); +void rleClip(SwRle* rle, const SwRle* clip); +void rleClip(SwRle* rle, const SwBBox* clip); SwMpool* mpoolInit(uint32_t threads); bool mpoolTerm(SwMpool* mpool); @@ -567,9 +568,17 @@ bool rasterStroke(SwSurface* surface, SwShape* shape, uint8_t r, uint8_t g, uint bool rasterGradientStroke(SwSurface* surface, SwShape* shape, const Fill* fdata, uint8_t opacity); bool rasterClear(SwSurface* surface, uint32_t x, uint32_t y, uint32_t w, uint32_t h, pixel_t val = 0); void rasterPixel32(uint32_t *dst, uint32_t val, uint32_t offset, int32_t len); +void rasterTranslucentPixel32(uint32_t* dst, uint32_t* src, uint32_t len, uint8_t opacity); +void rasterPixel32(uint32_t* dst, uint32_t* src, uint32_t len, uint8_t opacity); void rasterGrayscale8(uint8_t *dst, uint8_t val, uint32_t offset, int32_t len); -void rasterUnpremultiply(Surface* surface); -void rasterPremultiply(Surface* surface); -bool rasterConvertCS(Surface* surface, ColorSpace to); +void rasterXYFlip(uint32_t* src, uint32_t* dst, int32_t stride, int32_t w, int32_t h, const SwBBox& bbox, bool flipped); +void rasterUnpremultiply(RenderSurface* surface); +void rasterPremultiply(RenderSurface* surface); +bool rasterConvertCS(RenderSurface* surface, ColorSpace to); + +bool effectGaussianBlur(SwCompositor* cmp, SwSurface* surface, const RenderEffectGaussianBlur* params); +bool effectGaussianBlurPrepare(RenderEffectGaussianBlur* effect); +bool effectDropShadow(SwCompositor* cmp, SwSurface* surfaces[2], const RenderEffectDropShadow* params, uint8_t opacity, bool direct); +bool effectDropShadowPrepare(RenderEffectDropShadow* effect); #endif /* _TVG_SW_COMMON_H_ */ diff --git a/thirdparty/thorvg/src/renderer/sw_engine/tvgSwFill.cpp b/thirdparty/thorvg/src/renderer/sw_engine/tvgSwFill.cpp index 631294ad40be..f70ba7a13d61 100644 --- a/thirdparty/thorvg/src/renderer/sw_engine/tvgSwFill.cpp +++ b/thirdparty/thorvg/src/renderer/sw_engine/tvgSwFill.cpp @@ -66,15 +66,15 @@ static void _calculateCoefficients(const SwFill* fill, uint32_t x, uint32_t y, f static uint32_t _estimateAAMargin(const Fill* fdata) { constexpr float marginScalingFactor = 800.0f; - if (fdata->identifier() == TVG_CLASS_ID_RADIAL) { + if (fdata->type() == Type::RadialGradient) { auto radius = P(static_cast(fdata))->r; - return mathZero(radius) ? 0 : static_cast(marginScalingFactor / radius); + return tvg::zero(radius) ? 0 : static_cast(marginScalingFactor / radius); } auto grad = P(static_cast(fdata)); Point p1 {grad->x1, grad->y1}; Point p2 {grad->x2, grad->y2}; - auto length = mathLength(&p1, &p2); - return mathZero(length) ? 0 : static_cast(marginScalingFactor / length); + auto len = length(&p1, &p2); + return tvg::zero(len) ? 0 : static_cast(marginScalingFactor / len); } @@ -217,7 +217,7 @@ bool _prepareLinear(SwFill* fill, const LinearGradient* linear, const Matrix& tr auto len = fill->linear.dx * fill->linear.dx + fill->linear.dy * fill->linear.dy; if (len < FLOAT_EPSILON) { - if (mathZero(fill->linear.dx) && mathZero(fill->linear.dy)) { + if (tvg::zero(fill->linear.dx) && tvg::zero(fill->linear.dy)) { fill->solid = true; } return true; @@ -228,7 +228,7 @@ bool _prepareLinear(SwFill* fill, const LinearGradient* linear, const Matrix& tr fill->linear.offset = -fill->linear.dx * x1 - fill->linear.dy * y1; auto gradTransform = linear->transform(); - bool isTransformation = !mathIdentity((const Matrix*)(&gradTransform)); + bool isTransformation = !identity((const Matrix*)(&gradTransform)); if (isTransformation) { gradTransform = transform * gradTransform; @@ -239,7 +239,7 @@ bool _prepareLinear(SwFill* fill, const LinearGradient* linear, const Matrix& tr if (isTransformation) { Matrix invTransform; - if (!mathInverse(&gradTransform, &invTransform)) return false; + if (!inverse(&gradTransform, &invTransform)) return false; fill->linear.offset += fill->linear.dx * invTransform.e13 + fill->linear.dy * invTransform.e23; @@ -261,7 +261,7 @@ bool _prepareRadial(SwFill* fill, const RadialGradient* radial, const Matrix& tr auto fy = P(radial)->fy; auto fr = P(radial)->fr; - if (mathZero(r)) { + if (tvg::zero(r)) { fill->solid = true; return true; } @@ -295,7 +295,7 @@ bool _prepareRadial(SwFill* fill, const RadialGradient* radial, const Matrix& tr if (fill->radial.a > 0) fill->radial.invA = 1.0f / fill->radial.a; auto gradTransform = radial->transform(); - bool isTransformation = !mathIdentity((const Matrix*)(&gradTransform)); + bool isTransformation = !identity((const Matrix*)(&gradTransform)); if (isTransformation) gradTransform = transform * gradTransform; else { @@ -305,7 +305,7 @@ bool _prepareRadial(SwFill* fill, const RadialGradient* radial, const Matrix& tr if (isTransformation) { Matrix invTransform; - if (!mathInverse(&gradTransform, &invTransform)) return false; + if (!inverse(&gradTransform, &invTransform)) return false; fill->radial.a11 = invTransform.e11; fill->radial.a12 = invTransform.e12; fill->radial.a13 = invTransform.e13; @@ -553,7 +553,7 @@ void fillLinear(const SwFill* fill, uint32_t* dst, uint32_t y, uint32_t x, uint3 float inc = (fill->linear.dx) * (GRADIENT_STOP_SIZE - 1); if (opacity == 255) { - if (mathZero(inc)) { + if (tvg::zero(inc)) { auto color = _fixedPixel(fill, static_cast(t * FIXPT_SIZE)); for (uint32_t i = 0; i < len; ++i, ++dst, cmp += csize) { *dst = opBlendNormal(color, *dst, alpha(cmp)); @@ -584,7 +584,7 @@ void fillLinear(const SwFill* fill, uint32_t* dst, uint32_t y, uint32_t x, uint3 } } } else { - if (mathZero(inc)) { + if (tvg::zero(inc)) { auto color = _fixedPixel(fill, static_cast(t * FIXPT_SIZE)); for (uint32_t i = 0; i < len; ++i, ++dst, cmp += csize) { *dst = opBlendNormal(color, *dst, MULTIPLY(alpha(cmp), opacity)); @@ -626,7 +626,7 @@ void fillLinear(const SwFill* fill, uint8_t* dst, uint32_t y, uint32_t x, uint32 float t = (fill->linear.dx * rx + fill->linear.dy * ry + fill->linear.offset) * (GRADIENT_STOP_SIZE - 1); float inc = (fill->linear.dx) * (GRADIENT_STOP_SIZE - 1); - if (mathZero(inc)) { + if (tvg::zero(inc)) { auto src = MULTIPLY(a, A(_fixedPixel(fill, static_cast(t * FIXPT_SIZE)))); for (uint32_t i = 0; i < len; ++i, ++dst) { *dst = maskOp(src, *dst, ~src); @@ -643,7 +643,7 @@ void fillLinear(const SwFill* fill, uint8_t* dst, uint32_t y, uint32_t x, uint32 auto t2 = static_cast(t * FIXPT_SIZE); auto inc2 = static_cast(inc * FIXPT_SIZE); for (uint32_t j = 0; j < len; ++j, ++dst) { - auto src = MULTIPLY(_fixedPixel(fill, t2), a); + auto src = MULTIPLY(A(_fixedPixel(fill, t2)), a); *dst = maskOp(src, *dst, ~src); t2 += inc2; } @@ -651,7 +651,7 @@ void fillLinear(const SwFill* fill, uint8_t* dst, uint32_t y, uint32_t x, uint32 } else { uint32_t counter = 0; while (counter++ < len) { - auto src = MULTIPLY(_pixel(fill, t / GRADIENT_STOP_SIZE), a); + auto src = MULTIPLY(A(_pixel(fill, t / GRADIENT_STOP_SIZE)), a); *dst = maskOp(src, *dst, ~src); ++dst; t += inc; @@ -668,7 +668,7 @@ void fillLinear(const SwFill* fill, uint8_t* dst, uint32_t y, uint32_t x, uint32 float t = (fill->linear.dx * rx + fill->linear.dy * ry + fill->linear.offset) * (GRADIENT_STOP_SIZE - 1); float inc = (fill->linear.dx) * (GRADIENT_STOP_SIZE - 1); - if (mathZero(inc)) { + if (tvg::zero(inc)) { auto src = A(_fixedPixel(fill, static_cast(t * FIXPT_SIZE))); src = MULTIPLY(src, a); for (uint32_t i = 0; i < len; ++i, ++dst, ++cmp) { @@ -715,7 +715,7 @@ void fillLinear(const SwFill* fill, uint32_t* dst, uint32_t y, uint32_t x, uint3 float t = (fill->linear.dx * rx + fill->linear.dy * ry + fill->linear.offset) * (GRADIENT_STOP_SIZE - 1); float inc = (fill->linear.dx) * (GRADIENT_STOP_SIZE - 1); - if (mathZero(inc)) { + if (tvg::zero(inc)) { auto color = _fixedPixel(fill, static_cast(t * FIXPT_SIZE)); for (uint32_t i = 0; i < len; ++i, ++dst) { *dst = op(color, *dst, a); @@ -755,7 +755,7 @@ void fillLinear(const SwFill* fill, uint32_t* dst, uint32_t y, uint32_t x, uint3 float t = (fill->linear.dx * rx + fill->linear.dy * ry + fill->linear.offset) * (GRADIENT_STOP_SIZE - 1); float inc = (fill->linear.dx) * (GRADIENT_STOP_SIZE - 1); - if (mathZero(inc)) { + if (tvg::zero(inc)) { auto color = _fixedPixel(fill, static_cast(t * FIXPT_SIZE)); if (a == 255) { for (uint32_t i = 0; i < len; ++i, ++dst) { @@ -828,9 +828,9 @@ bool fillGenColorTable(SwFill* fill, const Fill* fdata, const Matrix& transform, fill->spread = fdata->spread(); - if (fdata->identifier() == TVG_CLASS_ID_LINEAR) { + if (fdata->type() == Type::LinearGradient) { if (!_prepareLinear(fill, static_cast(fdata), transform)) return false; - } else if (fdata->identifier() == TVG_CLASS_ID_RADIAL) { + } else if (fdata->type() == Type::RadialGradient) { if (!_prepareRadial(fill, static_cast(fdata), transform)) return false; } diff --git a/thirdparty/thorvg/src/renderer/sw_engine/tvgSwImage.cpp b/thirdparty/thorvg/src/renderer/sw_engine/tvgSwImage.cpp index 3fc64ce036a9..d2c02bb932df 100644 --- a/thirdparty/thorvg/src/renderer/sw_engine/tvgSwImage.cpp +++ b/thirdparty/thorvg/src/renderer/sw_engine/tvgSwImage.cpp @@ -29,7 +29,7 @@ static inline bool _onlyShifted(const Matrix& m) { - if (mathEqual(m.e11, 1.0f) && mathEqual(m.e22, 1.0f) && mathZero(m.e12) && mathZero(m.e21)) return true; + if (tvg::equal(m.e11, 1.0f) && tvg::equal(m.e22, 1.0f) && tvg::zero(m.e12) && tvg::zero(m.e21)) return true; return false; } @@ -86,7 +86,7 @@ bool imagePrepare(SwImage* image, const Matrix& transform, const SwBBox& clipReg auto scaleY = sqrtf((transform.e22 * transform.e22) + (transform.e12 * transform.e12)); image->scale = (fabsf(scaleX - scaleY) > 0.01f) ? 1.0f : scaleX; - if (mathZero(transform.e12) && mathZero(transform.e21)) image->scaled = true; + if (tvg::zero(transform.e12) && tvg::zero(transform.e21)) image->scaled = true; else image->scaled = false; } diff --git a/thirdparty/thorvg/src/renderer/sw_engine/tvgSwMath.cpp b/thirdparty/thorvg/src/renderer/sw_engine/tvgSwMath.cpp index fb809c4f7e91..1ff99f6aece9 100644 --- a/thirdparty/thorvg/src/renderer/sw_engine/tvgSwMath.cpp +++ b/thirdparty/thorvg/src/renderer/sw_engine/tvgSwMath.cpp @@ -44,7 +44,7 @@ SwFixed mathMean(SwFixed angle1, SwFixed angle2) } -bool mathSmallCubic(const SwPoint* base, SwFixed& angleIn, SwFixed& angleMid, SwFixed& angleOut) +int mathCubicAngle(const SwPoint* base, SwFixed& angleIn, SwFixed& angleMid, SwFixed& angleOut) { auto d1 = base[2] - base[3]; auto d2 = base[1] - base[2]; @@ -54,7 +54,7 @@ bool mathSmallCubic(const SwPoint* base, SwFixed& angleIn, SwFixed& angleMid, Sw if (d2.small()) { if (d3.small()) { angleIn = angleMid = angleOut = 0; - return true; + return -1; //ignoreable } else { angleIn = angleMid = angleOut = mathAtan(d3); } @@ -90,8 +90,8 @@ bool mathSmallCubic(const SwPoint* base, SwFixed& angleIn, SwFixed& angleMid, Sw auto theta1 = abs(mathDiff(angleIn, angleMid)); auto theta2 = abs(mathDiff(angleMid, angleOut)); - if ((theta1 < (SW_ANGLE_PI / 8)) && (theta2 < (SW_ANGLE_PI / 8))) return true; - return false; + if ((theta1 < (SW_ANGLE_PI / 8)) && (theta2 < (SW_ANGLE_PI / 8))) return 0; //small size + return 1; } @@ -179,7 +179,7 @@ SwFixed mathTan(SwFixed angle) SwFixed mathAtan(const SwPoint& pt) { if (pt.zero()) return 0; - return SwFixed(mathAtan2(TO_FLOAT(pt.y), TO_FLOAT(pt.x)) * (180.0f / MATH_PI) * 65536.0f); + return SwFixed(tvg::atan2(TO_FLOAT(pt.y), TO_FLOAT(pt.x)) * (180.0f / MATH_PI) * 65536.0f); } @@ -242,6 +242,15 @@ void mathSplitCubic(SwPoint* base) } +void mathSplitLine(SwPoint* base) +{ + base[2] = base[1]; + + base[1].x = (base[0].x + base[1].x) >> 1; + base[1].y = (base[0].y + base[1].y) >> 1; +} + + SwFixed mathDiff(SwFixed angle1, SwFixed angle2) { auto delta = angle2 - angle1; @@ -263,19 +272,19 @@ SwPoint mathTransform(const Point* to, const Matrix& transform) } -bool mathClipBBox(const SwBBox& clipper, SwBBox& clipee) +bool mathClipBBox(const SwBBox& clipper, SwBBox& clippee) { - clipee.max.x = (clipee.max.x < clipper.max.x) ? clipee.max.x : clipper.max.x; - clipee.max.y = (clipee.max.y < clipper.max.y) ? clipee.max.y : clipper.max.y; - clipee.min.x = (clipee.min.x > clipper.min.x) ? clipee.min.x : clipper.min.x; - clipee.min.y = (clipee.min.y > clipper.min.y) ? clipee.min.y : clipper.min.y; + clippee.max.x = (clippee.max.x < clipper.max.x) ? clippee.max.x : clipper.max.x; + clippee.max.y = (clippee.max.y < clipper.max.y) ? clippee.max.y : clipper.max.y; + clippee.min.x = (clippee.min.x > clipper.min.x) ? clippee.min.x : clipper.min.x; + clippee.min.y = (clippee.min.y > clipper.min.y) ? clippee.min.y : clipper.min.y; //Check valid region - if (clipee.max.x - clipee.min.x < 1 && clipee.max.y - clipee.min.y < 1) return false; + if (clippee.max.x - clippee.min.x < 1 && clippee.max.y - clippee.min.y < 1) return false; //Check boundary - if (clipee.min.x >= clipper.max.x || clipee.min.y >= clipper.max.y || - clipee.max.x <= clipper.min.x || clipee.max.y <= clipper.min.y) return false; + if (clippee.min.x >= clipper.max.x || clippee.min.y >= clipper.max.y || + clippee.max.x <= clipper.min.x || clippee.max.y <= clipper.min.y) return false; return true; } @@ -303,9 +312,7 @@ bool mathUpdateOutlineBBox(const SwOutline* outline, const SwBBox& clipRegion, S if (yMin > pt->y) yMin = pt->y; if (yMax < pt->y) yMax = pt->y; } - //Since no antialiasing is applied in the Fast Track case, - //the rasterization region has to be rearranged. - //https://github.com/Samsung/thorvg/issues/916 + if (fastTrack) { renderRegion.min.x = static_cast(nearbyint(xMin / 64.0f)); renderRegion.max.x = static_cast(nearbyint(xMax / 64.0f)); diff --git a/thirdparty/thorvg/src/renderer/sw_engine/tvgSwPostEffect.cpp b/thirdparty/thorvg/src/renderer/sw_engine/tvgSwPostEffect.cpp new file mode 100644 index 000000000000..fd8e532e12a3 --- /dev/null +++ b/thirdparty/thorvg/src/renderer/sw_engine/tvgSwPostEffect.cpp @@ -0,0 +1,410 @@ +/* + * Copyright (c) 2024 the ThorVG project. All rights reserved. + + * 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 "tvgMath.h" +#include "tvgSwCommon.h" + +/************************************************************************/ +/* Gaussian Blur Implementation */ +/************************************************************************/ + +struct SwGaussianBlur +{ + static constexpr int MAX_LEVEL = 3; + int level; + int kernel[MAX_LEVEL]; +}; + + +static void _gaussianExtendRegion(RenderRegion& region, int extra, int8_t direction) +{ + //bbox region expansion for feathering + if (direction != 2) { + region.x = -extra; + region.w = extra * 2; + } + if (direction != 1) { + region.y = -extra; + region.h = extra * 2; + } +} + + +static int _gaussianEdgeWrap(int end, int idx) +{ + auto r = idx % end; + return (r < 0) ? end + r : r; +} + + +static int _gaussianEdgeExtend(int end, int idx) +{ + if (idx < 0) return 0; + else if (idx >= end) return end - 1; + return idx; +} + + +static int _gaussianRemap(int end, int idx, int border) +{ + if (border == 1) return _gaussianEdgeWrap(end, idx); + return _gaussianEdgeExtend(end, idx); +} + + +//TODO: SIMD OPTIMIZATION? +static void _gaussianFilter(uint8_t* dst, uint8_t* src, int32_t stride, int32_t w, int32_t h, const SwBBox& bbox, int32_t dimension, int border, bool flipped) +{ + if (flipped) { + src += (bbox.min.x * stride + bbox.min.y) << 2; + dst += (bbox.min.x * stride + bbox.min.y) << 2; + } else { + src += (bbox.min.y * stride + bbox.min.x) << 2; + dst += (bbox.min.y * stride + bbox.min.x) << 2; + } + + auto iarr = 1.0f / (dimension + dimension + 1); + + #pragma omp parallel for + for (int y = 0; y < h; ++y) { + auto p = y * stride; + auto i = p * 4; //current index + auto l = -(dimension + 1); //left index + auto r = dimension; //right index + int acc[4] = {0, 0, 0, 0}; //sliding accumulator + + //initial accumulation + for (int x = l; x < r; ++x) { + auto id = (_gaussianRemap(w, x, border) + p) * 4; + acc[0] += src[id++]; + acc[1] += src[id++]; + acc[2] += src[id++]; + acc[3] += src[id]; + } + //perform filtering + for (int x = 0; x < w; ++x, ++r, ++l) { + auto rid = (_gaussianRemap(w, r, border) + p) * 4; + auto lid = (_gaussianRemap(w, l, border) + p) * 4; + acc[0] += src[rid++] - src[lid++]; + acc[1] += src[rid++] - src[lid++]; + acc[2] += src[rid++] - src[lid++]; + acc[3] += src[rid] - src[lid]; + dst[i++] = static_cast(acc[0] * iarr + 0.5f); + dst[i++] = static_cast(acc[1] * iarr + 0.5f); + dst[i++] = static_cast(acc[2] * iarr + 0.5f); + dst[i++] = static_cast(acc[3] * iarr + 0.5f); + } + } +} + + +static int _gaussianInit(SwGaussianBlur* data, float sigma, int quality) +{ + const auto MAX_LEVEL = SwGaussianBlur::MAX_LEVEL; + + if (tvg::zero(sigma)) return 0; + + data->level = int(SwGaussianBlur::MAX_LEVEL * ((quality - 1) * 0.01f)) + 1; + + //compute box kernel sizes + auto wl = (int) sqrt((12 * sigma / MAX_LEVEL) + 1); + if (wl % 2 == 0) --wl; + auto wu = wl + 2; + auto mi = (12 * sigma - MAX_LEVEL * wl * wl - 4 * MAX_LEVEL * wl - 3 * MAX_LEVEL) / (-4 * wl - 4); + auto m = int(mi + 0.5f); + auto extends = 0; + + for (int i = 0; i < data->level; i++) { + data->kernel[i] = ((i < m ? wl : wu) - 1) / 2; + extends += data->kernel[i]; + } + + return extends; +} + + +bool effectGaussianBlurPrepare(RenderEffectGaussianBlur* params) +{ + auto rd = (SwGaussianBlur*)malloc(sizeof(SwGaussianBlur)); + + auto extends = _gaussianInit(rd, params->sigma * params->sigma, params->quality); + + //invalid + if (extends == 0) { + params->invalid = true; + free(rd); + return false; + } + + _gaussianExtendRegion(params->extend, extends, params->direction); + + params->rd = rd; + + return true; +} + + +bool effectGaussianBlur(SwCompositor* cmp, SwSurface* surface, const RenderEffectGaussianBlur* params) +{ + if (cmp->image.channelSize != sizeof(uint32_t)) { + TVGERR("SW_ENGINE", "Not supported grayscale Gaussian Blur!"); + return false; + } + + auto& buffer = surface->compositor->image; + auto data = static_cast(params->rd); + auto& bbox = cmp->bbox; + auto w = (bbox.max.x - bbox.min.x); + auto h = (bbox.max.y - bbox.min.y); + auto stride = cmp->image.stride; + auto front = cmp->image.buf32; + auto back = buffer.buf32; + auto swapped = false; + + TVGLOG("SW_ENGINE", "GaussianFilter region(%ld, %ld, %ld, %ld) params(%f %d %d), level(%d)", bbox.min.x, bbox.min.y, bbox.max.x, bbox.max.y, params->sigma, params->direction, params->border, data->level); + + /* It is best to take advantage of the Gaussian blur’s separable property + by dividing the process into two passes. horizontal and vertical. + We can expect fewer calculations. */ + + //horizontal + if (params->direction != 2) { + for (int i = 0; i < data->level; ++i) { + _gaussianFilter(reinterpret_cast(back), reinterpret_cast(front), stride, w, h, bbox, data->kernel[i], params->border, false); + std::swap(front, back); + swapped = !swapped; + } + } + + //vertical. x/y flipping and horionztal access is pretty compatible with the memory architecture. + if (params->direction != 1) { + rasterXYFlip(front, back, stride, w, h, bbox, false); + std::swap(front, back); + + for (int i = 0; i < data->level; ++i) { + _gaussianFilter(reinterpret_cast(back), reinterpret_cast(front), stride, h, w, bbox, data->kernel[i], params->border, true); + std::swap(front, back); + swapped = !swapped; + } + + rasterXYFlip(front, back, stride, h, w, bbox, true); + std::swap(front, back); + } + + if (swapped) std::swap(cmp->image.buf8, buffer.buf8); + + return true; +} + +/************************************************************************/ +/* Drop Shadow Implementation */ +/************************************************************************/ + +struct SwDropShadow : SwGaussianBlur +{ + SwPoint offset; +}; + + +//TODO: SIMD OPTIMIZATION? +static void _dropShadowFilter(uint32_t* dst, uint32_t* src, int stride, int w, int h, const SwBBox& bbox, int32_t dimension, uint32_t color, bool flipped) +{ + if (flipped) { + src += (bbox.min.x * stride + bbox.min.y); + dst += (bbox.min.x * stride + bbox.min.y); + } else { + src += (bbox.min.y * stride + bbox.min.x); + dst += (bbox.min.y * stride + bbox.min.x); + } + auto iarr = 1.0f / (dimension + dimension + 1); + + #pragma omp parallel for + for (int y = 0; y < h; ++y) { + auto p = y * stride; + auto i = p; //current index + auto l = -(dimension + 1); //left index + auto r = dimension; //right index + int acc = 0; //sliding accumulator + + //initial accumulation + for (int x = l; x < r; ++x) { + auto id = _gaussianEdgeExtend(w, x) + p; + acc += A(src[id]); + } + //perform filtering + for (int x = 0; x < w; ++x, ++r, ++l) { + auto rid = _gaussianEdgeExtend(w, r) + p; + auto lid = _gaussianEdgeExtend(w, l) + p; + acc += A(src[rid]) - A(src[lid]); + dst[i++] = ALPHA_BLEND(color, static_cast(acc * iarr + 0.5f)); + } + } +} + + +static void _dropShadowShift(uint32_t* dst, uint32_t* src, int stride, SwBBox& region, SwPoint& offset, uint8_t opacity, bool direct) +{ + src += (region.min.y * stride + region.min.x); + dst += (region.min.y * stride + region.min.x); + + auto w = region.max.x - region.min.x; + auto h = region.max.y - region.min.y; + auto translucent = (direct || opacity < 255); + + //shift offset + if (region.min.x + offset.x < 0) src -= offset.x; + else dst += offset.x; + + if (region.min.y + offset.y < 0) src -= (offset.y * stride); + else dst += (offset.y * stride); + + for (auto y = 0; y < h; ++y) { + if (translucent) rasterTranslucentPixel32(dst, src, w, opacity); + else rasterPixel32(dst, src, w, opacity); + src += stride; + dst += stride; + } +} + + +static void _dropShadowExtendRegion(RenderRegion& region, int extra, SwPoint& offset) +{ + //bbox region expansion for feathering + region.x = -extra; + region.w = extra * 2; + region.y = -extra; + region.h = extra * 2; + + region.x = std::min(region.x + (int32_t)offset.x, region.x); + region.y = std::min(region.y + (int32_t)offset.y, region.y); + region.w += abs(offset.x); + region.h += abs(offset.y); +} + + +bool effectDropShadowPrepare(RenderEffectDropShadow* params) +{ + auto rd = (SwDropShadow*)malloc(sizeof(SwDropShadow)); + + //compute box kernel sizes + auto extends = _gaussianInit(rd, params->sigma * params->sigma, params->quality); + + //invalid + if (extends == 0 || params->color[3] == 0) { + params->invalid = true; + free(rd); + return false; + } + + //offset + if (params->distance > 0.0f) { + auto radian = tvg::deg2rad(90.0f - params->angle); + rd->offset = {(SwCoord)(params->distance * cosf(radian)), (SwCoord)(-1.0f * params->distance * sinf(radian))}; + } else { + rd->offset = {0, 0}; + } + + //bbox region expansion for feathering + _dropShadowExtendRegion(params->extend, extends, rd->offset); + + params->rd = rd; + + return true; +} + + +//A quite same integration with effectGaussianBlur(). See it for detailed comments. +//surface[0]: the original image, to overlay it into the filtered image. +//surface[1]: temporary buffer for generating the filtered image. +bool effectDropShadow(SwCompositor* cmp, SwSurface* surface[2], const RenderEffectDropShadow* params, uint8_t opacity, bool direct) +{ + if (cmp->image.channelSize != sizeof(uint32_t)) { + TVGERR("SW_ENGINE", "Not supported grayscale Drop Shadow!"); + return false; + } + + //FIXME: if the body is partially visible due to clipping, the shadow also becomes partially visible. + + auto data = static_cast(params->rd); + auto& bbox = cmp->bbox; + auto w = (bbox.max.x - bbox.min.x); + auto h = (bbox.max.y - bbox.min.y); + + //outside the screen + if (abs(data->offset.x) >= w || abs(data->offset.y) >= h) return true; + + SwImage* buffer[] = {&surface[0]->compositor->image, &surface[1]->compositor->image}; + auto color = cmp->recoverSfc->join(params->color[0], params->color[1], params->color[2], 255); + auto stride = cmp->image.stride; + auto front = cmp->image.buf32; + auto back = buffer[1]->buf32; + opacity = MULTIPLY(params->color[3], opacity); + + TVGLOG("SW_ENGINE", "DropShadow region(%ld, %ld, %ld, %ld) params(%f %f %f), level(%d)", bbox.min.x, bbox.min.y, bbox.max.x, bbox.max.y, params->angle, params->distance, params->sigma, data->level); + + //saving the original image in order to overlay it into the filtered image. + _dropShadowFilter(back, front, stride, w, h, bbox, data->kernel[0], color, false); + std::swap(front, buffer[0]->buf32); + std::swap(front, back); + + //horizontal + for (int i = 1; i < data->level; ++i) { + _dropShadowFilter(back, front, stride, w, h, bbox, data->kernel[i], color, false); + std::swap(front, back); + } + + //vertical + rasterXYFlip(front, back, stride, w, h, bbox, false); + std::swap(front, back); + + for (int i = 0; i < data->level; ++i) { + _dropShadowFilter(back, front, stride, h, w, bbox, data->kernel[i], color, true); + std::swap(front, back); + } + + rasterXYFlip(front, back, stride, h, w, bbox, true); + std::swap(cmp->image.buf32, back); + + //draw to the main surface directly + if (direct) { + _dropShadowShift(cmp->recoverSfc->buf32, cmp->image.buf32, stride, bbox, data->offset, opacity, direct); + std::swap(cmp->image.buf32, buffer[0]->buf32); + return true; + } + + //draw to the intermediate surface + rasterClear(surface[1], bbox.min.x, bbox.min.y, w, h); + _dropShadowShift(buffer[1]->buf32, cmp->image.buf32, stride, bbox, data->offset, opacity, direct); + std::swap(cmp->image.buf32, buffer[1]->buf32); + + //compositing shadow and body + auto s = buffer[0]->buf32 + (bbox.min.y * buffer[0]->stride + bbox.min.x); + auto d = cmp->image.buf32 + (bbox.min.y * cmp->image.stride + bbox.min.x); + + for (auto y = 0; y < h; ++y) { + rasterTranslucentPixel32(d, s, w, 255); + s += buffer[0]->stride; + d += cmp->image.stride; + } + + return true; +} diff --git a/thirdparty/thorvg/src/renderer/sw_engine/tvgSwRaster.cpp b/thirdparty/thorvg/src/renderer/sw_engine/tvgSwRaster.cpp index b9e95f88cf79..18ffc18e1e0e 100644 --- a/thirdparty/thorvg/src/renderer/sw_engine/tvgSwRaster.cpp +++ b/thirdparty/thorvg/src/renderer/sw_engine/tvgSwRaster.cpp @@ -490,7 +490,7 @@ static bool _rasterRect(SwSurface* surface, const SwBBox& region, uint8_t r, uin /* Rle */ /************************************************************************/ -static bool _rasterCompositeMaskedRle(SwSurface* surface, SwRleData* rle, SwMask maskOp, uint8_t r, uint8_t g, uint8_t b, uint8_t a) +static bool _rasterCompositeMaskedRle(SwSurface* surface, SwRle* rle, SwMask maskOp, uint8_t r, uint8_t g, uint8_t b, uint8_t a) { auto span = rle->spans; auto cbuffer = surface->compositor->image.buf8; @@ -510,7 +510,7 @@ static bool _rasterCompositeMaskedRle(SwSurface* surface, SwRleData* rle, SwMask } -static bool _rasterDirectMaskedRle(SwSurface* surface, SwRleData* rle, SwMask maskOp, uint8_t r, uint8_t g, uint8_t b, uint8_t a) +static bool _rasterDirectMaskedRle(SwSurface* surface, SwRle* rle, SwMask maskOp, uint8_t r, uint8_t g, uint8_t b, uint8_t a) { auto span = rle->spans; auto cbuffer = surface->compositor->image.buf8; @@ -531,7 +531,7 @@ static bool _rasterDirectMaskedRle(SwSurface* surface, SwRleData* rle, SwMask ma } -static bool _rasterMaskedRle(SwSurface* surface, SwRleData* rle, uint8_t r, uint8_t g, uint8_t b, uint8_t a) +static bool _rasterMaskedRle(SwSurface* surface, SwRle* rle, uint8_t r, uint8_t g, uint8_t b, uint8_t a) { TVGLOG("SW_ENGINE", "Masked(%d) Rle", (int)surface->compositor->method); @@ -545,7 +545,7 @@ static bool _rasterMaskedRle(SwSurface* surface, SwRleData* rle, uint8_t r, uint } -static bool _rasterMattedRle(SwSurface* surface, SwRleData* rle, uint8_t r, uint8_t g, uint8_t b, uint8_t a) +static bool _rasterMattedRle(SwSurface* surface, SwRle* rle, uint8_t r, uint8_t g, uint8_t b, uint8_t a) { TVGLOG("SW_ENGINE", "Matted(%d) Rle", (int)surface->compositor->method); @@ -568,10 +568,8 @@ static bool _rasterMattedRle(SwSurface* surface, SwRleData* rle, uint8_t r, uint *dst = tmp + ALPHA_BLEND(*dst, IA(tmp)); } } - return true; - } //8bit grayscale - if (surface->channelSize == sizeof(uint8_t)) { + } else if (surface->channelSize == sizeof(uint8_t)) { uint8_t src; for (uint32_t i = 0; i < rle->size; ++i, ++span) { auto dst = &surface->buf8[span->y * surface->stride + span->x]; @@ -582,13 +580,12 @@ static bool _rasterMattedRle(SwSurface* surface, SwRleData* rle, uint8_t r, uint *dst = INTERPOLATE8(src, *dst, alpha(cmp)); } } - return true; } - return false; + return true; } -static bool _rasterBlendingRle(SwSurface* surface, const SwRleData* rle, uint8_t r, uint8_t g, uint8_t b, uint8_t a) +static bool _rasterBlendingRle(SwSurface* surface, const SwRle* rle, uint8_t r, uint8_t g, uint8_t b, uint8_t a) { if (surface->channelSize != sizeof(uint32_t)) return false; @@ -612,7 +609,7 @@ static bool _rasterBlendingRle(SwSurface* surface, const SwRleData* rle, uint8_t } -static bool _rasterTranslucentRle(SwSurface* surface, const SwRleData* rle, uint8_t r, uint8_t g, uint8_t b, uint8_t a) +static bool _rasterTranslucentRle(SwSurface* surface, const SwRle* rle, uint8_t r, uint8_t g, uint8_t b, uint8_t a) { #if defined(THORVG_AVX_VECTOR_SUPPORT) return avxRasterTranslucentRle(surface, rle, r, g, b, a); @@ -624,7 +621,7 @@ static bool _rasterTranslucentRle(SwSurface* surface, const SwRleData* rle, uint } -static bool _rasterSolidRle(SwSurface* surface, const SwRleData* rle, uint8_t r, uint8_t g, uint8_t b) +static bool _rasterSolidRle(SwSurface* surface, const SwRle* rle, uint8_t r, uint8_t g, uint8_t b) { auto span = rle->spans; @@ -661,7 +658,7 @@ static bool _rasterSolidRle(SwSurface* surface, const SwRleData* rle, uint8_t r, } -static bool _rasterRle(SwSurface* surface, SwRleData* rle, uint8_t r, uint8_t g, uint8_t b, uint8_t a) +static bool _rasterRle(SwSurface* surface, SwRle* rle, uint8_t r, uint8_t g, uint8_t b, uint8_t a) { if (!rle) return false; @@ -697,66 +694,9 @@ static bool _rasterRle(SwSurface* surface, SwRleData* rle, uint8_t r, uint8_t g, auto sx = (x) * itransform->e11 + itransform->e13 - 0.49f; \ if (sx <= -0.5f || (uint32_t)(sx + 0.5f) >= image->w) continue; \ - -#if 0 //Enable it when GRAYSCALE image is supported -static bool _rasterCompositeScaledMaskedRleImage(SwSurface* surface, const SwImage* image, const Matrix* itransform, const SwBBox& region, SwMask maskOp, uint8_t opacity) -{ - auto scaleMethod = image->scale < DOWN_SCALE_TOLERANCE ? _interpDownScaler : _interpUpScaler; - auto sampleSize = _sampleSize(image->scale); - auto span = image->rle->spans; - int32_t miny = 0, maxy = 0; - - for (uint32_t i = 0; i < image->rle->size; ++i, ++span) { - SCALED_IMAGE_RANGE_Y(span->y) - auto cmp = &surface->compositor->image.buf8[span->y * surface->compositor->image.stride + span->x]; - auto a = MULTIPLY(span->coverage, opacity); - for (uint32_t x = static_cast(span->x); x < static_cast(span->x) + span->len; ++x, ++cmp) { - SCALED_IMAGE_RANGE_X - auto src = scaleMethod(image->buf8, image->stride, image->w, image->h, sx, sy, miny, maxy, sampleSize); - if (a < 255) src = MULTIPLY(src, a); - *cmp = maskOp(src, *cmp, ~src); - } - } - return true; -} - - -static bool _rasterDirectScaledMaskedRleImage(SwSurface* surface, const SwImage* image, const Matrix* itransform, const SwBBox& region, SwMask maskOp, uint8_t opacity) -{ - auto scaleMethod = image->scale < DOWN_SCALE_TOLERANCE ? _interpDownScaler : _interpUpScaler; - auto sampleSize = _sampleSize(image->scale); - auto span = image->rle->spans; - int32_t miny = 0, maxy = 0; - - for (uint32_t i = 0; i < image->rle->size; ++i, ++span) { - SCALED_IMAGE_RANGE_Y(span->y) - auto cmp = &surface->compositor->image.buf8[span->y * surface->compositor->image.stride + span->x]; - auto dst = &surface->buf8[span->y * surface->stride + span->x]; - auto a = MULTIPLY(span->coverage, opacity); - for (uint32_t x = static_cast(span->x); x < static_cast(span->x) + span->len; ++x, ++cmp, ++dst) { - SCALED_IMAGE_RANGE_X - auto src = scaleMethod(image->buf8, image->stride, image->w, image->h, sx, sy, miny, maxy, sampleSize); - if (a < 255) src = MULTIPLY(src, a); - src = maskOp(src, *cmp, 0); //not use alpha - *dst = src + MULTIPLY(*dst, ~src); - } - } - return _compositeMaskImage(surface, &surface->compositor->image, surface->compositor->bbox); -} -#endif - static bool _rasterScaledMaskedRleImage(SwSurface* surface, const SwImage* image, const Matrix* itransform, const SwBBox& region, uint8_t opacity) { -#if 0 //Enable it when GRAYSCALE image is supported - TVGLOG("SW_ENGINE", "Scaled Masked(%d) Rle Image", (int)surface->compositor->method); - - //8bit masking channels composition - if (surface->channelSize != sizeof(uint8_t)) return false; - - auto maskOp = _getMaskOp(surface->compositor->method); - if (_direct(surface->compositor->method)) return _rasterDirectScaledMaskedRleImage(surface, image, itransform, region, maskOp, opacity); - else return _rasterCompositeScaledMaskedRleImage(surface, image, itransform, region, maskOp, opacity); -#endif + TVGERR("SW_ENGINE", "Not Supported Scaled Masked(%d) Rle Image", (int)surface->compositor->method); return false; } @@ -784,7 +724,6 @@ static bool _rasterScaledMattedRleImage(SwSurface* surface, const SwImage* image *dst = src + ALPHA_BLEND(*dst, IA(src)); } } - return true; } @@ -851,7 +790,7 @@ static bool _scaledRleImage(SwSurface* surface, const SwImage* image, const Matr Matrix itransform; - if (!mathInverse(&transform, &itransform)) return true; + if (!inverse(&transform, &itransform)) return true; if (_compositing(surface)) { if (_matting(surface)) return _rasterScaledMattedRleImage(surface, image, &itransform, region, opacity); @@ -869,75 +808,6 @@ static bool _scaledRleImage(SwSurface* surface, const SwImage* image, const Matr /* RLE Direct Image */ /************************************************************************/ -#if 0 //Enable it when GRAYSCALE image is supported -static bool _rasterCompositeDirectMaskedRleImage(SwSurface* surface, const SwImage* image, SwMask maskOp, uint8_t opacity) -{ - auto span = image->rle->spans; - auto cbuffer = surface->compositor->image.buf8; - auto ctride = surface->compositor->image.stride; - - for (uint32_t i = 0; i < image->rle->size; ++i, ++span) { - auto src = image->buf8 + (span->y + image->oy) * image->stride + (span->x + image->ox); - auto cmp = &cbuffer[span->y * ctride + span->x]; - auto alpha = MULTIPLY(span->coverage, opacity); - if (alpha == 255) { - for (uint32_t x = 0; x < span->len; ++x, ++src, ++cmp) { - *cmp = maskOp(*src, *cmp, ~*src); - } - } else { - for (uint32_t x = 0; x < span->len; ++x, ++src, ++cmp) { - auto tmp = MULTIPLY(*src, alpha); - *cmp = maskOp(*src, *cmp, ~tmp); - } - } - } - return _compositeMaskImage(surface, &surface->compositor->image, surface->compositor->bbox); -} - - -static bool _rasterDirectDirectMaskedRleImage(SwSurface* surface, const SwImage* image, SwMask maskOp, uint8_t opacity) -{ - auto span = image->rle->spans; - auto cbuffer = surface->compositor->image.buf8; - auto ctride = surface->compositor->image.stride; - - for (uint32_t i = 0; i < image->rle->size; ++i, ++span) { - auto src = image->buf8 + (span->y + image->oy) * image->stride + (span->x + image->ox); - auto cmp = &cbuffer[span->y * ctride + span->x]; - auto dst = &surface->buf8[span->y * surface->stride + span->x]; - auto alpha = MULTIPLY(span->coverage, opacity); - if (alpha == 255) { - for (uint32_t x = 0; x < span->len; ++x, ++src, ++cmp, ++dst) { - auto tmp = maskOp(*src, *cmp, 0); //not use alpha - *dst = INTERPOLATE8(tmp, *dst, (255 - tmp)); - } - } else { - for (uint32_t x = 0; x < span->len; ++x, ++src, ++cmp, ++dst) { - auto tmp = maskOp(MULTIPLY(*src, alpha), *cmp, 0); //not use alpha - *dst = INTERPOLATE8(tmp, *dst, (255 - tmp)); - } - } - } - return true; -} -#endif - -static bool _rasterDirectMaskedRleImage(SwSurface* surface, const SwImage* image, uint8_t opacity) -{ -#if 0 //Enable it when GRAYSCALE image is supported - TVGLOG("SW_ENGINE", "Direct Masked(%d) Rle Image", (int)surface->compositor->method); - - //8bit masking channels composition - if (surface->channelSize != sizeof(uint8_t)) return false; - - auto maskOp = _getMaskOp(surface->compositor->method); - if (_direct(surface->compositor->method)) _rasterDirectDirectMaskedRleImage(surface, image, maskOp, opacity); - else return _rasterCompositeDirectMaskedRleImage(surface, image, maskOp, opacity); -#endif - return false; -} - - static bool _rasterDirectMattedRleImage(SwSurface* surface, const SwImage* image, uint8_t opacity) { TVGLOG("SW_ENGINE", "Direct Matted(%d) Rle Image", (int)surface->compositor->method); @@ -999,21 +869,19 @@ static bool _rasterDirectRleImage(SwSurface* surface, const SwImage* image, uint auto dst = &surface->buf32[span->y * surface->stride + span->x]; auto img = image->buf32 + (span->y + image->oy) * image->stride + (span->x + image->ox); auto alpha = MULTIPLY(span->coverage, opacity); - if (alpha == 255) { - for (uint32_t x = 0; x < span->len; ++x, ++dst, ++img) { - *dst = *img + ALPHA_BLEND(*dst, IA(*img)); - } - } else { - for (uint32_t x = 0; x < span->len; ++x, ++dst, ++img) { - auto src = ALPHA_BLEND(*img, alpha); - *dst = src + ALPHA_BLEND(*dst, IA(src)); - } - } + rasterTranslucentPixel32(dst, img, span->len, alpha); } return true; } +static bool _rasterDirectMaskedRleImage(SwSurface* surface, const SwImage* image, uint8_t opacity) +{ + TVGERR("SW_ENGINE", "Not Supported Direct Masked(%d) Rle Image", (int)surface->compositor->method); + return false; +} + + static bool _directRleImage(SwSurface* surface, const SwImage* image, uint8_t opacity) { if (surface->channelSize == sizeof(uint8_t)) { @@ -1037,67 +905,9 @@ static bool _directRleImage(SwSurface* surface, const SwImage* image, uint8_t op /*Scaled Image */ /************************************************************************/ -#if 0 //Enable it when GRAYSCALE image is supported -static bool _rasterCompositeScaledMaskedImage(SwSurface* surface, const SwImage* image, const Matrix* itransform, const SwBBox& region, SwMask maskOp, uint8_t opacity) -{ - auto scaleMethod = image->scale < DOWN_SCALE_TOLERANCE ? _interpDownScaler : _interpUpScaler; - auto sampleSize = _sampleSize(image->scale); - auto cstride = surface->compositor->image.stride; - auto cbuffer = surface->compositor->image.buf8 + (region.min.y * cstride + region.min.x); - int32_t miny = 0, maxy = 0; - - for (auto y = region.min.y; y < region.max.y; ++y) { - SCALED_IMAGE_RANGE_Y(y) - auto cmp = cbuffer; - for (auto x = region.min.x; x < region.max.x; ++x, ++cmp) { - SCALED_IMAGE_RANGE_X - auto src = scaleMethod(image->buf8, image->stride, image->w, image->h, sx, sy, miny, maxy, sampleSize); - if (opacity < 255) src = MULTIPLY(src, opacity); - *cmp = maskOp(src, *cmp, ~src); - } - cbuffer += cstride; - } - return _compositeMaskImage(surface, &surface->compositor->image, surface->compositor->bbox); -} - - -static bool _rasterDirectScaledMaskedImage(SwSurface* surface, const SwImage* image, const Matrix* itransform, const SwBBox& region, SwMask maskOp, uint8_t opacity) -{ - auto scaleMethod = image->scale < DOWN_SCALE_TOLERANCE ? _interpDownScaler : _interpUpScaler; - auto sampleSize = _sampleSize(image->scale); - auto cstride = surface->compositor->image.stride; - auto cbuffer = surface->compositor->image.buf8 + (region.min.y * cstride + region.min.x); - auto dbuffer = surface->buf8 + (region.min.y * surface->stride + region.min.x); - int32_t miny = 0, maxy = 0; - - for (auto y = region.min.y; y < region.max.y; ++y) { - SCALED_IMAGE_RANGE_Y(y) - auto cmp = cbuffer; - auto dst = dbuffer; - for (auto x = region.min.x; x < region.max.x; ++x, ++cmp, ++dst) { - SCALED_IMAGE_RANGE_X - auto src = scaleMethod(image->buf8, image->stride, image->w, image->h, sx, sy, miny, maxy, sampleSize); - if (opacity < 255) src = MULTIPLY(src, opacity); - src = maskOp(src, *cmp, 0); //not use alpha - *dst = src + MULTIPLY(*dst, ~src); - } - cbuffer += cstride; - dbuffer += surface->stride; - } - return true; -} -#endif - static bool _rasterScaledMaskedImage(SwSurface* surface, const SwImage* image, const Matrix* itransform, const SwBBox& region, uint8_t opacity) { - TVGERR("SW_ENGINE", "Not supported ScaledMaskedImage!"); -#if 0 //Enable it when GRAYSCALE image is supported - TVGLOG("SW_ENGINE", "Scaled Masked(%d) Image [Region: %lu %lu %lu %lu]", (int)surface->compositor->method, region.min.x, region.min.y, region.max.x - region.min.x, region.max.y - region.min.y); - - auto maskOp = _getMaskOp(surface->compositor->method); - if (_direct(surface->compositor->method)) return _rasterDirectScaledMaskedImage(surface, image, itransform, region, maskOp, opacity); - else return _rasterCompositeScaledMaskedImage(surface, image, itransform, region, maskOp, opacity); -#endif + TVGERR("SW_ENGINE", "Not Supported Scaled Masked Image!"); return false; } @@ -1202,7 +1012,7 @@ static bool _scaledImage(SwSurface* surface, const SwImage* image, const Matrix& { Matrix itransform; - if (!mathInverse(&transform, &itransform)) return true; + if (!inverse(&transform, &itransform)) return true; if (_compositing(surface)) { if (_matting(surface)) return _rasterScaledMattedImage(surface, image, &itransform, region, opacity); @@ -1220,78 +1030,9 @@ static bool _scaledImage(SwSurface* surface, const SwImage* image, const Matrix& /* Direct Image */ /************************************************************************/ -#if 0 //Enable it when GRAYSCALE image is supported -static bool _rasterCompositeDirectMaskedImage(SwSurface* surface, const SwImage* image, const SwBBox& region, SwMask maskOp, uint8_t opacity) -{ - auto h = static_cast(region.max.y - region.min.y); - auto w = static_cast(region.max.x - region.min.x); - auto cstride = surface->compositor->image.stride; - - auto cbuffer = surface->compositor->image.buf8 + (region.min.y * cstride + region.min.x); //compositor buffer - auto sbuffer = image->buf8 + (region.min.y + image->oy) * image->stride + (region.min.x + image->ox); - - for (uint32_t y = 0; y < h; ++y) { - auto cmp = cbuffer; - auto src = sbuffer; - if (opacity == 255) { - for (uint32_t x = 0; x < w; ++x, ++src, ++cmp) { - *cmp = maskOp(*src, *cmp, ~*src); - } - } else { - for (uint32_t x = 0; x < w; ++x, ++src, ++cmp) { - auto tmp = MULTIPLY(*src, opacity); - *cmp = maskOp(tmp, *cmp, ~tmp); - } - } - cbuffer += cstride; - sbuffer += image->stride; - } - return _compositeMaskImage(surface, &surface->compositor->image, surface->compositor->bbox); -} - - -static bool _rasterDirectDirectMaskedImage(SwSurface* surface, const SwImage* image, const SwBBox& region, SwMask maskOp, uint8_t opacity) -{ - auto h = static_cast(region.max.y - region.min.y); - auto w = static_cast(region.max.x - region.min.x); - auto cstride = surface->compositor->image.stride; - - auto cbuffer = surface->compositor->image.buf32 + (region.min.y * cstride + region.min.x); //compositor buffer - auto dbuffer = surface->buf8 + (region.min.y * surface->stride + region.min.x); //destination buffer - auto sbuffer = image->buf8 + (region.min.y + image->oy) * image->stride + (region.min.x + image->ox); - - for (uint32_t y = 0; y < h; ++y) { - auto cmp = cbuffer; - auto dst = dbuffer; - auto src = sbuffer; - if (opacity == 255) { - for (uint32_t x = 0; x < w; ++x, ++src, ++cmp, ++dst) { - auto tmp = maskOp(*src, *cmp, 0); //not use alpha - *dst = tmp + MULTIPLY(*dst, ~tmp); - } - } else { - for (uint32_t x = 0; x < w; ++x, ++src, ++cmp, ++dst) { - auto tmp = maskOp(MULTIPLY(*src, opacity), *cmp, 0); //not use alpha - *dst = tmp + MULTIPLY(*dst, ~tmp); - } - } - cbuffer += cstride; - dbuffer += surface->stride; - sbuffer += image->stride; - } - return true; -} -#endif - static bool _rasterDirectMaskedImage(SwSurface* surface, const SwImage* image, const SwBBox& region, uint8_t opacity) { - TVGERR("SW_ENGINE", "Not Supported: Direct Masked(%d) Image [Region: %lu %lu %lu %lu]", (int)surface->compositor->method, region.min.x, region.min.y, region.max.x - region.min.x, region.max.y - region.min.y); - -#if 0 //Enable it when GRAYSCALE image is supported - auto maskOp = _getMaskOp(surface->compositor->method); - if (_direct(surface->compositor->method)) return _rasterDirectDirectMaskedImage(surface, image, region, maskOp, opacity); - else return _rasterCompositeDirectMaskedImage(surface, image, region, maskOp, opacity); -#endif + TVGERR("SW_ENGINE", "Not Supported: Direct Masked Image"); return false; } @@ -1394,37 +1135,24 @@ static bool _rasterDirectImage(SwSurface* surface, const SwImage* image, const S //32bits channels if (surface->channelSize == sizeof(uint32_t)) { auto dbuffer = &surface->buf32[region.min.y * surface->stride + region.min.x]; - for (auto y = region.min.y; y < region.max.y; ++y) { - auto dst = dbuffer; - auto src = sbuffer; - if (opacity == 255) { - for (auto x = region.min.x; x < region.max.x; x++, dst++, src++) { - *dst = *src + ALPHA_BLEND(*dst, IA(*src)); - } - } else { - for (auto x = region.min.x; x < region.max.x; ++x, ++dst, ++src) { - auto tmp = ALPHA_BLEND(*src, opacity); - *dst = tmp + ALPHA_BLEND(*dst, IA(tmp)); - } - } + rasterTranslucentPixel32(dbuffer, sbuffer, region.max.x - region.min.x, opacity); dbuffer += surface->stride; sbuffer += image->stride; } //8bits grayscale } else if (surface->channelSize == sizeof(uint8_t)) { auto dbuffer = &surface->buf8[region.min.y * surface->stride + region.min.x]; - for (auto y = region.min.y; y < region.max.y; ++y, dbuffer += surface->stride, sbuffer += image->stride) { auto dst = dbuffer; auto src = sbuffer; if (opacity == 255) { for (auto x = region.min.x; x < region.max.x; ++x, ++dst, ++src) { - *dst = *src + MULTIPLY(*dst, ~*src); + *dst = *src + MULTIPLY(*dst, IA(*src)); } } else { for (auto x = region.min.x; x < region.max.x; ++x, ++dst, ++src) { - *dst = INTERPOLATE8(*src, *dst, opacity); + *dst = INTERPOLATE8(A(*src), *dst, opacity); } } } @@ -1602,13 +1330,23 @@ static bool _rasterBlendingGradientRect(SwSurface* surface, const SwBBox& region template static bool _rasterTranslucentGradientRect(SwSurface* surface, const SwBBox& region, const SwFill* fill) { - auto buffer = surface->buf32 + (region.min.y * surface->stride) + region.min.x; auto h = static_cast(region.max.y - region.min.y); auto w = static_cast(region.max.x - region.min.x); - for (uint32_t y = 0; y < h; ++y) { - fillMethod()(fill, buffer, region.min.y + y, region.min.x, w, opBlendPreNormal, 255); - buffer += surface->stride; + //32 bits + if (surface->channelSize == sizeof(uint32_t)) { + auto buffer = surface->buf32 + (region.min.y * surface->stride) + region.min.x; + for (uint32_t y = 0; y < h; ++y) { + fillMethod()(fill, buffer, region.min.y + y, region.min.x, w, opBlendPreNormal, 255); + buffer += surface->stride; + } + //8 bits + } else if (surface->channelSize == sizeof(uint8_t)) { + auto buffer = surface->buf8 + (region.min.y * surface->stride) + region.min.x; + for (uint32_t y = 0; y < h; ++y) { + fillMethod()(fill, buffer, region.min.y + y, region.min.x, w, _opMaskAdd, 255); + buffer += surface->stride; + } } return true; } @@ -1617,12 +1355,23 @@ static bool _rasterTranslucentGradientRect(SwSurface* surface, const SwBBox& reg template static bool _rasterSolidGradientRect(SwSurface* surface, const SwBBox& region, const SwFill* fill) { - auto buffer = surface->buf32 + (region.min.y * surface->stride) + region.min.x; auto w = static_cast(region.max.x - region.min.x); auto h = static_cast(region.max.y - region.min.y); - for (uint32_t y = 0; y < h; ++y) { - fillMethod()(fill, buffer + y * surface->stride, region.min.y + y, region.min.x, w, opBlendSrcOver, 255); + //32 bits + if (surface->channelSize == sizeof(uint32_t)) { + auto buffer = surface->buf32 + (region.min.y * surface->stride) + region.min.x; + for (uint32_t y = 0; y < h; ++y) { + fillMethod()(fill, buffer, region.min.y + y, region.min.x, w, opBlendSrcOver, 255); + buffer += surface->stride; + } + //8 bits + } else if (surface->channelSize == sizeof(uint8_t)) { + auto buffer = surface->buf8 + (region.min.y * surface->stride) + region.min.x; + for (uint32_t y = 0; y < h; ++y) { + fillMethod()(fill, buffer, region.min.y + y, region.min.x, w, _opMaskNone, 255); + buffer += surface->stride; + } } return true; } @@ -1663,7 +1412,7 @@ static bool _rasterRadialGradientRect(SwSurface* surface, const SwBBox& region, /************************************************************************/ template -static bool _rasterCompositeGradientMaskedRle(SwSurface* surface, const SwRleData* rle, const SwFill* fill, SwMask maskOp) +static bool _rasterCompositeGradientMaskedRle(SwSurface* surface, const SwRle* rle, const SwFill* fill, SwMask maskOp) { auto span = rle->spans; auto cstride = surface->compositor->image.stride; @@ -1678,7 +1427,7 @@ static bool _rasterCompositeGradientMaskedRle(SwSurface* surface, const SwRleDat template -static bool _rasterDirectGradientMaskedRle(SwSurface* surface, const SwRleData* rle, const SwFill* fill, SwMask maskOp) +static bool _rasterDirectGradientMaskedRle(SwSurface* surface, const SwRle* rle, const SwFill* fill, SwMask maskOp) { auto span = rle->spans; auto cstride = surface->compositor->image.stride; @@ -1695,7 +1444,7 @@ static bool _rasterDirectGradientMaskedRle(SwSurface* surface, const SwRleData* template -static bool _rasterGradientMaskedRle(SwSurface* surface, const SwRleData* rle, const SwFill* fill) +static bool _rasterGradientMaskedRle(SwSurface* surface, const SwRle* rle, const SwFill* fill) { auto method = surface->compositor->method; @@ -1710,7 +1459,7 @@ static bool _rasterGradientMaskedRle(SwSurface* surface, const SwRleData* rle, c template -static bool _rasterGradientMattedRle(SwSurface* surface, const SwRleData* rle, const SwFill* fill) +static bool _rasterGradientMattedRle(SwSurface* surface, const SwRle* rle, const SwFill* fill) { TVGLOG("SW_ENGINE", "Matted(%d) Rle Linear Gradient", (int)surface->compositor->method); @@ -1729,7 +1478,7 @@ static bool _rasterGradientMattedRle(SwSurface* surface, const SwRleData* rle, c template -static bool _rasterBlendingGradientRle(SwSurface* surface, const SwRleData* rle, const SwFill* fill) +static bool _rasterBlendingGradientRle(SwSurface* surface, const SwRle* rle, const SwFill* fill) { auto span = rle->spans; @@ -1742,7 +1491,7 @@ static bool _rasterBlendingGradientRle(SwSurface* surface, const SwRleData* rle, template -static bool _rasterTranslucentGradientRle(SwSurface* surface, const SwRleData* rle, const SwFill* fill) +static bool _rasterTranslucentGradientRle(SwSurface* surface, const SwRle* rle, const SwFill* fill) { auto span = rle->spans; @@ -1757,7 +1506,7 @@ static bool _rasterTranslucentGradientRle(SwSurface* surface, const SwRleData* r } else if (surface->channelSize == sizeof(uint8_t)) { for (uint32_t i = 0; i < rle->size; ++i, ++span) { auto dst = &surface->buf8[span->y * surface->stride + span->x]; - fillMethod()(fill, dst, span->y, span->x, span->len, _opMaskAdd, 255); + fillMethod()(fill, dst, span->y, span->x, span->len, _opMaskAdd, span->coverage); } } return true; @@ -1765,7 +1514,7 @@ static bool _rasterTranslucentGradientRle(SwSurface* surface, const SwRleData* r template -static bool _rasterSolidGradientRle(SwSurface* surface, const SwRleData* rle, const SwFill* fill) +static bool _rasterSolidGradientRle(SwSurface* surface, const SwRle* rle, const SwFill* fill) { auto span = rle->spans; @@ -1789,7 +1538,7 @@ static bool _rasterSolidGradientRle(SwSurface* surface, const SwRleData* rle, co } -static bool _rasterLinearGradientRle(SwSurface* surface, const SwRleData* rle, const SwFill* fill) +static bool _rasterLinearGradientRle(SwSurface* surface, const SwRle* rle, const SwFill* fill) { if (!rle) return false; @@ -1806,7 +1555,7 @@ static bool _rasterLinearGradientRle(SwSurface* surface, const SwRleData* rle, c } -static bool _rasterRadialGradientRle(SwSurface* surface, const SwRleData* rle, const SwFill* fill) +static bool _rasterRadialGradientRle(SwSurface* surface, const SwRle* rle, const SwFill* fill) { if (!rle) return false; @@ -1814,9 +1563,9 @@ static bool _rasterRadialGradientRle(SwSurface* surface, const SwRleData* rle, c if (_matting(surface)) return _rasterGradientMattedRle(surface, rle, fill); else return _rasterGradientMaskedRle(surface, rle, fill); } else if (_blending(surface)) { - _rasterBlendingGradientRle(surface, rle, fill); + return _rasterBlendingGradientRle(surface, rle, fill); } else { - if (fill->translucent) _rasterTranslucentGradientRle(surface, rle, fill); + if (fill->translucent) return _rasterTranslucentGradientRle(surface, rle, fill); else return _rasterSolidGradientRle(surface, rle, fill); } return false; @@ -1827,6 +1576,19 @@ static bool _rasterRadialGradientRle(SwSurface* surface, const SwRleData* rle, c /* External Class Implementation */ /************************************************************************/ +void rasterTranslucentPixel32(uint32_t* dst, uint32_t* src, uint32_t len, uint8_t opacity) +{ + //TODO: Support SIMD accelerations + cRasterTranslucentPixels(dst, src, len, opacity); +} + + +void rasterPixel32(uint32_t* dst, uint32_t* src, uint32_t len, uint8_t opacity) +{ + //TODO: Support SIMD accelerations + cRasterPixels(dst, src, len, opacity); +} + void rasterGrayscale8(uint8_t *dst, uint8_t val, uint32_t offset, int32_t len) { @@ -1905,7 +1667,7 @@ bool rasterClear(SwSurface* surface, uint32_t x, uint32_t y, uint32_t w, uint32_ } -void rasterUnpremultiply(Surface* surface) +void rasterUnpremultiply(RenderSurface* surface) { if (surface->channelSize != sizeof(uint32_t)) return; @@ -1935,7 +1697,7 @@ void rasterUnpremultiply(Surface* surface) } -void rasterPremultiply(Surface* surface) +void rasterPremultiply(RenderSurface* surface) { ScopedLock lock(surface->key); if (surface->premultiplied || (surface->channelSize != sizeof(uint32_t))) return; @@ -1965,13 +1727,13 @@ bool rasterGradientShape(SwSurface* surface, SwShape* shape, const Fill* fdata, return a > 0 ? rasterShape(surface, shape, color->r, color->g, color->b, a) : true; } - auto id = fdata->identifier(); + auto type = fdata->type(); if (shape->fastTrack) { - if (id == TVG_CLASS_ID_LINEAR) return _rasterLinearGradientRect(surface, shape->bbox, shape->fill); - else if (id == TVG_CLASS_ID_RADIAL)return _rasterRadialGradientRect(surface, shape->bbox, shape->fill); + if (type == Type::LinearGradient) return _rasterLinearGradientRect(surface, shape->bbox, shape->fill); + else if (type == Type::RadialGradient)return _rasterRadialGradientRect(surface, shape->bbox, shape->fill); } else { - if (id == TVG_CLASS_ID_LINEAR) return _rasterLinearGradientRle(surface, shape->rle, shape->fill); - else if (id == TVG_CLASS_ID_RADIAL) return _rasterRadialGradientRle(surface, shape->rle, shape->fill); + if (type == Type::LinearGradient) return _rasterLinearGradientRle(surface, shape->rle, shape->fill); + else if (type == Type::RadialGradient) return _rasterRadialGradientRle(surface, shape->rle, shape->fill); } return false; } @@ -1986,9 +1748,9 @@ bool rasterGradientStroke(SwSurface* surface, SwShape* shape, const Fill* fdata, return a > 0 ? rasterStroke(surface, shape, color->r, color->g, color->b, a) : true; } - auto id = fdata->identifier(); - if (id == TVG_CLASS_ID_LINEAR) return _rasterLinearGradientRle(surface, shape->strokeRle, shape->stroke->fill); - else if (id == TVG_CLASS_ID_RADIAL) return _rasterRadialGradientRle(surface, shape->strokeRle, shape->stroke->fill); + auto type = fdata->type(); + if (type == Type::LinearGradient) return _rasterLinearGradientRle(surface, shape->strokeRle, shape->stroke->fill); + else if (type == Type::RadialGradient) return _rasterRadialGradientRle(surface, shape->strokeRle, shape->stroke->fill); return false; } @@ -2027,12 +1789,12 @@ bool rasterImage(SwSurface* surface, SwImage* image, const Matrix& transform, co } -bool rasterConvertCS(Surface* surface, ColorSpace to) +bool rasterConvertCS(RenderSurface* surface, ColorSpace to) { ScopedLock lock(surface->key); if (surface->cs == to) return true; - //TOOD: Support SIMD accelerations + //TODO: Support SIMD accelerations auto from = surface->cs; if (((from == ColorSpace::ABGR8888) || (from == ColorSpace::ABGR8888S)) && ((to == ColorSpace::ARGB8888) || (to == ColorSpace::ARGB8888S))) { @@ -2045,3 +1807,39 @@ bool rasterConvertCS(Surface* surface, ColorSpace to) } return false; } + + +//TODO: SIMD OPTIMIZATION? +void rasterXYFlip(uint32_t* src, uint32_t* dst, int32_t stride, int32_t w, int32_t h, const SwBBox& bbox, bool flipped) +{ + constexpr int BLOCK = 8; //experimental decision + + if (flipped) { + src += ((bbox.min.x * stride) + bbox.min.y); + dst += ((bbox.min.y * stride) + bbox.min.x); + } else { + src += ((bbox.min.y * stride) + bbox.min.x); + dst += ((bbox.min.x * stride) + bbox.min.y); + } + + #pragma omp parallel for + for (int x = 0; x < w; x += BLOCK) { + auto bx = std::min(w, x + BLOCK) - x; + auto in = &src[x]; + auto out = &dst[x * stride]; + for (int y = 0; y < h; y += BLOCK) { + auto p = &in[y * stride]; + auto q = &out[y]; + auto by = std::min(h, y + BLOCK) - y; + for (int xx = 0; xx < bx; ++xx) { + for (int yy = 0; yy < by; ++yy) { + *q = *p; + p += stride; + ++q; + } + p += 1 - by * stride; + q += stride - by; + } + } + } +} diff --git a/thirdparty/thorvg/src/renderer/sw_engine/tvgSwRasterAvx.h b/thirdparty/thorvg/src/renderer/sw_engine/tvgSwRasterAvx.h index a072a88819a9..79cab043f243 100644 --- a/thirdparty/thorvg/src/renderer/sw_engine/tvgSwRasterAvx.h +++ b/thirdparty/thorvg/src/renderer/sw_engine/tvgSwRasterAvx.h @@ -158,7 +158,7 @@ static bool avxRasterTranslucentRect(SwSurface* surface, const SwBBox& region, u } -static bool avxRasterTranslucentRle(SwSurface* surface, const SwRleData* rle, uint8_t r, uint8_t g, uint8_t b, uint8_t a) +static bool avxRasterTranslucentRle(SwSurface* surface, const SwRle* rle, uint8_t r, uint8_t g, uint8_t b, uint8_t a) { auto span = rle->spans; @@ -185,7 +185,7 @@ static bool avxRasterTranslucentRle(SwSurface* surface, const SwRleData* rle, ui } //2. fill the aligned memory using avx - N_32BITS_IN_128REG pixels processed at once - //In order to avoid unneccessary avx variables declarations a check is made whether there are any iterations at all + //In order to avoid unnecessary avx variables declarations a check is made whether there are any iterations at all uint32_t iterations = (span->len - notAligned) / N_32BITS_IN_128REG; uint32_t avxFilled = 0; if (iterations > 0) { diff --git a/thirdparty/thorvg/src/renderer/sw_engine/tvgSwRasterC.h b/thirdparty/thorvg/src/renderer/sw_engine/tvgSwRasterC.h index 6d0bd9383dad..d79da0e4d8ce 100644 --- a/thirdparty/thorvg/src/renderer/sw_engine/tvgSwRasterC.h +++ b/thirdparty/thorvg/src/renderer/sw_engine/tvgSwRasterC.h @@ -20,6 +20,38 @@ * SOFTWARE. */ + +template +static void inline cRasterTranslucentPixels(PIXEL_T* dst, PIXEL_T* src, uint32_t len, uint32_t opacity) +{ + //TODO: 64bits faster? + if (opacity == 255) { + for (uint32_t x = 0; x < len; ++x, ++dst, ++src) { + *dst = *src + ALPHA_BLEND(*dst, IA(*src)); + } + } else { + for (uint32_t x = 0; x < len; ++x, ++dst, ++src) { + auto tmp = ALPHA_BLEND(*src, opacity); + *dst = tmp + ALPHA_BLEND(*dst, IA(tmp)); + } + } +} + + +template +static void inline cRasterPixels(PIXEL_T* dst, PIXEL_T* src, uint32_t len, uint32_t opacity) +{ + //TODO: 64bits faster? + if (opacity == 255) { + for (uint32_t x = 0; x < len; ++x, ++dst, ++src) { + *dst = *src; + } + } else { + cRasterTranslucentPixels(dst, src, len, opacity); + } +} + + template static void inline cRasterPixels(PIXEL_T* dst, PIXEL_T val, uint32_t offset, int32_t len) { @@ -60,7 +92,7 @@ static void inline cRasterPixels(PIXEL_T* dst, PIXEL_T val, uint32_t offset, int } -static bool inline cRasterTranslucentRle(SwSurface* surface, const SwRleData* rle, uint8_t r, uint8_t g, uint8_t b, uint8_t a) +static bool inline cRasterTranslucentRle(SwSurface* surface, const SwRle* rle, uint8_t r, uint8_t g, uint8_t b, uint8_t a) { auto span = rle->spans; @@ -125,7 +157,7 @@ static bool inline cRasterTranslucentRect(SwSurface* surface, const SwBBox& regi } -static bool inline cRasterABGRtoARGB(Surface* surface) +static bool inline cRasterABGRtoARGB(RenderSurface* surface) { TVGLOG("SW_ENGINE", "Convert ColorSpace ABGR - ARGB [Size: %d x %d]", surface->w, surface->h); @@ -156,7 +188,7 @@ static bool inline cRasterABGRtoARGB(Surface* surface) } -static bool inline cRasterARGBtoABGR(Surface* surface) +static bool inline cRasterARGBtoABGR(RenderSurface* surface) { //exactly same with ABGRtoARGB return cRasterABGRtoARGB(surface); diff --git a/thirdparty/thorvg/src/renderer/sw_engine/tvgSwRasterNeon.h b/thirdparty/thorvg/src/renderer/sw_engine/tvgSwRasterNeon.h index 91cf7743c190..fe693b7f33ac 100644 --- a/thirdparty/thorvg/src/renderer/sw_engine/tvgSwRasterNeon.h +++ b/thirdparty/thorvg/src/renderer/sw_engine/tvgSwRasterNeon.h @@ -89,7 +89,7 @@ static void neonRasterPixel32(uint32_t *dst, uint32_t val, uint32_t offset, int3 } -static bool neonRasterTranslucentRle(SwSurface* surface, const SwRleData* rle, uint8_t r, uint8_t g, uint8_t b, uint8_t a) +static bool neonRasterTranslucentRle(SwSurface* surface, const SwRle* rle, uint8_t r, uint8_t g, uint8_t b, uint8_t a) { auto span = rle->spans; diff --git a/thirdparty/thorvg/src/renderer/sw_engine/tvgSwRasterTexmap.h b/thirdparty/thorvg/src/renderer/sw_engine/tvgSwRasterTexmap.h index 88ef2118f278..1162edc8381c 100644 --- a/thirdparty/thorvg/src/renderer/sw_engine/tvgSwRasterTexmap.h +++ b/thirdparty/thorvg/src/renderer/sw_engine/tvgSwRasterTexmap.h @@ -20,6 +20,17 @@ * SOFTWARE. */ +struct Vertex +{ + Point pt; + Point uv; +}; + +struct Polygon +{ + Vertex vertex[3]; +}; + struct AALine { int32_t x[2]; @@ -53,10 +64,12 @@ static bool _arrange(const SwImage* image, const SwBBox* region, int& yStart, in regionBottom = image->rle->spans[image->rle->size - 1].y; } + if (yStart >= regionBottom) return false; + if (yStart < regionTop) yStart = regionTop; if (yEnd > regionBottom) yEnd = regionBottom; - return yEnd > yStart; + return true; } @@ -673,7 +686,7 @@ static void _rasterPolygonImage(SwSurface* surface, const SwImage* image, const auto denom = ((x[2] - x[0]) * (y[1] - y[0]) - (x[1] - x[0]) * (y[2] - y[0])); //Skip poly if it's an infinitely thin line - if (mathZero(denom)) return; + if (tvg::zero(denom)) return; denom = 1 / denom; //Reciprocal for speeding up dudx = ((u[2] - u[0]) * (y[1] - y[0]) - (u[1] - u[0]) * (y[2] - y[0])) * denom; @@ -689,8 +702,8 @@ static void _rasterPolygonImage(SwSurface* surface, const SwImage* image, const //Determine which side of the polygon the longer edge is on auto side = (dxdy[1] > dxdy[0]) ? true : false; - if (mathEqual(y[0], y[1])) side = x[0] > x[1]; - if (mathEqual(y[1], y[2])) side = x[2] > x[1]; + if (tvg::equal(y[0], y[1])) side = x[0] > x[1]; + if (tvg::equal(y[1], y[2])) side = x[2] > x[1]; auto regionTop = region ? region->min.y : image->rle->spans->y; //Normal Image or Rle Image? auto compositing = _compositing(surface); //Composition required diff --git a/thirdparty/thorvg/src/renderer/sw_engine/tvgSwRenderer.cpp b/thirdparty/thorvg/src/renderer/sw_engine/tvgSwRenderer.cpp index 116c4e813527..180f3cc37a2b 100644 --- a/thirdparty/thorvg/src/renderer/sw_engine/tvgSwRenderer.cpp +++ b/thirdparty/thorvg/src/renderer/sw_engine/tvgSwRenderer.cpp @@ -20,6 +20,9 @@ * SOFTWARE. */ +#ifdef THORVG_SW_OPENMP_SUPPORT + #include +#endif #include #include "tvgMath.h" #include "tvgSwCommon.h" @@ -38,7 +41,7 @@ struct SwTask : Task { SwSurface* surface = nullptr; SwMpool* mpool = nullptr; - SwBBox bbox = {{0, 0}, {0, 0}}; //Whole Rendering Region + SwBBox bbox; //Rendering Region Matrix transform; Array clips; RenderUpdateFlag flags = RenderUpdateFlag::None; @@ -65,9 +68,7 @@ struct SwTask : Task } virtual void dispose() = 0; - virtual bool clip(SwRleData* target) = 0; - virtual SwRleData* rle() = 0; - + virtual bool clip(SwRle* target) = 0; virtual ~SwTask() {} }; @@ -92,38 +93,34 @@ struct SwShapeTask : SwTask if (!rshape->stroke) return 0.0f; auto width = rshape->stroke->width; - if (mathZero(width)) return 0.0f; + if (tvg::zero(width)) return 0.0f; if (!rshape->stroke->fill && (MULTIPLY(rshape->stroke->color[3], opacity) == 0)) return 0.0f; - if (mathZero(rshape->stroke->trim.begin - rshape->stroke->trim.end)) return 0.0f; + if (tvg::zero(rshape->stroke->trim.begin - rshape->stroke->trim.end)) return 0.0f; return (width * sqrt(transform.e11 * transform.e11 + transform.e12 * transform.e12)); } - bool clip(SwRleData* target) override + bool clip(SwRle* target) override { - if (shape.fastTrack) rleClipRect(target, &bbox); - else if (shape.rle) rleClipPath(target, shape.rle); + if (shape.fastTrack) rleClip(target, &bbox); + else if (shape.rle) rleClip(target, shape.rle); else return false; return true; } - SwRleData* rle() override - { - if (!shape.rle && shape.fastTrack) { - shape.rle = rleRender(&shape.bbox); - } - return shape.rle; - } - void run(unsigned tid) override { - if (opacity == 0 && !clipper) return; //Invisible + //Invisible + if (opacity == 0 && !clipper) { + bbox.reset(); + return; + } auto strokeWidth = validStrokeWidth(); - bool visibleFill = false; - auto clipRegion = bbox; + SwBBox renderRegion{}; + auto visibleFill = false; //This checks also for the case, if the invisible shape turned to visible by alpha. auto prepareShape = false; @@ -135,10 +132,11 @@ struct SwShapeTask : SwTask rshape->fillColor(nullptr, nullptr, nullptr, &alpha); alpha = MULTIPLY(alpha, opacity); visibleFill = (alpha > 0 || rshape->fill); + shapeReset(&shape); if (visibleFill || clipper) { - shapeReset(&shape); - if (!shapePrepare(&shape, rshape, transform, clipRegion, bbox, mpool, tid, clips.count > 0 ? true : false)) { + if (!shapePrepare(&shape, rshape, transform, bbox, renderRegion, mpool, tid, clips.count > 0 ? true : false)) { visibleFill = false; + renderRegion.reset(); } } } @@ -159,8 +157,8 @@ struct SwShapeTask : SwTask if (flags & (RenderUpdateFlag::Path | RenderUpdateFlag::Stroke | RenderUpdateFlag::Transform)) { if (strokeWidth > 0.0f) { shapeResetStroke(&shape, rshape, transform); - if (!shapeGenStrokeRle(&shape, rshape, transform, clipRegion, bbox, mpool, tid)) goto err; + if (!shapeGenStrokeRle(&shape, rshape, transform, bbox, renderRegion, mpool, tid)) goto err; if (auto fill = rshape->strokeFill()) { auto ctable = (flags & RenderUpdateFlag::GradientStroke) ? true : false; if (ctable) shapeResetStrokeFill(&shape); @@ -184,9 +182,13 @@ struct SwShapeTask : SwTask //Clip stroke rle if (shape.strokeRle && !clipper->clip(shape.strokeRle)) goto err; } + + bbox = renderRegion; //sync + return; err: + bbox.reset(); shapeReset(&shape); shapeDelOutline(&shape, mpool, tid); } @@ -201,20 +203,14 @@ struct SwShapeTask : SwTask struct SwImageTask : SwTask { SwImage image; - Surface* source; //Image source + RenderSurface* source; //Image source - bool clip(SwRleData* target) override + bool clip(SwRle* target) override { TVGERR("SW_ENGINE", "Image is used as ClipPath?"); return true; } - SwRleData* rle() override - { - TVGERR("SW_ENGINE", "Image is used as Scene ClipPath?"); - return nullptr; - } - void run(unsigned tid) override { auto clipRegion = bbox; @@ -452,27 +448,18 @@ bool SwRenderer::blend(BlendMethod method) surface->blendMethod = method; switch (method) { - case BlendMethod::Add: - surface->blender = opBlendAdd; - break; - case BlendMethod::Screen: - surface->blender = opBlendScreen; + case BlendMethod::Normal: + surface->blender = nullptr; break; case BlendMethod::Multiply: surface->blender = opBlendMultiply; break; + case BlendMethod::Screen: + surface->blender = opBlendScreen; + break; case BlendMethod::Overlay: surface->blender = opBlendOverlay; break; - case BlendMethod::Difference: - surface->blender = opBlendDifference; - break; - case BlendMethod::Exclusion: - surface->blender = opBlendExclusion; - break; - case BlendMethod::SrcOver: - surface->blender = opBlendSrcOver; - break; case BlendMethod::Darken: surface->blender = opBlendDarken; break; @@ -491,7 +478,17 @@ bool SwRenderer::blend(BlendMethod method) case BlendMethod::SoftLight: surface->blender = opBlendSoftLight; break; + case BlendMethod::Difference: + surface->blender = opBlendDifference; + break; + case BlendMethod::Exclusion: + surface->blender = opBlendExclusion; + break; + case BlendMethod::Add: + surface->blender = opBlendAdd; + break; default: + TVGLOG("SW_ENGINE", "Non supported blending option = %d", (int) method); surface->blender = nullptr; break; } @@ -505,7 +502,7 @@ RenderRegion SwRenderer::region(RenderData data) } -bool SwRenderer::beginComposite(Compositor* cmp, CompositeMethod method, uint8_t opacity) +bool SwRenderer::beginComposite(RenderCompositor* cmp, CompositeMethod method, uint8_t opacity) { if (!cmp) return false; auto p = static_cast(cmp); @@ -543,31 +540,19 @@ bool SwRenderer::mempool(bool shared) } -const Surface* SwRenderer::mainSurface() +const RenderSurface* SwRenderer::mainSurface() { return surface; } -Compositor* SwRenderer::target(const RenderRegion& region, ColorSpace cs) +SwSurface* SwRenderer::request(int channelSize) { - auto x = region.x; - auto y = region.y; - auto w = region.w; - auto h = region.h; - auto sw = static_cast(surface->w); - auto sh = static_cast(surface->h); - - //Out of boundary - if (x >= sw || y >= sh || x + w < 0 || y + h < 0) return nullptr; - SwSurface* cmp = nullptr; - auto reqChannelSize = CHANNEL_SIZE(cs); - //Use cached data for (auto p = compositors.begin(); p < compositors.end(); ++p) { - if ((*p)->compositor->valid && (*p)->compositor->image.channelSize == reqChannelSize) { + if ((*p)->compositor->valid && (*p)->compositor->image.channelSize == channelSize) { cmp = *p; break; } @@ -578,18 +563,48 @@ Compositor* SwRenderer::target(const RenderRegion& region, ColorSpace cs) //Inherits attributes from main surface cmp = new SwSurface(surface); cmp->compositor = new SwCompositor; - - //TODO: We can optimize compositor surface size from (surface->stride x surface->h) to Parameter(w x h) - cmp->compositor->image.data = (pixel_t*)malloc(reqChannelSize * surface->stride * surface->h); - cmp->channelSize = cmp->compositor->image.channelSize = reqChannelSize; + cmp->compositor->image.data = (pixel_t*)malloc(channelSize * surface->stride * surface->h); + cmp->compositor->image.w = surface->w; + cmp->compositor->image.h = surface->h; + cmp->compositor->image.stride = surface->stride; + cmp->compositor->image.direct = true; + cmp->compositor->valid = true; + cmp->channelSize = cmp->compositor->image.channelSize = channelSize; + cmp->w = cmp->compositor->image.w; + cmp->h = cmp->compositor->image.h; compositors.push(cmp); } + //Sync. This may have been modified by post-processing. + cmp->data = cmp->compositor->image.data; + + return cmp; +} + + +RenderCompositor* SwRenderer::target(const RenderRegion& region, ColorSpace cs) +{ + auto x = region.x; + auto y = region.y; + auto w = region.w; + auto h = region.h; + auto sw = static_cast(surface->w); + auto sh = static_cast(surface->h); + + //Out of boundary + if (x >= sw || y >= sh || x + w < 0 || y + h < 0) return nullptr; + + auto cmp = request(CHANNEL_SIZE(cs)); + //Boundary Check + if (x < 0) x = 0; + if (y < 0) y = 0; if (x + w > sw) w = (sw - x); if (y + h > sh) h = (sh - y); + if (w == 0 || h == 0) return nullptr; + cmp->compositor->recoverSfc = surface; cmp->compositor->recoverCmp = surface->compositor; cmp->compositor->valid = false; @@ -597,14 +612,6 @@ Compositor* SwRenderer::target(const RenderRegion& region, ColorSpace cs) cmp->compositor->bbox.min.y = y; cmp->compositor->bbox.max.x = x + w; cmp->compositor->bbox.max.y = y + h; - cmp->compositor->image.stride = surface->stride; - cmp->compositor->image.w = surface->w; - cmp->compositor->image.h = surface->h; - cmp->compositor->image.direct = true; - - cmp->data = cmp->compositor->image.data; - cmp->w = cmp->compositor->image.w; - cmp->h = cmp->compositor->image.h; /* TODO: Currently, only blending might work. Blending and composition must be handled together. */ @@ -618,7 +625,7 @@ Compositor* SwRenderer::target(const RenderRegion& region, ColorSpace cs) } -bool SwRenderer::endComposite(Compositor* cmp) +bool SwRenderer::endComposite(RenderCompositor* cmp) { if (!cmp) return false; @@ -639,6 +646,40 @@ bool SwRenderer::endComposite(Compositor* cmp) } +bool SwRenderer::prepare(RenderEffect* effect) +{ + switch (effect->type) { + case SceneEffect::GaussianBlur: return effectGaussianBlurPrepare(static_cast(effect)); + case SceneEffect::DropShadow: return effectDropShadowPrepare(static_cast(effect)); + default: return false; + } +} + + +bool SwRenderer::effect(RenderCompositor* cmp, const RenderEffect* effect, uint8_t opacity, bool direct) +{ + if (effect->invalid) return false; + + auto p = static_cast(cmp); + + switch (effect->type) { + case SceneEffect::GaussianBlur: { + return effectGaussianBlur(p, request(surface->channelSize), static_cast(effect)); + } + case SceneEffect::DropShadow: { + auto cmp1 = request(surface->channelSize); + cmp1->compositor->valid = false; + auto cmp2 = request(surface->channelSize); + SwSurface* surfaces[] = {cmp1, cmp2}; + auto ret = effectDropShadow(p, surfaces, static_cast(effect), opacity, direct); + cmp1->compositor->valid = true; + return ret; + } + default: return false; + } +} + + ColorSpace SwRenderer::colorSpace() { if (surface) return surface->cs; @@ -681,10 +722,10 @@ void* SwRenderer::prepareCommon(SwTask* task, const Matrix& transform, const Arr task->surface = surface; task->mpool = mpool; task->flags = flags; - task->bbox.min.x = mathMax(static_cast(0), static_cast(vport.x)); - task->bbox.min.y = mathMax(static_cast(0), static_cast(vport.y)); - task->bbox.max.x = mathMin(static_cast(surface->w), static_cast(vport.x + vport.w)); - task->bbox.max.y = mathMin(static_cast(surface->h), static_cast(vport.y + vport.h)); + task->bbox.min.x = std::max(static_cast(0), static_cast(vport.x)); + task->bbox.min.y = std::max(static_cast(0), static_cast(vport.y)); + task->bbox.max.x = std::min(static_cast(surface->w), static_cast(vport.x + vport.w)); + task->bbox.max.y = std::min(static_cast(surface->h), static_cast(vport.y + vport.h)); if (!task->pushed) { task->pushed = true; @@ -697,7 +738,7 @@ void* SwRenderer::prepareCommon(SwTask* task, const Matrix& transform, const Arr } -RenderData SwRenderer::prepare(Surface* surface, RenderData data, const Matrix& transform, Array& clips, uint8_t opacity, RenderUpdateFlag flags) +RenderData SwRenderer::prepare(RenderSurface* surface, RenderData data, const Matrix& transform, Array& clips, uint8_t opacity, RenderUpdateFlag flags) { //prepare task auto task = static_cast(data); @@ -748,6 +789,10 @@ bool SwRenderer::init(uint32_t threads) int32_t SwRenderer::init() { +#ifdef THORVG_SW_OPENMP_SUPPORT + omp_set_num_threads(TaskScheduler::threads()); +#endif + return initEngineCnt; } diff --git a/thirdparty/thorvg/src/renderer/sw_engine/tvgSwRenderer.h b/thirdparty/thorvg/src/renderer/sw_engine/tvgSwRenderer.h index fcd8ad46205f..bd6beb8d85ce 100644 --- a/thirdparty/thorvg/src/renderer/sw_engine/tvgSwRenderer.h +++ b/thirdparty/thorvg/src/renderer/sw_engine/tvgSwRenderer.h @@ -37,7 +37,7 @@ class SwRenderer : public RenderMethod { public: RenderData prepare(const RenderShape& rshape, RenderData data, const Matrix& transform, Array& clips, uint8_t opacity, RenderUpdateFlag flags, bool clipper) override; - RenderData prepare(Surface* surface, RenderData data, const Matrix& transform, Array& clips, uint8_t opacity, RenderUpdateFlag flags) override; + RenderData prepare(RenderSurface* surface, RenderData data, const Matrix& transform, Array& clips, uint8_t opacity, RenderUpdateFlag flags) override; bool preRender() override; bool renderShape(RenderData data) override; bool renderImage(RenderData data) override; @@ -48,18 +48,21 @@ class SwRenderer : public RenderMethod bool viewport(const RenderRegion& vp) override; bool blend(BlendMethod method) override; ColorSpace colorSpace() override; - const Surface* mainSurface() override; + const RenderSurface* mainSurface() override; bool clear() override; bool sync() override; bool target(pixel_t* data, uint32_t stride, uint32_t w, uint32_t h, ColorSpace cs); bool mempool(bool shared); - Compositor* target(const RenderRegion& region, ColorSpace cs) override; - bool beginComposite(Compositor* cmp, CompositeMethod method, uint8_t opacity) override; - bool endComposite(Compositor* cmp) override; + RenderCompositor* target(const RenderRegion& region, ColorSpace cs) override; + bool beginComposite(RenderCompositor* cmp, CompositeMethod method, uint8_t opacity) override; + bool endComposite(RenderCompositor* cmp) override; void clearCompositors(); + bool prepare(RenderEffect* effect) override; + bool effect(RenderCompositor* cmp, const RenderEffect* effect, uint8_t opacity, bool direct) override; + static SwRenderer* gen(); static bool init(uint32_t threads); static int32_t init(); @@ -76,6 +79,7 @@ class SwRenderer : public RenderMethod SwRenderer(); ~SwRenderer(); + SwSurface* request(int channelSize); RenderData prepareCommon(SwTask* task, const Matrix& transform, const Array& clips, uint8_t opacity, RenderUpdateFlag flags); }; diff --git a/thirdparty/thorvg/src/renderer/sw_engine/tvgSwRle.cpp b/thirdparty/thorvg/src/renderer/sw_engine/tvgSwRle.cpp index 42b08de6a581..3e4ad679a8af 100644 --- a/thirdparty/thorvg/src/renderer/sw_engine/tvgSwRle.cpp +++ b/thirdparty/thorvg/src/renderer/sw_engine/tvgSwRle.cpp @@ -217,7 +217,7 @@ struct Cell struct RleWorker { - SwRleData* rle; + SwRle* rle; SwPoint cellPos; SwPoint cellMin; @@ -235,6 +235,7 @@ struct RleWorker SwPoint pos; SwPoint bezStack[32 * 3 + 1]; + SwPoint lineStack[32 + 1]; int levStack[32]; SwOutline* outline; @@ -297,7 +298,7 @@ static inline SwCoord HYPOT(SwPoint pt) } -static void _horizLine(RleWorker& rw, SwCoord x, SwCoord y, SwCoord area, SwCoord acount) +static void _horizLine(RleWorker& rw, SwCoord x, SwCoord y, SwCoord area, SwCoord aCount) { x += rw.cellMin.x; y += rw.cellMin.y; @@ -341,11 +342,11 @@ static void _horizLine(RleWorker& rw, SwCoord x, SwCoord y, SwCoord area, SwCoor if ((span->coverage == coverage) && (span->y == y) && (span->x + span->len == x)) { //Clip x range SwCoord xOver = 0; - if (x + acount >= rw.cellMax.x) xOver -= (x + acount - rw.cellMax.x); + if (x + aCount >= rw.cellMax.x) xOver -= (x + aCount - rw.cellMax.x); if (x < rw.cellMin.x) xOver -= (rw.cellMin.x - x); - //span->len += (acount + xOver) - 1; - span->len += (acount + xOver); + //span->len += (aCount + xOver) - 1; + span->len += (aCount + xOver); return; } } @@ -361,20 +362,20 @@ static void _horizLine(RleWorker& rw, SwCoord x, SwCoord y, SwCoord area, SwCoor //Clip x range SwCoord xOver = 0; - if (x + acount >= rw.cellMax.x) xOver -= (x + acount - rw.cellMax.x); + if (x + aCount >= rw.cellMax.x) xOver -= (x + aCount - rw.cellMax.x); if (x < rw.cellMin.x) { xOver -= (rw.cellMin.x - x); x = rw.cellMin.x; } //Nothing to draw - if (acount + xOver <= 0) return; + if (aCount + xOver <= 0) return; //add a span to the current list auto span = rle->spans + rle->size; span->x = x; span->y = y; - span->len = (acount + xOver); + span->len = (aCount + xOver); span->coverage = coverage; rle->size++; } @@ -513,98 +514,116 @@ static void _lineTo(RleWorker& rw, const SwPoint& to) return; } - auto diff = to - rw.pos; - auto f1 = rw.pos - SUBPIXELS(e1); - SwPoint f2; - - //inside one cell - if (e1 == e2) { - ; - //any horizontal line - } else if (diff.y == 0) { - e1.x = e2.x; - _setCell(rw, e1); - } else if (diff.x == 0) { - //vertical line up - if (diff.y > 0) { - do { - f2.y = ONE_PIXEL; - rw.cover += (f2.y - f1.y); - rw.area += (f2.y - f1.y) * f1.x * 2; - f1.y = 0; - ++e1.y; - _setCell(rw, e1); - } while(e1.y != e2.y); - //vertical line down + auto line = rw.lineStack; + line[0] = to; + line[1] = rw.pos; + + while (true) { + auto diff = line[0] - line[1]; + auto L = HYPOT(diff); + + if (L > SHRT_MAX) { + mathSplitLine(line); + ++line; + continue; + } + e1 = TRUNC(line[1]); + e2 = TRUNC(line[0]); + + auto f1 = line[1] - SUBPIXELS(e1); + SwPoint f2; + + //inside one cell + if (e1 == e2) { + ; + //any horizontal line + } else if (diff.y == 0) { + e1.x = e2.x; + _setCell(rw, e1); + } else if (diff.x == 0) { + //vertical line up + if (diff.y > 0) { + do { + f2.y = ONE_PIXEL; + rw.cover += (f2.y - f1.y); + rw.area += (f2.y - f1.y) * f1.x * 2; + f1.y = 0; + ++e1.y; + _setCell(rw, e1); + } while(e1.y != e2.y); + //vertical line down + } else { + do { + f2.y = 0; + rw.cover += (f2.y - f1.y); + rw.area += (f2.y - f1.y) * f1.x * 2; + f1.y = ONE_PIXEL; + --e1.y; + _setCell(rw, e1); + } while(e1.y != e2.y); + } + //any other line } else { + Area prod = diff.x * f1.y - diff.y * f1.x; + + /* These macros speed up repetitive divisions by replacing them + with multiplications and right shifts. */ + auto dx_r = static_cast(ULONG_MAX >> PIXEL_BITS) / (diff.x); + auto dy_r = static_cast(ULONG_MAX >> PIXEL_BITS) / (diff.y); + + /* The fundamental value `prod' determines which side and the */ + /* exact coordinate where the line exits current cell. It is */ + /* also easily updated when moving from one cell to the next. */ do { - f2.y = 0; - rw.cover += (f2.y - f1.y); - rw.area += (f2.y - f1.y) * f1.x * 2; - f1.y = ONE_PIXEL; - --e1.y; + auto px = diff.x * ONE_PIXEL; + auto py = diff.y * ONE_PIXEL; + + //left + if (prod <= 0 && prod - px > 0) { + f2 = {0, SW_UDIV(-prod, -dx_r)}; + prod -= py; + rw.cover += (f2.y - f1.y); + rw.area += (f2.y - f1.y) * (f1.x + f2.x); + f1 = {ONE_PIXEL, f2.y}; + --e1.x; + //up + } else if (prod - px <= 0 && prod - px + py > 0) { + prod -= px; + f2 = {SW_UDIV(-prod, dy_r), ONE_PIXEL}; + rw.cover += (f2.y - f1.y); + rw.area += (f2.y - f1.y) * (f1.x + f2.x); + f1 = {f2.x, 0}; + ++e1.y; + //right + } else if (prod - px + py <= 0 && prod + py >= 0) { + prod += py; + f2 = {ONE_PIXEL, SW_UDIV(prod, dx_r)}; + rw.cover += (f2.y - f1.y); + rw.area += (f2.y - f1.y) * (f1.x + f2.x); + f1 = {0, f2.y}; + ++e1.x; + //down + } else { + f2 = {SW_UDIV(prod, -dy_r), 0}; + prod += px; + rw.cover += (f2.y - f1.y); + rw.area += (f2.y - f1.y) * (f1.x + f2.x); + f1 = {f2.x, ONE_PIXEL}; + --e1.y; + } + _setCell(rw, e1); - } while(e1.y != e2.y); + + } while(e1 != e2); } - //any other line - } else { - Area prod = diff.x * f1.y - diff.y * f1.x; - - /* These macros speed up repetitive divisions by replacing them - with multiplications and right shifts. */ - auto dx_r = static_cast(ULONG_MAX >> PIXEL_BITS) / (diff.x); - auto dy_r = static_cast(ULONG_MAX >> PIXEL_BITS) / (diff.y); - - /* The fundamental value `prod' determines which side and the */ - /* exact coordinate where the line exits current cell. It is */ - /* also easily updated when moving from one cell to the next. */ - do { - auto px = diff.x * ONE_PIXEL; - auto py = diff.y * ONE_PIXEL; - - //left - if (prod <= 0 && prod - px > 0) { - f2 = {0, SW_UDIV(-prod, -dx_r)}; - prod -= py; - rw.cover += (f2.y - f1.y); - rw.area += (f2.y - f1.y) * (f1.x + f2.x); - f1 = {ONE_PIXEL, f2.y}; - --e1.x; - //up - } else if (prod - px <= 0 && prod - px + py > 0) { - prod -= px; - f2 = {SW_UDIV(-prod, dy_r), ONE_PIXEL}; - rw.cover += (f2.y - f1.y); - rw.area += (f2.y - f1.y) * (f1.x + f2.x); - f1 = {f2.x, 0}; - ++e1.y; - //right - } else if (prod - px + py <= 0 && prod + py >= 0) { - prod += py; - f2 = {ONE_PIXEL, SW_UDIV(prod, dx_r)}; - rw.cover += (f2.y - f1.y); - rw.area += (f2.y - f1.y) * (f1.x + f2.x); - f1 = {0, f2.y}; - ++e1.x; - //down - } else { - f2 = {SW_UDIV(prod, -dy_r), 0}; - prod += px; - rw.cover += (f2.y - f1.y); - rw.area += (f2.y - f1.y) * (f1.x + f2.x); - f1 = {f2.x, ONE_PIXEL}; - --e1.y; - } - _setCell(rw, e1); + f2 = {line[0].x - SUBPIXELS(e2.x), line[0].y - SUBPIXELS(e2.y)}; + rw.cover += (f2.y - f1.y); + rw.area += (f2.y - f1.y) * (f1.x + f2.x); + rw.pos = line[0]; - } while(e1 != e2); + if (line-- == rw.lineStack) return; } - - f2 = {to.x - SUBPIXELS(e2.x), to.y - SUBPIXELS(e2.y)}; - rw.cover += (f2.y - f1.y); - rw.area += (f2.y - f1.y) * (f1.x + f2.x); - rw.pos = to; } @@ -690,31 +709,27 @@ static void _decomposeOutline(RleWorker& rw) auto start = UPSCALE(outline->pts[first]); auto pt = outline->pts.data + first; auto types = outline->types.data + first; + ++types; _moveTo(rw, UPSCALE(outline->pts[first])); while (pt < limit) { - ++pt; - ++types; - //emit a single line_to if (types[0] == SW_CURVE_TYPE_POINT) { + ++pt; + ++types; _lineTo(rw, UPSCALE(*pt)); //types cubic } else { - pt += 2; - types += 2; - - if (pt <= limit) { - _cubicTo(rw, UPSCALE(pt[-2]), UPSCALE(pt[-1]), UPSCALE(pt[0])); - continue; - } - _cubicTo(rw, UPSCALE(pt[-2]), UPSCALE(pt[-1]), start); - goto close; + pt += 3; + types += 3; + if (pt <= limit) _cubicTo(rw, UPSCALE(pt[-2]), UPSCALE(pt[-1]), UPSCALE(pt[0])); + else if (pt - 1 == limit) _cubicTo(rw, UPSCALE(pt[-2]), UPSCALE(pt[-1]), start); + else goto close; } } - _lineTo(rw, start); close: + _lineTo(rw, start); first = last + 1; } } @@ -731,7 +746,7 @@ static int _genRle(RleWorker& rw) } -static SwSpan* _intersectSpansRegion(const SwRleData *clip, const SwRleData *target, SwSpan *outSpans, uint32_t outSpansCnt) +static SwSpan* _intersectSpansRegion(const SwRle *clip, const SwRle *target, SwSpan *outSpans, uint32_t outSpansCnt) { auto out = outSpans; auto spans = target->spans; @@ -740,7 +755,7 @@ static SwSpan* _intersectSpansRegion(const SwRleData *clip, const SwRleData *tar auto clipEnd = clip->spans + clip->size; while (spans < end && clipSpans < clipEnd) { - //align y cooridnates. + //align y-coordinates. if (clipSpans->y > spans->y) { ++spans; continue; @@ -750,7 +765,7 @@ static SwSpan* _intersectSpansRegion(const SwRleData *clip, const SwRleData *tar continue; } - //Try clipping with all clip spans which have a same y coordinate. + //Try clipping with all clip spans which have a same y-coordinate. auto temp = clipSpans; while(temp < clipEnd && outSpansCnt > 0 && temp->y == clipSpans->y) { auto sx1 = spans->x; @@ -783,7 +798,7 @@ static SwSpan* _intersectSpansRegion(const SwRleData *clip, const SwRleData *tar } -static SwSpan* _intersectSpansRect(const SwBBox *bbox, const SwRleData *targetRle, SwSpan *outSpans, uint32_t outSpansCnt) +static SwSpan* _intersectSpansRect(const SwBBox *bbox, const SwRle *targetRle, SwSpan *outSpans, uint32_t outSpansCnt) { auto out = outSpans; auto spans = targetRle->spans; @@ -822,7 +837,7 @@ static SwSpan* _intersectSpansRect(const SwBBox *bbox, const SwRleData *targetRl } -void _replaceClipSpan(SwRleData *rle, SwSpan* clippedSpans, uint32_t size) +void _replaceClipSpan(SwRle *rle, SwSpan* clippedSpans, uint32_t size) { free(rle->spans); rle->spans = clippedSpans; @@ -834,7 +849,7 @@ void _replaceClipSpan(SwRleData *rle, SwSpan* clippedSpans, uint32_t size) /* External Class Implementation */ /************************************************************************/ -SwRleData* rleRender(SwRleData* rle, const SwOutline* outline, const SwBBox& renderRegion, bool antiAlias) +SwRle* rleRender(SwRle* rle, const SwOutline* outline, const SwBBox& renderRegion, bool antiAlias) { constexpr auto RENDER_POOL_SIZE = 16384L; constexpr auto BAND_SIZE = 40; @@ -862,7 +877,7 @@ SwRleData* rleRender(SwRleData* rle, const SwOutline* outline, const SwBBox& ren rw.bandShoot = 0; rw.antiAlias = antiAlias; - if (!rle) rw.rle = reinterpret_cast(calloc(1, sizeof(SwRleData))); + if (!rle) rw.rle = reinterpret_cast(calloc(1, sizeof(SwRle))); else rw.rle = rle; //Generate RLE @@ -953,12 +968,12 @@ SwRleData* rleRender(SwRleData* rle, const SwOutline* outline, const SwBBox& ren } -SwRleData* rleRender(const SwBBox* bbox) +SwRle* rleRender(const SwBBox* bbox) { auto width = static_cast(bbox->max.x - bbox->min.x); auto height = static_cast(bbox->max.y - bbox->min.y); - auto rle = static_cast(malloc(sizeof(SwRleData))); + auto rle = static_cast(malloc(sizeof(SwRle))); rle->spans = static_cast(malloc(sizeof(SwSpan) * height)); rle->size = height; rle->alloc = height; @@ -975,14 +990,14 @@ SwRleData* rleRender(const SwBBox* bbox) } -void rleReset(SwRleData* rle) +void rleReset(SwRle* rle) { if (!rle) return; rle->size = 0; } -void rleFree(SwRleData* rle) +void rleFree(SwRle* rle) { if (!rle) return; if (rle->spans) free(rle->spans); @@ -990,7 +1005,7 @@ void rleFree(SwRleData* rle) } -void rleClipPath(SwRleData *rle, const SwRleData *clip) +void rleClip(SwRle *rle, const SwRle *clip) { if (rle->size == 0 || clip->size == 0) return; auto spanCnt = rle->size > clip->size ? rle->size : clip->size; @@ -999,11 +1014,11 @@ void rleClipPath(SwRleData *rle, const SwRleData *clip) _replaceClipSpan(rle, spans, spansEnd - spans); - TVGLOG("SW_ENGINE", "Using ClipPath!"); + TVGLOG("SW_ENGINE", "Using Path Clipping!"); } -void rleClipRect(SwRleData *rle, const SwBBox* clip) +void rleClip(SwRle *rle, const SwBBox* clip) { if (rle->size == 0) return; auto spans = static_cast(malloc(sizeof(SwSpan) * (rle->size))); @@ -1011,5 +1026,5 @@ void rleClipRect(SwRleData *rle, const SwBBox* clip) _replaceClipSpan(rle, spans, spansEnd - spans); - TVGLOG("SW_ENGINE", "Using ClipRect!"); + TVGLOG("SW_ENGINE", "Using Box Clipping!"); } diff --git a/thirdparty/thorvg/src/renderer/sw_engine/tvgSwShape.cpp b/thirdparty/thorvg/src/renderer/sw_engine/tvgSwShape.cpp index 24c4a9e37255..4408db0b86f2 100644 --- a/thirdparty/thorvg/src/renderer/sw_engine/tvgSwShape.cpp +++ b/thirdparty/thorvg/src/renderer/sw_engine/tvgSwShape.cpp @@ -22,7 +22,6 @@ #include "tvgSwCommon.h" #include "tvgMath.h" -#include "tvgLines.h" /************************************************************************/ /* Internal Class Implementation */ @@ -102,9 +101,9 @@ static bool _outlineClose(SwOutline& outline) static void _dashLineTo(SwDashStroke& dash, const Point* to, const Matrix& transform) { Line cur = {dash.ptCur, *to}; - auto len = lineLength(cur.pt1, cur.pt2); + auto len = cur.length(); - if (mathZero(len)) { + if (tvg::zero(len)) { _outlineMoveTo(*dash.outline, &dash.ptCur, transform); //draw the current line fully } else if (len <= dash.curLen) { @@ -122,7 +121,7 @@ static void _dashLineTo(SwDashStroke& dash, const Point* to, const Matrix& trans Line left, right; if (dash.curLen > 0) { len -= dash.curLen; - lineSplitAt(cur, dash.curLen, left, right); + cur.split(dash.curLen, left, right); if (!dash.curOpGap) { if (dash.move || dash.pattern[dash.curIdx] - dash.curLen < FLOAT_EPSILON) { _outlineMoveTo(*dash.outline, &left.pt1, transform); @@ -163,10 +162,10 @@ static void _dashLineTo(SwDashStroke& dash, const Point* to, const Matrix& trans static void _dashCubicTo(SwDashStroke& dash, const Point* ctrl1, const Point* ctrl2, const Point* to, const Matrix& transform) { Bezier cur = {dash.ptCur, *ctrl1, *ctrl2, *to}; - auto len = bezLength(cur); + auto len = cur.length(); //draw the current line fully - if (mathZero(len)) { + if (tvg::zero(len)) { _outlineMoveTo(*dash.outline, &dash.ptCur, transform); } else if (len <= dash.curLen) { dash.curLen -= len; @@ -183,7 +182,7 @@ static void _dashCubicTo(SwDashStroke& dash, const Point* ctrl1, const Point* ct Bezier left, right; if (dash.curLen > 0) { len -= dash.curLen; - bezSplitAt(cur, dash.curLen, left, right); + cur.split(dash.curLen, left, right); if (!dash.curOpGap) { if (dash.move || dash.pattern[dash.curIdx] - dash.curLen < FLOAT_EPSILON) { _outlineMoveTo(*dash.outline, &left.start, transform); @@ -284,7 +283,7 @@ static float _outlineLength(const RenderShape* rshape, uint32_t shiftPts, uint32 if (cmdCnt <= 0 || ptsCnt <= 0) return 0.0f; const Point* close = nullptr; - auto length = 0.0f; + auto len = 0.0f; //must begin with moveTo if (cmds[0] == PathCommand::MoveTo) { @@ -297,30 +296,30 @@ static float _outlineLength(const RenderShape* rshape, uint32_t shiftPts, uint32 while (cmdCnt-- > 0) { switch (*cmds) { case PathCommand::Close: { - length += mathLength(pts - 1, close); - if (subpath) return length; + len += length(pts - 1, close); + if (subpath) return len; break; } case PathCommand::MoveTo: { - if (subpath) return length; + if (subpath) return len; close = pts; ++pts; break; } case PathCommand::LineTo: { - length += mathLength(pts - 1, pts); + len += length(pts - 1, pts); ++pts; break; } case PathCommand::CubicTo: { - length += bezLength({*(pts - 1), *pts, *(pts + 1), *(pts + 2)}); + len += Bezier{*(pts - 1), *pts, *(pts + 1), *(pts + 2)}.length(); pts += 3; break; } } ++cmds; } - return length; + return len; } @@ -355,7 +354,7 @@ static SwOutline* _genDashOutline(const RenderShape* rshape, const Matrix& trans //offset auto patternLength = 0.0f; uint32_t offIdx = 0; - if (!mathZero(offset)) { + if (!tvg::zero(offset)) { for (size_t i = 0; i < dash.cnt; ++i) patternLength += dash.pattern[i]; bool isOdd = dash.cnt % 2; if (isOdd) patternLength *= 2; @@ -499,7 +498,6 @@ bool shapePrepare(SwShape* shape, const RenderShape* rshape, const Matrix& trans if (!_genOutline(shape, rshape, transform, mpool, tid, hasComposite)) return false; if (!mathUpdateOutlineBBox(shape->outline, clipRegion, renderRegion, shape->fastTrack)) return false; - //Keep it for Rasterization Region shape->bbox = renderRegion; //Check valid region diff --git a/thirdparty/thorvg/src/renderer/sw_engine/tvgSwStroke.cpp b/thirdparty/thorvg/src/renderer/sw_engine/tvgSwStroke.cpp index 75ac96be04d4..e195f72adf2a 100644 --- a/thirdparty/thorvg/src/renderer/sw_engine/tvgSwStroke.cpp +++ b/thirdparty/thorvg/src/renderer/sw_engine/tvgSwStroke.cpp @@ -441,13 +441,23 @@ static void _cubicTo(SwStroke& stroke, const SwPoint& ctrl1, const SwPoint& ctrl //initialize with current direction angleIn = angleOut = angleMid = stroke.angleIn; - if (arc < limit && !mathSmallCubic(arc, angleIn, angleMid, angleOut)) { + auto valid = mathCubicAngle(arc, angleIn, angleMid, angleOut); + + //valid size + if (valid > 0 && arc < limit) { if (stroke.firstPt) stroke.angleIn = angleIn; mathSplitCubic(arc); arc += 3; continue; } + //ignoreable size + if (valid < 0 && arc == bezStack) { + stroke.center = to; + return; + } + + //small size if (firstArc) { firstArc = false; //process corner if necessary @@ -662,7 +672,7 @@ static void _beginSubPath(SwStroke& stroke, const SwPoint& to, bool closed) /* Determine if we need to check whether the border radius is greater than the radius of curvature of a curve, to handle this case specially. This is only required if bevel joins or butt caps may be created because - round & miter joins and round & square caps cover the nagative sector + round & miter joins and round & square caps cover the negative sector created with wide strokes. */ if ((stroke.join != StrokeJoin::Round) || (!stroke.closedSubPath && stroke.cap == StrokeCap::Butt)) stroke.handleWideStrokes = true; @@ -715,7 +725,7 @@ static void _endSubPath(SwStroke& stroke) _addCap(stroke, stroke.subPathAngle + SW_ANGLE_PI, 0); /* now end the right subpath accordingly. The left one is rewind - and deosn't need further processing */ + and doesn't need further processing */ _borderClose(right, false); } } @@ -845,31 +855,25 @@ bool strokeParseOutline(SwStroke* stroke, const SwOutline& outline) //A contour cannot start with a cubic control point if (type == SW_CURVE_TYPE_CUBIC) return false; + ++types; auto closed = outline.closed.data ? outline.closed.data[i]: false; _beginSubPath(*stroke, start, closed); while (pt < limit) { - ++pt; - ++types; - - //emit a signel line_to + //emit a single line_to if (types[0] == SW_CURVE_TYPE_POINT) { + ++pt; + ++types; _lineTo(*stroke, *pt); //types cubic } else { - if (pt + 1 > limit || types[1] != SW_CURVE_TYPE_CUBIC) return false; - - pt += 2; - types += 2; - - if (pt <= limit) { - _cubicTo(*stroke, pt[-2], pt[-1], pt[0]); - continue; - } - _cubicTo(*stroke, pt[-2], pt[-1], start); - goto close; + pt += 3; + types += 3; + if (pt <= limit) _cubicTo(*stroke, pt[-2], pt[-1], pt[0]); + else if (pt - 1 == limit) _cubicTo(*stroke, pt[-2], pt[-1], start); + else goto close; } } close: diff --git a/thirdparty/thorvg/src/renderer/tvgBinaryDesc.h b/thirdparty/thorvg/src/renderer/tvgBinaryDesc.h index 29f84eb82a6d..e40859b6db91 100644 --- a/thirdparty/thorvg/src/renderer/tvgBinaryDesc.h +++ b/thirdparty/thorvg/src/renderer/tvgBinaryDesc.h @@ -36,7 +36,7 @@ using TvgBinFlag = TvgBinByte; #define TVG_HEADER_SIZE 33 //TVG_HEADER_SIGNATURE_LENGTH + TVG_HEADER_VERSION_LENGTH + 2*SIZE(float) + TVG_HEADER_RESERVED_LENGTH + TVG_HEADER_COMPRESS_SIZE #define TVG_HEADER_SIGNATURE "ThorVG" #define TVG_HEADER_SIGNATURE_LENGTH 6 -#define TVG_HEADER_VERSION "001200" //Major 00, Minor 12, Micro 00 +#define TVG_HEADER_VERSION "001500" //Major 00, Minor 15, Micro 00 #define TVG_HEADER_VERSION_LENGTH 6 #define TVG_HEADER_RESERVED_LENGTH 1 //Storing flags for extensions #define TVG_HEADER_COMPRESS_SIZE 12 //TVG_HEADER_UNCOMPRESSED_SIZE + TVG_HEADER_COMPRESSED_SIZE + TVG_HEADER_COMPRESSED_SIZE_BITS diff --git a/thirdparty/thorvg/src/renderer/tvgCanvas.h b/thirdparty/thorvg/src/renderer/tvgCanvas.h index 199e823034f7..c5d2127f9c63 100644 --- a/thirdparty/thorvg/src/renderer/tvgCanvas.h +++ b/thirdparty/thorvg/src/renderer/tvgCanvas.h @@ -26,7 +26,7 @@ #include "tvgPaint.h" -enum Status : uint8_t {Synced = 0, Updating, Drawing, Damanged}; +enum Status : uint8_t {Synced = 0, Updating, Drawing, Damaged}; struct Canvas::Impl { @@ -42,7 +42,7 @@ struct Canvas::Impl ~Impl() { - //make it sure any deffered jobs + //make it sure any deferred jobs renderer->sync(); renderer->clear(); @@ -61,7 +61,7 @@ struct Canvas::Impl Result push(unique_ptr paint) { - //You can not push paints during rendering. + //You cannot push paints during rendering. if (status == Status::Drawing) return Result::InsufficientCondition; auto p = paint.release(); @@ -91,7 +91,7 @@ struct Canvas::Impl Array clips; auto flag = RenderUpdateFlag::None; - if (status == Status::Damanged || force) flag = RenderUpdateFlag::All; + if (status == Status::Damaged || force) flag = RenderUpdateFlag::All; auto m = Matrix{1, 0, 0, 0, 1, 0, 0, 0, 1}; @@ -108,7 +108,7 @@ struct Canvas::Impl Result draw() { - if (status == Status::Damanged) update(nullptr, false); + if (status == Status::Damaged) update(nullptr, false); if (status == Status::Drawing || paints.empty() || !renderer->preRender()) return Result::InsufficientCondition; bool rendered = false; @@ -124,7 +124,7 @@ struct Canvas::Impl Result sync() { - if (status == Status::Synced || status == Status::Damanged) return Result::InsufficientCondition; + if (status == Status::Synced || status == Status::Damaged) return Result::InsufficientCondition; if (renderer->sync()) { status = Status::Synced; @@ -136,7 +136,7 @@ struct Canvas::Impl Result viewport(int32_t x, int32_t y, int32_t w, int32_t h) { - if (status != Status::Damanged && status != Status::Synced) return Result::InsufficientCondition; + if (status != Status::Damaged && status != Status::Synced) return Result::InsufficientCondition; RenderRegion val = {x, y, w, h}; //intersect if the target buffer is already set. @@ -147,7 +147,7 @@ struct Canvas::Impl if (vport == val) return Result::Success; renderer->viewport(val); vport = val; - status = Status::Damanged; + status = Status::Damaged; return Result::Success; } }; diff --git a/thirdparty/thorvg/src/renderer/tvgCommon.h b/thirdparty/thorvg/src/renderer/tvgCommon.h index 15a2cc4ef045..527221625b66 100644 --- a/thirdparty/thorvg/src/renderer/tvgCommon.h +++ b/thirdparty/thorvg/src/renderer/tvgCommon.h @@ -54,15 +54,6 @@ using namespace tvg; #define strdup _strdup #endif -//TVG class identifier values -#define TVG_CLASS_ID_UNDEFINED 0 -#define TVG_CLASS_ID_SHAPE 1 -#define TVG_CLASS_ID_SCENE 2 -#define TVG_CLASS_ID_PICTURE 3 -#define TVG_CLASS_ID_LINEAR 4 -#define TVG_CLASS_ID_RADIAL 5 -#define TVG_CLASS_ID_TEXT 6 - enum class FileType { Png = 0, Jpg, Webp, Tvg, Svg, Lottie, Ttf, Raw, Gif, Unknown }; using Size = Point; diff --git a/thirdparty/thorvg/src/renderer/tvgFill.cpp b/thirdparty/thorvg/src/renderer/tvgFill.cpp index ea1010051ee5..19edff5a2cd6 100644 --- a/thirdparty/thorvg/src/renderer/tvgFill.cpp +++ b/thirdparty/thorvg/src/renderer/tvgFill.cpp @@ -155,15 +155,14 @@ Fill* Fill::duplicate() const noexcept } -uint32_t Fill::identifier() const noexcept +TVG_DEPRECATED uint32_t Fill::identifier() const noexcept { - return pImpl->id; + return (uint32_t) type(); } RadialGradient::RadialGradient():pImpl(new Impl()) { - Fill::pImpl->id = TVG_CLASS_ID_RADIAL; Fill::pImpl->method(new FillDup(pImpl)); } @@ -196,15 +195,20 @@ unique_ptr RadialGradient::gen() noexcept } -uint32_t RadialGradient::identifier() noexcept +TVG_DEPRECATED uint32_t RadialGradient::identifier() noexcept { - return TVG_CLASS_ID_RADIAL; + return (uint32_t) Type::RadialGradient; +} + + +Type RadialGradient::type() const noexcept +{ + return Type::RadialGradient; } LinearGradient::LinearGradient():pImpl(new Impl()) { - Fill::pImpl->id = TVG_CLASS_ID_LINEAR; Fill::pImpl->method(new FillDup(pImpl)); } @@ -243,8 +247,13 @@ unique_ptr LinearGradient::gen() noexcept } -uint32_t LinearGradient::identifier() noexcept +TVG_DEPRECATED uint32_t LinearGradient::identifier() noexcept { - return TVG_CLASS_ID_LINEAR; + return (uint32_t) Type::LinearGradient; } + +Type LinearGradient::type() const noexcept +{ + return Type::LinearGradient; +} diff --git a/thirdparty/thorvg/src/renderer/tvgFill.h b/thirdparty/thorvg/src/renderer/tvgFill.h index 47f0c051c025..f249356aa2a7 100644 --- a/thirdparty/thorvg/src/renderer/tvgFill.h +++ b/thirdparty/thorvg/src/renderer/tvgFill.h @@ -55,7 +55,6 @@ struct Fill::Impl uint32_t cnt = 0; FillSpread spread; DuplicateMethod* dup = nullptr; - uint8_t id; ~Impl() { diff --git a/thirdparty/thorvg/src/renderer/tvgGlCanvas.cpp b/thirdparty/thorvg/src/renderer/tvgGlCanvas.cpp index 82666b7ae31f..24e2fb8b1beb 100644 --- a/thirdparty/thorvg/src/renderer/tvgGlCanvas.cpp +++ b/thirdparty/thorvg/src/renderer/tvgGlCanvas.cpp @@ -62,7 +62,7 @@ GlCanvas::~GlCanvas() Result GlCanvas::target(int32_t id, uint32_t w, uint32_t h) noexcept { #ifdef THORVG_GL_RASTER_SUPPORT - if (Canvas::pImpl->status != Status::Damanged && Canvas::pImpl->status != Status::Synced) { + if (Canvas::pImpl->status != Status::Damaged && Canvas::pImpl->status != Status::Synced) { return Result::InsufficientCondition; } @@ -75,7 +75,7 @@ Result GlCanvas::target(int32_t id, uint32_t w, uint32_t h) noexcept renderer->viewport(Canvas::pImpl->vport); //Paints must be updated again with this new target. - Canvas::pImpl->status = Status::Damanged; + Canvas::pImpl->status = Status::Damaged; return Result::Success; #endif diff --git a/thirdparty/thorvg/src/renderer/tvgLoadModule.h b/thirdparty/thorvg/src/renderer/tvgLoadModule.h index 1b81d81a4f70..a9c1a6854428 100644 --- a/thirdparty/thorvg/src/renderer/tvgLoadModule.h +++ b/thirdparty/thorvg/src/renderer/tvgLoadModule.h @@ -80,14 +80,14 @@ struct ImageLoader : LoadModule static ColorSpace cs; //desired value float w = 0, h = 0; //default image size - Surface surface; + RenderSurface surface; ImageLoader(FileType type) : LoadModule(type) {} virtual bool animatable() { return false; } //true if this loader supports animation. virtual Paint* paint() { return nullptr; } - virtual Surface* bitmap() + virtual RenderSurface* bitmap() { if (surface.data) return &surface; return nullptr; diff --git a/thirdparty/thorvg/src/renderer/tvgLoader.cpp b/thirdparty/thorvg/src/renderer/tvgLoader.cpp index 6a81ddcdbb31..db51fc215a0b 100644 --- a/thirdparty/thorvg/src/renderer/tvgLoader.cpp +++ b/thirdparty/thorvg/src/renderer/tvgLoader.cpp @@ -294,10 +294,10 @@ LoadModule* LoaderMgr::loader(const string& path, bool* invalid) { *invalid = false; - //TODO: lottie is not sharable. + //TODO: svg & lottie is not sharable. auto allowCache = true; auto ext = path.substr(path.find_last_of(".") + 1); - if (!ext.compare("json")) allowCache = false; + if (!ext.compare("svg") || !ext.compare("json")) allowCache = false; if (allowCache) { if (auto loader = _findFromCache(path)) return loader; @@ -317,7 +317,7 @@ LoadModule* LoaderMgr::loader(const string& path, bool* invalid) } delete(loader); } - //Unkown MimeType. Try with the candidates in the order + //Unknown MimeType. Try with the candidates in the order for (int i = 0; i < static_cast(FileType::Raw); i++) { if (auto loader = _find(static_cast(i))) { if (loader->open(path)) { @@ -392,7 +392,7 @@ LoadModule* LoaderMgr::loader(const char* data, uint32_t size, const string& mim } } } - //Unkown MimeType. Try with the candidates in the order + //Unknown MimeType. Try with the candidates in the order for (int i = 0; i < static_cast(FileType::Raw); i++) { auto loader = _find(static_cast(i)); if (loader) { diff --git a/thirdparty/thorvg/src/renderer/tvgPaint.cpp b/thirdparty/thorvg/src/renderer/tvgPaint.cpp index 37813b19ef7e..536e18785211 100644 --- a/thirdparty/thorvg/src/renderer/tvgPaint.cpp +++ b/thirdparty/thorvg/src/renderer/tvgPaint.cpp @@ -32,11 +32,11 @@ /************************************************************************/ #define PAINT_METHOD(ret, METHOD) \ - switch (id) { \ - case TVG_CLASS_ID_SHAPE: ret = P((Shape*)paint)->METHOD; break; \ - case TVG_CLASS_ID_SCENE: ret = P((Scene*)paint)->METHOD; break; \ - case TVG_CLASS_ID_PICTURE: ret = P((Picture*)paint)->METHOD; break; \ - case TVG_CLASS_ID_TEXT: ret = P((Text*)paint)->METHOD; break; \ + switch (paint->type()) { \ + case Type::Shape: ret = P((Shape*)paint)->METHOD; break; \ + case Type::Scene: ret = P((Scene*)paint)->METHOD; break; \ + case Type::Picture: ret = P((Picture*)paint)->METHOD; break; \ + case Type::Text: ret = P((Text*)paint)->METHOD; break; \ default: ret = {}; \ } @@ -91,8 +91,8 @@ static Result _compFastTrack(RenderMethod* renderer, Paint* cmpTarget, const Mat //No rotation and no skewing, still can try out clipping the rect region. auto tryClip = false; - if ((!mathRightAngle(pm) || mathSkewed(pm))) tryClip = true; - if ((!mathRightAngle(rm) || mathSkewed(rm))) tryClip = true; + if ((!rightAngle(pm) || skewed(pm))) tryClip = true; + if ((!rightAngle(rm) || skewed(rm))) tryClip = true; if (tryClip) return _clipRect(renderer, pts, pm, rm, before); @@ -102,8 +102,8 @@ static Result _compFastTrack(RenderMethod* renderer, Paint* cmpTarget, const Mat auto pt3 = pts + 2; auto pt4 = pts + 3; - if ((mathEqual(pt1->x, pt2->x) && mathEqual(pt2->y, pt3->y) && mathEqual(pt3->x, pt4->x) && mathEqual(pt1->y, pt4->y)) || - (mathEqual(pt2->x, pt3->x) && mathEqual(pt1->y, pt2->y) && mathEqual(pt1->x, pt4->x) && mathEqual(pt3->y, pt4->y))) { + if ((tvg::equal(pt1->x, pt2->x) && tvg::equal(pt2->y, pt3->y) && tvg::equal(pt3->x, pt4->x) && tvg::equal(pt1->y, pt4->y)) || + (tvg::equal(pt2->x, pt3->x) && tvg::equal(pt1->y, pt2->y) && tvg::equal(pt1->x, pt4->x) && tvg::equal(pt3->y, pt4->y))) { RenderRegion after; @@ -164,6 +164,7 @@ Paint* Paint::Impl::duplicate(Paint* ret) ret->pImpl->opacity = opacity; if (compData) ret->pImpl->composite(ret, compData->target->duplicate(), compData->method); + if (clipper) ret->pImpl->clip(clipper->duplicate()); return ret; } @@ -172,7 +173,7 @@ Paint* Paint::Impl::duplicate(Paint* ret) bool Paint::Impl::rotate(float degree) { if (tr.overriding) return false; - if (mathEqual(degree, tr.degree)) return true; + if (tvg::equal(degree, tr.degree)) return true; tr.degree = degree; renderFlag |= RenderUpdateFlag::Transform; @@ -183,7 +184,7 @@ bool Paint::Impl::rotate(float degree) bool Paint::Impl::scale(float factor) { if (tr.overriding) return false; - if (mathEqual(factor, tr.scale)) return true; + if (tvg::equal(factor, tr.scale)) return true; tr.scale = factor; renderFlag |= RenderUpdateFlag::Transform; @@ -194,7 +195,7 @@ bool Paint::Impl::scale(float factor) bool Paint::Impl::translate(float x, float y) { if (tr.overriding) return false; - if (mathEqual(x, tr.m.e13) && mathEqual(y, tr.m.e23)) return true; + if (tvg::equal(x, tr.m.e13) && tvg::equal(y, tr.m.e23)) return true; tr.m.e13 = x; tr.m.e23 = y; renderFlag |= RenderUpdateFlag::Transform; @@ -207,11 +208,9 @@ bool Paint::Impl::render(RenderMethod* renderer) { if (opacity == 0) return true; - Compositor* cmp = nullptr; + RenderCompositor* cmp = nullptr; - /* Note: only ClipPath is processed in update() step. - Create a composition image. */ - if (compData && compData->method != CompositeMethod::ClipPath && !(compData->target->pImpl->ctxFlag & ContextFlag::FastTrack)) { + if (compData && !(compData->target->pImpl->ctxFlag & ContextFlag::FastTrack)) { RenderRegion region; PAINT_METHOD(region, bounds(renderer)); @@ -248,43 +247,49 @@ RenderData Paint::Impl::update(RenderMethod* renderer, const Matrix& pm, Arraytarget; auto method = compData->method; P(target)->ctxFlag &= ~ContextFlag::FastTrack; //reset - /* If the transformation has no rotational factors and the ClipPath/Alpha(InvAlpha)Masking involves a simple rectangle, - we can optimize by using the viewport instead of the regular ClipPath/AlphaMasking sequence for improved performance. */ - auto tryFastTrack = false; - if (target->identifier() == TVG_CLASS_ID_SHAPE) { - if (method == CompositeMethod::ClipPath) tryFastTrack = true; - else { - auto shape = static_cast(target); - uint8_t a; - shape->fillColor(nullptr, nullptr, nullptr, &a); - //no gradient fill & no compositions of the composition target. - if (!shape->fill() && !(PP(shape)->compData)) { - if (method == CompositeMethod::AlphaMask && a == 255 && PP(shape)->opacity == 255) tryFastTrack = true; - else if (method == CompositeMethod::InvAlphaMask && (a == 0 || PP(shape)->opacity == 0)) tryFastTrack = true; - } - } - if (tryFastTrack) { - viewport = renderer->viewport(); - if ((compFastTrack = _compFastTrack(renderer, target, pm, viewport)) == Result::Success) { - P(target)->ctxFlag |= ContextFlag::FastTrack; + /* If the transformation has no rotational factors and the Alpha(InvAlpha)Masking involves a simple rectangle, + we can optimize by using the viewport instead of the regular AlphaMasking sequence for improved performance. */ + if (target->type() == Type::Shape) { + auto shape = static_cast(target); + uint8_t a; + shape->fillColor(nullptr, nullptr, nullptr, &a); + //no gradient fill & no compositions of the composition target. + if (!shape->fill() && !(PP(shape)->compData)) { + if ((method == CompositeMethod::AlphaMask && a == 255 && PP(shape)->opacity == 255) || (method == CompositeMethod::InvAlphaMask && (a == 0 || PP(shape)->opacity == 0))) { + viewport = renderer->viewport(); + if ((compFastTrack = _compFastTrack(renderer, target, pm, viewport)) == Result::Success) { + P(target)->ctxFlag |= ContextFlag::FastTrack; + } } } } if (compFastTrack == Result::InsufficientCondition) { - childClipper = compData->method == CompositeMethod::ClipPath ? true : false; - trd = P(target)->update(renderer, pm, clips, 255, pFlag, childClipper); - if (childClipper) clips.push(trd); + trd = P(target)->update(renderer, pm, clips, 255, pFlag, false); + } + } + + /* 2. Clipping */ + if (this->clipper) { + P(this->clipper)->ctxFlag &= ~ContextFlag::FastTrack; //reset + viewport = renderer->viewport(); + /* TODO: Intersect the clipper's clipper, if both are FastTrack. + Update the subsequent clipper first and check its ctxFlag. */ + if (!P(this->clipper)->clipper && (compFastTrack = _compFastTrack(renderer, this->clipper, pm, viewport)) == Result::Success) { + P(this->clipper)->ctxFlag |= ContextFlag::FastTrack; + } + if (compFastTrack == Result::InsufficientCondition) { + trd = P(this->clipper)->update(renderer, pm, clips, 255, pFlag, true); + clips.push(trd); } } - /* 2. Main Update */ + /* 3. Main Update */ auto newFlag = static_cast(pFlag | renderFlag); renderFlag = RenderUpdateFlag::None; opacity = MULTIPLY(opacity, this->opacity); @@ -294,9 +299,9 @@ RenderData Paint::Impl::update(RenderMethod* renderer, const Matrix& pm, Arrayviewport(viewport); - else if (childClipper) clips.pop(); + else if (this->clipper) clips.pop(); return rd; } @@ -308,7 +313,7 @@ bool Paint::Impl::bounds(float* x, float* y, float* w, float* h, bool transforme const auto& m = this->transform(origin); //Case: No transformed, quick return! - if (!transformed || mathIdentity(&m)) { + if (!transformed || identity(&m)) { PAINT_METHOD(ret, bounds(x, y, w, h, stroking)); return ret; } @@ -351,12 +356,18 @@ bool Paint::Impl::bounds(float* x, float* y, float* w, float* h, bool transforme void Paint::Impl::reset() { + if (clipper) { + delete(clipper); + clipper = nullptr; + } + if (compData) { if (P(compData->target)->unref() == 0) delete(compData->target); free(compData); compData = nullptr; } - mathIdentity(&tr.m); + + tvg::identity(&tr.m); tr.degree = 0.0f; tr.scale = 1.0f; tr.overriding = false; @@ -437,15 +448,27 @@ Paint* Paint::duplicate() const noexcept } -Result Paint::composite(std::unique_ptr target, CompositeMethod method) noexcept +Result Paint::clip(std::unique_ptr clipper) noexcept { - if (method == CompositeMethod::ClipPath && target && target->identifier() != TVG_CLASS_ID_SHAPE) { - TVGERR("RENDERER", "ClipPath only allows the Shape!"); + auto p = clipper.release(); + + if (p && p->type() != Type::Shape) { + TVGERR("RENDERER", "Clipping only supports the Shape!"); return Result::NonSupport; } + pImpl->clip(p); + return Result::Success; +} + + +Result Paint::composite(std::unique_ptr target, CompositeMethod method) noexcept +{ + //TODO: remove. Keep this for the backward compatibility + if (target && method == CompositeMethod::ClipPath) return clip(std::move(target)); auto p = target.release(); if (pImpl->composite(this, p, method)) return Result::Success; + delete(p); return Result::InvalidArguments; } @@ -457,6 +480,11 @@ CompositeMethod Paint::composite(const Paint** target) const noexcept if (target) *target = pImpl->compData->target; return pImpl->compData->method; } else { + //TODO: remove. Keep this for the backward compatibility + if (pImpl->clipper) { + if (target) *target = pImpl->clipper; + return CompositeMethod::ClipPath; + } if (target) *target = nullptr; return CompositeMethod::None; } @@ -480,14 +508,17 @@ uint8_t Paint::opacity() const noexcept } -uint32_t Paint::identifier() const noexcept +TVG_DEPRECATED uint32_t Paint::identifier() const noexcept { - return pImpl->id; + return (uint32_t) type(); } Result Paint::blend(BlendMethod method) noexcept { + //TODO: Remove later + if (method == BlendMethod::Hue || method == BlendMethod::Saturation || method == BlendMethod::Color || method == BlendMethod::Luminosity || method == BlendMethod::HardMix) return Result::NonSupport; + if (pImpl->blendMethod != method) { pImpl->blendMethod = method; pImpl->renderFlag |= RenderUpdateFlag::Blend; @@ -495,9 +526,3 @@ Result Paint::blend(BlendMethod method) noexcept return Result::Success; } - - -BlendMethod Paint::blend() const noexcept -{ - return pImpl->blendMethod; -} diff --git a/thirdparty/thorvg/src/renderer/tvgPaint.h b/thirdparty/thorvg/src/renderer/tvgPaint.h index e43ca239bb7a..d78e9bb3d1c2 100644 --- a/thirdparty/thorvg/src/renderer/tvgPaint.h +++ b/thirdparty/thorvg/src/renderer/tvgPaint.h @@ -49,6 +49,7 @@ namespace tvg { Paint* paint = nullptr; Composite* compData = nullptr; + Paint* clipper = nullptr; RenderMethod* renderer = nullptr; struct { Matrix m; //input matrix @@ -67,8 +68,8 @@ namespace tvg m.e31 = 0.0f; m.e32 = 0.0f; m.e33 = 1.0f; - mathScale(&m, scale, scale); - mathRotate(&m, degree); + tvg::scale(&m, scale, scale); + tvg::rotate(&m, degree); } } tr; BlendMethod blendMethod; @@ -76,7 +77,6 @@ namespace tvg uint8_t ctxFlag; uint8_t opacity; uint8_t refCnt = 0; //reference count - uint8_t id; //TODO: deprecated, remove it Impl(Paint* pnt) : paint(pnt) { @@ -89,6 +89,7 @@ namespace tvg if (P(compData->target)->unref() == 0) delete(compData->target); free(compData); } + if (clipper && P(clipper)->unref() == 0) delete(clipper); if (renderer && (renderer->unref() == 0)) delete(renderer); } @@ -106,7 +107,7 @@ namespace tvg bool transform(const Matrix& m) { - tr.m = m; + if (&tr.m != &m) tr.m = m; tr.overriding = true; renderFlag |= RenderUpdateFlag::Transform; @@ -121,6 +122,20 @@ namespace tvg return tr.m; } + void clip(Paint* clp) + { + if (this->clipper) { + P(this->clipper)->unref(); + if (this->clipper != clp && P(this->clipper)->refCnt == 0) { + delete(this->clipper); + } + } + this->clipper = clp; + if (!clp) return; + + P(clipper)->ref(); + } + bool composite(Paint* source, Paint* target, CompositeMethod method) { //Invalid case diff --git a/thirdparty/thorvg/src/renderer/tvgPicture.cpp b/thirdparty/thorvg/src/renderer/tvgPicture.cpp index 223cc5b026f2..d3e31d198a3c 100644 --- a/thirdparty/thorvg/src/renderer/tvgPicture.cpp +++ b/thirdparty/thorvg/src/renderer/tvgPicture.cpp @@ -20,6 +20,7 @@ * SOFTWARE. */ +#include "tvgPaint.h" #include "tvgPicture.h" /************************************************************************/ @@ -73,11 +74,11 @@ bool Picture::Impl::needComposition(uint8_t opacity) bool Picture::Impl::render(RenderMethod* renderer) { bool ret = false; - renderer->blend(picture->blend()); + renderer->blend(PP(picture)->blendMethod); if (surface) return renderer->renderImage(rd); else if (paint) { - Compositor* cmp = nullptr; + RenderCompositor* cmp = nullptr; if (needComp) { cmp = renderer->target(bounds(renderer), renderer->colorSpace()); renderer->beginComposite(cmp, CompositeMethod::None, 255); @@ -134,7 +135,6 @@ Result Picture::Impl::load(ImageLoader* loader) Picture::Picture() : pImpl(new Impl(this)) { - Paint::pImpl->id = TVG_CLASS_ID_PICTURE; } @@ -150,9 +150,15 @@ unique_ptr Picture::gen() noexcept } -uint32_t Picture::identifier() noexcept +TVG_DEPRECATED uint32_t Picture::identifier() noexcept { - return TVG_CLASS_ID_PICTURE; + return (uint32_t) Type::Picture; +} + + +Type Picture::type() const noexcept +{ + return Type::Picture; } diff --git a/thirdparty/thorvg/src/renderer/tvgPicture.h b/thirdparty/thorvg/src/renderer/tvgPicture.h index 3a4880cabaa3..bbbc43910595 100644 --- a/thirdparty/thorvg/src/renderer/tvgPicture.h +++ b/thirdparty/thorvg/src/renderer/tvgPicture.h @@ -60,7 +60,7 @@ struct Picture::Impl ImageLoader* loader = nullptr; Paint* paint = nullptr; //vector picture uses - Surface* surface = nullptr; //bitmap picture uses + RenderSurface* surface = nullptr; //bitmap picture uses RenderData rd = nullptr; //engine data float w = 0, h = 0; Picture* picture = nullptr; diff --git a/thirdparty/thorvg/src/renderer/tvgRender.h b/thirdparty/thorvg/src/renderer/tvgRender.h index b0ee42db8d0d..eae44a2e8ad9 100644 --- a/thirdparty/thorvg/src/renderer/tvgRender.h +++ b/thirdparty/thorvg/src/renderer/tvgRender.h @@ -24,6 +24,7 @@ #define _TVG_RENDER_H_ #include +#include #include "tvgCommon.h" #include "tvgArray.h" #include "tvgLock.h" @@ -36,9 +37,8 @@ using pixel_t = uint32_t; enum RenderUpdateFlag : uint8_t {None = 0, Path = 1, Color = 2, Gradient = 4, Stroke = 8, Transform = 16, Image = 32, GradientStroke = 64, Blend = 128, All = 255}; -struct Surface; - -enum ColorSpace +//TODO: Move this in public header unifying with SwCanvas::Colorspace +enum ColorSpace : uint8_t { ABGR8888 = 0, //The channels are joined in the order: alpha, blue, green, red. Colors are alpha-premultiplied. ARGB8888, //The channels are joined in the order: alpha, red, green, blue. Colors are alpha-premultiplied. @@ -48,7 +48,7 @@ enum ColorSpace Unsupported //TODO: Change to the default, At the moment, we put it in the last to align with SwCanvas::Colorspace. }; -struct Surface +struct RenderSurface { union { pixel_t* data = nullptr; //system based data pointer @@ -62,11 +62,11 @@ struct Surface uint8_t channelSize = 0; bool premultiplied = false; //Alpha-premultiplied - Surface() + RenderSurface() { } - Surface(const Surface* rhs) + RenderSurface(const RenderSurface* rhs) { data = rhs->data; stride = rhs->stride; @@ -80,21 +80,10 @@ struct Surface }; -struct Compositor +struct RenderCompositor { CompositeMethod method; - uint8_t opacity; -}; - -struct Vertex -{ - Point pt; - Point uv; -}; - -struct Polygon -{ - Vertex vertex[3]; + uint8_t opacity; }; struct RenderRegion @@ -270,11 +259,66 @@ struct RenderShape float strokeMiterlimit() const { if (!stroke) return 4.0f; - return stroke->miterlimit;; } }; +struct RenderEffect +{ + RenderData rd = nullptr; + RenderRegion extend = {0, 0, 0, 0}; + SceneEffect type; + bool invalid = false; + + virtual ~RenderEffect() + { + free(rd); + } +}; + +struct RenderEffectGaussianBlur : RenderEffect +{ + float sigma; + uint8_t direction; //0: both, 1: horizontal, 2: vertical + uint8_t border; //0: duplicate, 1: wrap + uint8_t quality; //0 ~ 100 (optional) + + static RenderEffectGaussianBlur* gen(va_list& args) + { + auto inst = new RenderEffectGaussianBlur; + inst->sigma = std::max((float) va_arg(args, double), 0.0f); + inst->direction = std::min(va_arg(args, int), 2); + inst->border = std::min(va_arg(args, int), 1); + inst->quality = std::min(va_arg(args, int), 100); + inst->type = SceneEffect::GaussianBlur; + return inst; + } +}; + +struct RenderEffectDropShadow : RenderEffect +{ + uint8_t color[4]; //rgba + float angle; + float distance; + float sigma; + uint8_t quality; //0 ~ 100 (optional) + + static RenderEffectDropShadow* gen(va_list& args) + { + auto inst = new RenderEffectDropShadow; + inst->color[0] = va_arg(args, int); + inst->color[1] = va_arg(args, int); + inst->color[2] = va_arg(args, int); + inst->color[3] = std::min(va_arg(args, int), 255); + inst->angle = (float) va_arg(args, double); + inst->distance = (float) va_arg(args, double); + inst->sigma = std::max((float) va_arg(args, double), 0.0f); + inst->quality = std::min(va_arg(args, int), 100); + inst->type = SceneEffect::DropShadow; + return inst; + } +}; + class RenderMethod { private: @@ -287,7 +331,7 @@ class RenderMethod virtual ~RenderMethod() {} virtual RenderData prepare(const RenderShape& rshape, RenderData data, const Matrix& transform, Array& clips, uint8_t opacity, RenderUpdateFlag flags, bool clipper) = 0; - virtual RenderData prepare(Surface* surface, RenderData data, const Matrix& transform, Array& clips, uint8_t opacity, RenderUpdateFlag flags) = 0; + virtual RenderData prepare(RenderSurface* surface, RenderData data, const Matrix& transform, Array& clips, uint8_t opacity, RenderUpdateFlag flags) = 0; virtual bool preRender() = 0; virtual bool renderShape(RenderData data) = 0; virtual bool renderImage(RenderData data) = 0; @@ -298,14 +342,17 @@ class RenderMethod virtual bool viewport(const RenderRegion& vp) = 0; virtual bool blend(BlendMethod method) = 0; virtual ColorSpace colorSpace() = 0; - virtual const Surface* mainSurface() = 0; + virtual const RenderSurface* mainSurface() = 0; virtual bool clear() = 0; virtual bool sync() = 0; - virtual Compositor* target(const RenderRegion& region, ColorSpace cs) = 0; - virtual bool beginComposite(Compositor* cmp, CompositeMethod method, uint8_t opacity) = 0; - virtual bool endComposite(Compositor* cmp) = 0; + virtual RenderCompositor* target(const RenderRegion& region, ColorSpace cs) = 0; + virtual bool beginComposite(RenderCompositor* cmp, CompositeMethod method, uint8_t opacity) = 0; + virtual bool endComposite(RenderCompositor* cmp) = 0; + + virtual bool prepare(RenderEffect* effect) = 0; + virtual bool effect(RenderCompositor* cmp, const RenderEffect* effect, uint8_t opacity, bool direct) = 0; }; static inline bool MASK_REGION_MERGING(CompositeMethod method) @@ -374,7 +421,6 @@ static inline uint8_t MULTIPLY(uint8_t c, uint8_t a) return (((c) * (a) + 0xff) >> 8); } - } #endif //_TVG_RENDER_H_ diff --git a/thirdparty/thorvg/src/renderer/tvgSaver.cpp b/thirdparty/thorvg/src/renderer/tvgSaver.cpp index 79302f69fa56..993fe6d80f8d 100644 --- a/thirdparty/thorvg/src/renderer/tvgSaver.cpp +++ b/thirdparty/thorvg/src/renderer/tvgSaver.cpp @@ -20,6 +20,7 @@ * SOFTWARE. */ +#include #include "tvgCommon.h" #include "tvgSaveModule.h" #include "tvgPaint.h" @@ -122,7 +123,7 @@ Result Saver::save(std::unique_ptr paint, const string& path, bool compre auto p = paint.release(); if (!p) return Result::MemoryCorruption; - //Already on saving an other resource. + //Already on saving another resource. if (pImpl->saveModule) { if (P(p)->refCnt == 0) delete(p); return Result::InsufficientCondition; @@ -160,12 +161,12 @@ Result Saver::save(unique_ptr animation, const string& path, uint32_t //animation holds the picture, it must be 1 at the bottom. auto remove = PP(a->picture())->refCnt <= 1 ? true : false; - if (mathZero(a->totalFrame())) { + if (tvg::zero(a->totalFrame())) { if (remove) delete(a); return Result::InsufficientCondition; } - //Already on saving an other resource. + //Already on saving another resource. if (pImpl->saveModule) { if (remove) delete(a); return Result::InsufficientCondition; diff --git a/thirdparty/thorvg/src/renderer/tvgScene.cpp b/thirdparty/thorvg/src/renderer/tvgScene.cpp index f5809cf93b56..ce169d33ba67 100644 --- a/thirdparty/thorvg/src/renderer/tvgScene.cpp +++ b/thirdparty/thorvg/src/renderer/tvgScene.cpp @@ -20,15 +20,32 @@ * SOFTWARE. */ +#include #include "tvgScene.h" +/************************************************************************/ +/* Internal Class Implementation */ +/************************************************************************/ + +Result Scene::Impl::resetEffects() +{ + if (effects) { + for (auto e = effects->begin(); e < effects->end(); ++e) { + delete(*e); + } + delete(effects); + effects = nullptr; + } + return Result::Success; +} + + /************************************************************************/ /* External Class Implementation */ /************************************************************************/ Scene::Scene() : pImpl(new Impl(this)) { - Paint::pImpl->id = TVG_CLASS_ID_SCENE; } @@ -44,9 +61,15 @@ unique_ptr Scene::gen() noexcept } -uint32_t Scene::identifier() noexcept +TVG_DEPRECATED uint32_t Scene::identifier() noexcept +{ + return (uint32_t) Type::Scene; +} + + +Type Scene::type() const noexcept { - return TVG_CLASS_ID_SCENE; + return Type::Scene; } @@ -54,7 +77,11 @@ Result Scene::push(unique_ptr paint) noexcept { auto p = paint.release(); if (!p) return Result::MemoryCorruption; - PP(p)->ref(); + P(p)->ref(); + + //Relocated the paint to the current scene space + P(p)->renderFlag |= RenderUpdateFlag::Transform; + pImpl->paints.push_back(p); return Result::Success; @@ -79,3 +106,34 @@ list& Scene::paints() noexcept { return pImpl->paints; } + + +Result Scene::push(SceneEffect effect, ...) noexcept +{ + if (effect == SceneEffect::ClearAll) return pImpl->resetEffects(); + + if (!pImpl->effects) pImpl->effects = new Array; + + va_list args; + va_start(args, effect); + + RenderEffect* re = nullptr; + + switch (effect) { + case SceneEffect::GaussianBlur: { + re = RenderEffectGaussianBlur::gen(args); + break; + } + case SceneEffect::DropShadow: { + re = RenderEffectDropShadow::gen(args); + break; + } + default: break; + } + + if (!re) return Result::InvalidArguments; + + pImpl->effects->push(re); + + return Result::Success; +} diff --git a/thirdparty/thorvg/src/renderer/tvgScene.h b/thirdparty/thorvg/src/renderer/tvgScene.h index 190ecd31b91d..7972ae33fb10 100644 --- a/thirdparty/thorvg/src/renderer/tvgScene.h +++ b/thirdparty/thorvg/src/renderer/tvgScene.h @@ -23,10 +23,9 @@ #ifndef _TVG_SCENE_H_ #define _TVG_SCENE_H_ -#include +#include "tvgMath.h" #include "tvgPaint.h" - struct SceneIterator : Iterator { list* paints; @@ -61,8 +60,10 @@ struct Scene::Impl list paints; RenderData rd = nullptr; Scene* scene = nullptr; - uint8_t opacity; //for composition - bool needComp = false; //composite or not + RenderRegion vport = {0, 0, INT32_MAX, INT32_MAX}; + Array* effects = nullptr; + uint8_t opacity; //for composition + bool needComp = false; //composite or not Impl(Scene* s) : scene(s) { @@ -70,6 +71,8 @@ struct Scene::Impl ~Impl() { + resetEffects(); + for (auto paint : paints) { if (P(paint)->unref() == 0) delete(paint); } @@ -83,12 +86,15 @@ struct Scene::Impl { if (opacity == 0 || paints.empty()) return false; + //post effects requires composition + if (effects) return true; + //Masking may require composition (even if opacity == 255) auto compMethod = scene->composite(nullptr); if (compMethod != CompositeMethod::None && compMethod != CompositeMethod::ClipPath) return true; //Blending may require composition (even if opacity == 255) - if (scene->blend() != BlendMethod::Normal) return true; + if (PP(scene)->blendMethod != BlendMethod::Normal) return true; //Half translucent requires intermediate composition. if (opacity == 255) return false; @@ -96,31 +102,34 @@ struct Scene::Impl //If scene has several children or only scene, it may require composition. //OPTIMIZE: the bitmap type of the picture would not need the composition. //OPTIMIZE: a single paint of a scene would not need the composition. - if (paints.size() == 1 && paints.front()->identifier() == TVG_CLASS_ID_SHAPE) return false; + if (paints.size() == 1 && paints.front()->type() == Type::Shape) return false; return true; } RenderData update(RenderMethod* renderer, const Matrix& transform, Array& clips, uint8_t opacity, RenderUpdateFlag flag, TVG_UNUSED bool clipper) { + this->vport = renderer->viewport(); + if ((needComp = needComposition(opacity))) { /* Overriding opacity value. If this scene is half-translucent, - It must do intermeidate composition with that opacity value. */ + It must do intermediate composition with that opacity value. */ this->opacity = opacity; opacity = 255; } for (auto paint : paints) { paint->pImpl->update(renderer, transform, clips, opacity, flag, false); } + return nullptr; } bool render(RenderMethod* renderer) { - Compositor* cmp = nullptr; + RenderCompositor* cmp = nullptr; auto ret = true; - renderer->blend(scene->blend()); + renderer->blend(PP(scene)->blendMethod); if (needComp) { cmp = renderer->target(bounds(renderer), renderer->colorSpace()); @@ -131,7 +140,16 @@ struct Scene::Impl ret &= paint->pImpl->render(renderer); } - if (cmp) renderer->endComposite(cmp); + if (cmp) { + //Apply post effects if any. + if (effects) { + auto direct = effects->count == 1 ? true : false; + for (auto e = effects->begin(); e < effects->end(); ++e) { + renderer->effect(cmp, *e, opacity, direct); + } + } + renderer->endComposite(cmp); + } return ret; } @@ -155,7 +173,23 @@ struct Scene::Impl if (y2 < region.y + region.h) y2 = (region.y + region.h); } - return {x1, y1, (x2 - x1), (y2 - y1)}; + //Extends the render region if post effects require + int32_t ex = 0, ey = 0, ew = 0, eh = 0; + if (effects) { + for (auto e = effects->begin(); e < effects->end(); ++e) { + auto effect = *e; + if (effect->rd || renderer->prepare(effect)) { + ex = std::min(ex, effect->extend.x); + ey = std::min(ey, effect->extend.y); + ew = std::max(ew, effect->extend.w); + eh = std::max(eh, effect->extend.h); + } + } + } + + auto ret = RenderRegion{x1 + ex, y1 + ey, (x2 - x1) + ew, (y2 - y1) + eh}; + ret.intersect(this->vport); + return ret; } bool bounds(float* px, float* py, float* pw, float* ph, bool stroking) @@ -203,6 +237,8 @@ struct Scene::Impl dup->paints.push_back(cdup); } + if (effects) TVGERR("RENDERER", "TODO: Duplicate Effects?"); + return scene; } @@ -218,6 +254,8 @@ struct Scene::Impl { return new SceneIterator(&paints); } + + Result resetEffects(); }; #endif //_TVG_SCENE_H_ diff --git a/thirdparty/thorvg/src/renderer/tvgShape.cpp b/thirdparty/thorvg/src/renderer/tvgShape.cpp index 3b9293a00e05..269d951f05a3 100644 --- a/thirdparty/thorvg/src/renderer/tvgShape.cpp +++ b/thirdparty/thorvg/src/renderer/tvgShape.cpp @@ -34,7 +34,6 @@ Shape :: Shape() : pImpl(new Impl(this)) { - Paint::pImpl->id = TVG_CLASS_ID_SHAPE; } @@ -52,7 +51,13 @@ unique_ptr Shape::gen() noexcept uint32_t Shape::identifier() noexcept { - return TVG_CLASS_ID_SHAPE; + return (uint32_t) Type::Shape; +} + + +Type Shape::type() const noexcept +{ + return Type::Shape; } @@ -151,14 +156,14 @@ Result Shape::appendCircle(float cx, float cy, float rx, float ry) noexcept } -Result Shape::appendArc(float cx, float cy, float radius, float startAngle, float sweep, bool pie) noexcept +TVG_DEPRECATED Result Shape::appendArc(float cx, float cy, float radius, float startAngle, float sweep, bool pie) noexcept { //just circle if (sweep >= 360.0f || sweep <= -360.0f) return appendCircle(cx, cy, radius, radius); const float arcPrecision = 1e-5f; - startAngle = mathDeg2Rad(startAngle); - sweep = mathDeg2Rad(sweep); + startAngle = deg2rad(startAngle); + sweep = deg2rad(sweep); auto nCurves = static_cast(fabsf(sweep / MATH_PI2)); if (fabsf(sweep / MATH_PI2) - nCurves > arcPrecision) ++nCurves; @@ -409,12 +414,6 @@ Result Shape::strokeTrim(float begin, float end, bool simultaneous) noexcept } -bool Shape::strokeTrim(float* begin, float* end) const noexcept -{ - return pImpl->strokeTrim(begin, end); -} - - Result Shape::fill(FillRule r) noexcept { pImpl->rs.rule = r; diff --git a/thirdparty/thorvg/src/renderer/tvgShape.h b/thirdparty/thorvg/src/renderer/tvgShape.h index 440fb312b6c0..42f815206062 100644 --- a/thirdparty/thorvg/src/renderer/tvgShape.h +++ b/thirdparty/thorvg/src/renderer/tvgShape.h @@ -53,9 +53,9 @@ struct Shape::Impl { if (!rd) return false; - Compositor* cmp = nullptr; + RenderCompositor* cmp = nullptr; - renderer->blend(shape->blend()); + renderer->blend(PP(shape)->blendMethod); if (needComp) { cmp = renderer->target(bounds(renderer), renderer->colorSpace()); @@ -83,7 +83,7 @@ struct Shape::Impl auto method = shape->composite(&target); if (!target || method == CompositeMethod::ClipPath) return false; if (target->pImpl->opacity == 255 || target->pImpl->opacity == 0) { - if (target->identifier() == TVG_CLASS_ID_SHAPE) { + if (target->type() == Type::Shape) { auto shape = static_cast(target); if (!shape->fill()) { uint8_t r, g, b, a; @@ -106,7 +106,7 @@ struct Shape::Impl if ((needComp = needComposition(opacity))) { /* Overriding opacity value. If this scene is half-translucent, - It must do intermeidate composition with that opacity value. */ + It must do intermediate composition with that opacity value. */ this->opacity = opacity; opacity = 255; } @@ -219,7 +219,7 @@ struct Shape::Impl rs.stroke = new RenderStroke(); } - if (mathEqual(rs.stroke->trim.begin, begin) && mathEqual(rs.stroke->trim.end, end) && + if (tvg::equal(rs.stroke->trim.begin, begin) && tvg::equal(rs.stroke->trim.end, end) && rs.stroke->trim.simultaneous == simultaneous) return; rs.stroke->trim.begin = begin; diff --git a/thirdparty/thorvg/src/renderer/tvgSwCanvas.cpp b/thirdparty/thorvg/src/renderer/tvgSwCanvas.cpp index d762492f22dc..6c4b6da1de44 100644 --- a/thirdparty/thorvg/src/renderer/tvgSwCanvas.cpp +++ b/thirdparty/thorvg/src/renderer/tvgSwCanvas.cpp @@ -82,7 +82,7 @@ Result SwCanvas::mempool(MempoolPolicy policy) noexcept Result SwCanvas::target(uint32_t* buffer, uint32_t stride, uint32_t w, uint32_t h, Colorspace cs) noexcept { #ifdef THORVG_SW_RASTER_SUPPORT - if (Canvas::pImpl->status != Status::Damanged && Canvas::pImpl->status != Status::Synced) { + if (Canvas::pImpl->status != Status::Damaged && Canvas::pImpl->status != Status::Synced) { return Result::InsufficientCondition; } @@ -98,7 +98,7 @@ Result SwCanvas::target(uint32_t* buffer, uint32_t stride, uint32_t w, uint32_t ImageLoader::cs = static_cast(cs); //Paints must be updated again with this new target. - Canvas::pImpl->status = Status::Damanged; + Canvas::pImpl->status = Status::Damaged; return Result::Success; #endif diff --git a/thirdparty/thorvg/src/renderer/tvgText.cpp b/thirdparty/thorvg/src/renderer/tvgText.cpp index d54c78783c5d..b324b95049e9 100644 --- a/thirdparty/thorvg/src/renderer/tvgText.cpp +++ b/thirdparty/thorvg/src/renderer/tvgText.cpp @@ -37,7 +37,6 @@ Text::Text() : pImpl(new Impl(this)) { - Paint::pImpl->id = TVG_CLASS_ID_TEXT; } @@ -111,7 +110,7 @@ unique_ptr Text::gen() noexcept } -uint32_t Text::identifier() noexcept +Type Text::type() const noexcept { - return TVG_CLASS_ID_TEXT; + return Type::Text; } diff --git a/thirdparty/thorvg/src/renderer/tvgText.h b/thirdparty/thorvg/src/renderer/tvgText.h index cb9a76c05218..11e01b58ce6d 100644 --- a/thirdparty/thorvg/src/renderer/tvgText.h +++ b/thirdparty/thorvg/src/renderer/tvgText.h @@ -90,7 +90,7 @@ struct Text::Impl bool render(RenderMethod* renderer) { if (!loader) return true; - renderer->blend(paint->blend()); + renderer->blend(PP(paint)->blendMethod); return PP(shape)->render(renderer); } @@ -115,7 +115,7 @@ struct Text::Impl auto fill = P(shape)->rs.fill; if (fill && P(shape)->flag & RenderUpdateFlag::Gradient) { auto scale = 1.0f / loader->scale; - if (fill->identifier() == TVG_CLASS_ID_LINEAR) { + if (fill->type() == Type::LinearGradient) { P(static_cast(fill))->x1 *= scale; P(static_cast(fill))->y1 *= scale; P(static_cast(fill))->x2 *= scale; diff --git a/thirdparty/thorvg/src/renderer/tvgWgCanvas.cpp b/thirdparty/thorvg/src/renderer/tvgWgCanvas.cpp index 067e35b1f0d8..991f73fc542c 100644 --- a/thirdparty/thorvg/src/renderer/tvgWgCanvas.cpp +++ b/thirdparty/thorvg/src/renderer/tvgWgCanvas.cpp @@ -40,7 +40,7 @@ struct WgCanvas::Impl /************************************************************************/ #ifdef THORVG_WG_RASTER_SUPPORT -WgCanvas::WgCanvas() : Canvas(WgRenderer::gen()), pImpl(new Impl) +WgCanvas::WgCanvas() : Canvas(WgRenderer::gen()), pImpl(nullptr) #else WgCanvas::WgCanvas() : Canvas(nullptr), pImpl(nullptr) #endif @@ -50,14 +50,17 @@ WgCanvas::WgCanvas() : Canvas(nullptr), pImpl(nullptr) WgCanvas::~WgCanvas() { - delete pImpl; +#ifdef THORVG_WG_RASTER_SUPPORT + auto renderer = static_cast(Canvas::pImpl->renderer); + renderer->target(nullptr, 0, 0); +#endif } -Result WgCanvas::target(void* instance, void* surface, uint32_t w, uint32_t h) noexcept +Result WgCanvas::target(void* instance, void* surface, uint32_t w, uint32_t h, void* device) noexcept { #ifdef THORVG_WG_RASTER_SUPPORT - if (Canvas::pImpl->status != Status::Damanged && Canvas::pImpl->status != Status::Synced) { + if (Canvas::pImpl->status != Status::Damaged && Canvas::pImpl->status != Status::Synced) { return Result::InsufficientCondition; } @@ -67,12 +70,12 @@ Result WgCanvas::target(void* instance, void* surface, uint32_t w, uint32_t h) n auto renderer = static_cast(Canvas::pImpl->renderer); if (!renderer) return Result::MemoryCorruption; - if (!renderer->target((WGPUInstance)instance, (WGPUSurface)surface, w, h)) return Result::Unknown; + if (!renderer->target((WGPUInstance)instance, (WGPUSurface)surface, w, h, (WGPUDevice)device)) return Result::Unknown; Canvas::pImpl->vport = {0, 0, (int32_t)w, (int32_t)h}; renderer->viewport(Canvas::pImpl->vport); //Paints must be updated again with this new target. - Canvas::pImpl->status = Status::Damanged; + Canvas::pImpl->status = Status::Damaged; return Result::Success; #endif diff --git a/thirdparty/thorvg/update-thorvg.sh b/thirdparty/thorvg/update-thorvg.sh index 2c5d84d266cb..f9953f2fc9e5 100755 --- a/thirdparty/thorvg/update-thorvg.sh +++ b/thirdparty/thorvg/update-thorvg.sh @@ -1,6 +1,6 @@ #!/bin/bash -e -VERSION=0.14.10 +VERSION=0.15.5 # Uncomment and set a git hash to use specific commit instead of tag. #GIT_COMMIT= From 0d2e13bcb8f25ab3ace44163eb82a5589ab87aa1 Mon Sep 17 00:00:00 2001 From: Lukas Tenbrink Date: Wed, 4 Dec 2024 17:50:40 +0100 Subject: [PATCH 17/33] Optimize is_valid_filename and validate_filename by caching invalid filename characters, instead of re-splitting each call. --- core/string/ustring.cpp | 10 ++++------ 1 file changed, 4 insertions(+), 6 deletions(-) diff --git a/core/string/ustring.cpp b/core/string/ustring.cpp index 0e553a5680b6..b598326c4b7b 100644 --- a/core/string/ustring.cpp +++ b/core/string/ustring.cpp @@ -5286,7 +5286,7 @@ bool String::is_valid_html_color() const { } // Changes made to the set of invalid filename characters must also be reflected in the String documentation for is_valid_filename. -static const char *invalid_filename_characters = ": / \\ ? * \" | % < >"; +static const char *invalid_filename_characters[] = { ":", "/", "\\", "?", "*", "\"", "|", "%", "<", ">" }; bool String::is_valid_filename() const { String stripped = strip_edges(); @@ -5298,8 +5298,7 @@ bool String::is_valid_filename() const { return false; } - Vector chars = String(invalid_filename_characters).split(" "); - for (const String &ch : chars) { + for (const char *ch : invalid_filename_characters) { if (contains(ch)) { return false; } @@ -5308,10 +5307,9 @@ bool String::is_valid_filename() const { } String String::validate_filename() const { - Vector chars = String(invalid_filename_characters).split(" "); String name = strip_edges(); - for (int i = 0; i < chars.size(); i++) { - name = name.replace(chars[i], "_"); + for (const char *ch : invalid_filename_characters) { + name = name.replace(ch, "_"); } return name; } From 8d82933c7cfe4a16b0b5806e1f553fdcd1b8c57e Mon Sep 17 00:00:00 2001 From: Lukas Tenbrink Date: Wed, 4 Dec 2024 18:39:38 +0100 Subject: [PATCH 18/33] Optimize _camelcase_to_underscore (and thus String.capitalize) by using a rolling cache of character attributes, instead of re-computing each iteration. --- core/string/ustring.cpp | 24 ++++++++++++++++-------- 1 file changed, 16 insertions(+), 8 deletions(-) diff --git a/core/string/ustring.cpp b/core/string/ustring.cpp index 0e553a5680b6..7710d3ef91bf 100644 --- a/core/string/ustring.cpp +++ b/core/string/ustring.cpp @@ -1109,17 +1109,21 @@ String String::_camelcase_to_underscore() const { String new_string; int start_index = 0; - for (int i = 1; i < size(); i++) { - bool is_prev_upper = is_unicode_upper_case(cstr[i - 1]); - bool is_prev_lower = is_unicode_lower_case(cstr[i - 1]); - bool is_prev_digit = is_digit(cstr[i - 1]); + if (length() == 0) { + return *this; + } - bool is_curr_upper = is_unicode_upper_case(cstr[i]); - bool is_curr_lower = is_unicode_lower_case(cstr[i]); - bool is_curr_digit = is_digit(cstr[i]); + bool is_prev_upper = is_unicode_upper_case(cstr[0]); + bool is_prev_lower = is_unicode_lower_case(cstr[0]); + bool is_prev_digit = is_digit(cstr[0]); + + for (int i = 1; i < length(); i++) { + const bool is_curr_upper = is_unicode_upper_case(cstr[i]); + const bool is_curr_lower = is_unicode_lower_case(cstr[i]); + const bool is_curr_digit = is_digit(cstr[i]); bool is_next_lower = false; - if (i + 1 < size()) { + if (i + 1 < length()) { is_next_lower = is_unicode_lower_case(cstr[i + 1]); } @@ -1132,6 +1136,10 @@ String String::_camelcase_to_underscore() const { new_string += substr(start_index, i - start_index) + "_"; start_index = i; } + + is_prev_upper = is_curr_upper; + is_prev_lower = is_curr_lower; + is_prev_digit = is_curr_digit; } new_string += substr(start_index, size() - start_index); From 2336415623189d2d63c5bce389156e618bdcc956 Mon Sep 17 00:00:00 2001 From: Malcolm Anderson Date: Mon, 2 Dec 2024 17:50:20 -0800 Subject: [PATCH 19/33] Ignore `__MACOSX` directory in export template collection ZIP file and project import ZIP file Update editor/export/export_template_manager.cpp Co-authored-by: A Thousand Ships <96648715+AThousandShips@users.noreply.github.com> --- editor/export/export_template_manager.cpp | 10 +++++++++- editor/project_manager/project_dialog.cpp | 24 ++++++++++++++++++++++- 2 files changed, 32 insertions(+), 2 deletions(-) diff --git a/editor/export/export_template_manager.cpp b/editor/export/export_template_manager.cpp index a90c16f66e69..1e1999ddb702 100644 --- a/editor/export/export_template_manager.cpp +++ b/editor/export/export_template_manager.cpp @@ -437,6 +437,13 @@ bool ExportTemplateManager::_install_file_selected(const String &p_file, bool p_ } String file = String::utf8(fname); + + // Skip the __MACOSX directory created by macOS's built-in file zipper. + if (file.begins_with("__MACOSX")) { + ret = unzGoToNextFile(pkg); + continue; + } + if (file.ends_with("version.txt")) { Vector uncomp_data; uncomp_data.resize(info.uncompressed_size); @@ -512,7 +519,8 @@ bool ExportTemplateManager::_install_file_selected(const String &p_file, bool p_ String file = file_path.get_file(); - if (file.size() == 0) { + // Skip the __MACOSX directory created by macOS's built-in file zipper. + if (file.is_empty() || file.begins_with("__MACOSX")) { ret = unzGoToNextFile(pkg); continue; } diff --git a/editor/project_manager/project_dialog.cpp b/editor/project_manager/project_dialog.cpp index 8adc0473d6cc..7dd7506c172f 100644 --- a/editor/project_manager/project_dialog.cpp +++ b/editor/project_manager/project_dialog.cpp @@ -132,6 +132,13 @@ void ProjectDialog::_validate_path() { ERR_FAIL_COND_MSG(ret != UNZ_OK, "Failed to get current file info."); String name = String::utf8(fname); + + // Skip the __MACOSX directory created by macOS's built-in file zipper. + if (name.begins_with("__MACOSX")) { + ret = unzGoToNextFile(pkg); + continue; + } + if (name.get_file() == "project.godot") { break; // ret == UNZ_OK. } @@ -604,6 +611,13 @@ void ProjectDialog::ok_pressed() { ERR_FAIL_COND_MSG(ret != UNZ_OK, "Failed to get current file info."); String name = String::utf8(fname); + + // Skip the __MACOSX directory created by macOS's built-in file zipper. + if (name.begins_with("__MACOSX")) { + ret = unzGoToNextFile(pkg); + continue; + } + if (name.get_file() == "project.godot") { zip_root = name.get_base_dir(); break; @@ -636,7 +650,15 @@ void ProjectDialog::ok_pressed() { ret = unzGetCurrentFileInfo(pkg, &info, fname, 16384, nullptr, 0, nullptr, 0); ERR_FAIL_COND_MSG(ret != UNZ_OK, "Failed to get current file info."); - String rel_path = String::utf8(fname).trim_prefix(zip_root); + String name = String::utf8(fname); + + // Skip the __MACOSX directory created by macOS's built-in file zipper. + if (name.begins_with("__MACOSX")) { + ret = unzGoToNextFile(pkg); + continue; + } + + String rel_path = name.trim_prefix(zip_root); if (rel_path.is_empty()) { // Root. } else if (rel_path.ends_with("/")) { // Directory. Ref da = DirAccess::create(DirAccess::ACCESS_FILESYSTEM); From eaa0418debe9911b4aed2adf07465884cd905abb Mon Sep 17 00:00:00 2001 From: DeeJayLSP Date: Wed, 4 Dec 2024 16:46:18 -0300 Subject: [PATCH 20/33] TestAudioStreamWAV: use runtime load instead of importer --- tests/scene/test_audio_stream_wav.h | 27 ++------------------------- 1 file changed, 2 insertions(+), 25 deletions(-) diff --git a/tests/scene/test_audio_stream_wav.h b/tests/scene/test_audio_stream_wav.h index 7276dd0878c9..30e3f5b9ba47 100644 --- a/tests/scene/test_audio_stream_wav.h +++ b/tests/scene/test_audio_stream_wav.h @@ -37,11 +37,6 @@ #include "tests/test_macros.h" -#ifdef TOOLS_ENABLED -#include "core/io/resource_loader.h" -#include "editor/import/resource_importer_wav.h" -#endif - namespace TestAudioStreamWAV { // Default wav rate for test cases. @@ -148,25 +143,8 @@ void run_test(String file_name, AudioStreamWAV::Format data_format, bool stereo, Ref wav_file = FileAccess::open(save_path, FileAccess::READ, &error); REQUIRE(error == OK); -#ifdef TOOLS_ENABLED - // The WAV importer can be used if enabled to check that the saved file is valid. - Ref wav_importer = memnew(ResourceImporterWAV); - - List options_list; - wav_importer->get_import_options("", &options_list); - - HashMap options_map; - for (const ResourceImporter::ImportOption &E : options_list) { - options_map[E.option.name] = E.default_value; - } - // Compressed streams can't be saved, disable compression. - options_map["compress/mode"] = 0; - - REQUIRE(wav_importer->import(0, save_path, save_path, options_map, nullptr) == OK); - - String load_path = save_path + "." + wav_importer->get_save_extension(); - Ref loaded_stream = ResourceLoader::load(load_path, "AudioStreamWAV", ResourceFormatImporter::CACHE_MODE_IGNORE, &error); - REQUIRE(error == OK); + Dictionary options; + Ref loaded_stream = AudioStreamWAV::load_from_file(save_path, options); CHECK(loaded_stream->get_format() == stream->get_format()); CHECK(loaded_stream->get_loop_mode() == stream->get_loop_mode()); @@ -177,7 +155,6 @@ void run_test(String file_name, AudioStreamWAV::Format data_format, bool stereo, CHECK(loaded_stream->get_length() == stream->get_length()); CHECK(loaded_stream->is_monophonic() == stream->is_monophonic()); CHECK(loaded_stream->get_data() == stream->get_data()); -#endif } } From e316739a134a3bbe07584a586023cd5bf5e826b4 Mon Sep 17 00:00:00 2001 From: clayjohn Date: Wed, 4 Dec 2024 16:36:46 -0800 Subject: [PATCH 21/33] Use a vector instead of an array in canvas shader instance buffer. This avoids a pathological performance cliff on Adreno devices --- servers/rendering/renderer_rd/shaders/canvas_uniforms_inc.glsl | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/servers/rendering/renderer_rd/shaders/canvas_uniforms_inc.glsl b/servers/rendering/renderer_rd/shaders/canvas_uniforms_inc.glsl index da582ec1b46b..f5a806fbca39 100644 --- a/servers/rendering/renderer_rd/shaders/canvas_uniforms_inc.glsl +++ b/servers/rendering/renderer_rd/shaders/canvas_uniforms_inc.glsl @@ -37,7 +37,7 @@ struct InstanceData { #endif vec2 color_texture_pixel_size; - uint lights[4]; + uvec4 lights; }; //1 means enabled, 2+ means trails in use From 73f7e1379c3431f9ee9c7eb55d87649a69758f85 Mon Sep 17 00:00:00 2001 From: passivestar <60579014+passivestar@users.noreply.github.com> Date: Thu, 5 Dec 2024 05:27:34 +0400 Subject: [PATCH 22/33] Fix vertical alignment of the main menu bar --- editor/editor_node.cpp | 1 + 1 file changed, 1 insertion(+) diff --git a/editor/editor_node.cpp b/editor/editor_node.cpp index 222696a60846..7c30242ffb7d 100644 --- a/editor/editor_node.cpp +++ b/editor/editor_node.cpp @@ -7173,6 +7173,7 @@ EditorNode::EditorNode() { main_menu = memnew(MenuBar); main_menu->set_mouse_filter(Control::MOUSE_FILTER_STOP); title_bar->add_child(main_menu); + main_menu->set_v_size_flags(Control::SIZE_SHRINK_CENTER); main_menu->set_theme_type_variation("MainMenuBar"); main_menu->set_start_index(0); // Main menu, add to the start of global menu. main_menu->set_prefer_global_menu(global_menu); From 1ef3bd57688024a53fd583248a4e4afb041c81aa Mon Sep 17 00:00:00 2001 From: Haoyu Qiu Date: Tue, 3 Dec 2024 09:43:12 +0800 Subject: [PATCH 23/33] Fix UI inconsistencies in EditorFileDialog's toolbar - Sort button has different stylebox compared to others - The top toolbar has a separator on the right side when in open mode - Update fav up/down buttons when fav list changes --- editor/gui/editor_file_dialog.cpp | 21 +++++++++++++-------- editor/gui/editor_file_dialog.h | 2 ++ 2 files changed, 15 insertions(+), 8 deletions(-) diff --git a/editor/gui/editor_file_dialog.cpp b/editor/gui/editor_file_dialog.cpp index 6f257f492d40..8ece6f5dfef2 100644 --- a/editor/gui/editor_file_dialog.cpp +++ b/editor/gui/editor_file_dialog.cpp @@ -44,7 +44,6 @@ #include "scene/gui/check_box.h" #include "scene/gui/grid_container.h" #include "scene/gui/label.h" -#include "scene/gui/margin_container.h" #include "scene/gui/option_button.h" #include "scene/gui/separator.h" #include "scene/gui/split_container.h" @@ -1400,11 +1399,8 @@ void EditorFileDialog::set_file_mode(FileMode p_mode) { item_list->set_select_mode(ItemList::SELECT_SINGLE); } - if (can_create_dir) { - makedir->show(); - } else { - makedir->hide(); - } + makedir_sep->set_visible(can_create_dir); + makedir->set_visible(can_create_dir); } EditorFileDialog::FileMode EditorFileDialog::get_file_mode() const { @@ -1571,6 +1567,8 @@ void EditorFileDialog::_select_drive(int p_idx) { void EditorFileDialog::_update_drives(bool p_select) { int dc = dir_access->get_drive_count(); if (dc == 0 || access != ACCESS_FILESYSTEM) { + shortcuts_container->hide(); + drives_container->hide(); drives->hide(); } else { drives->clear(); @@ -1579,6 +1577,8 @@ void EditorFileDialog::_update_drives(bool p_select) { dp->remove_child(drives); } dp = dir_access->drives_are_shortcuts() ? shortcuts_container : drives_container; + shortcuts_container->set_visible(dir_access->drives_are_shortcuts()); + drives_container->set_visible(!dir_access->drives_are_shortcuts()); dp->add_child(drives); drives->show(); @@ -1768,6 +1768,9 @@ void EditorFileDialog::_update_favorites() { recent->deselect_all(); } } + + fav_up->set_disabled(current_favorite < 1); + fav_down->set_disabled(current_favorite == -1 || favorited_paths.size() - 1 <= current_favorite); } void EditorFileDialog::_favorite_pressed() { @@ -2354,7 +2357,8 @@ EditorFileDialog::EditorFileDialog() { drives->connect(SceneStringName(item_selected), callable_mp(this, &EditorFileDialog::_select_drive)); pathhb->add_child(drives); - pathhb->add_child(memnew(VSeparator)); + makedir_sep = memnew(VSeparator); + pathhb->add_child(makedir_sep); makedir = memnew(Button); makedir->set_theme_type_variation("FlatButton"); @@ -2473,7 +2477,8 @@ EditorFileDialog::EditorFileDialog() { lower_hb->add_child(memnew(VSeparator)); file_sort_button = memnew(MenuButton); - file_sort_button->set_flat(true); + file_sort_button->set_flat(false); + file_sort_button->set_theme_type_variation("FlatMenuButton"); file_sort_button->set_tooltip_text(TTR("Sort files")); show_search_filter_button = memnew(Button); diff --git a/editor/gui/editor_file_dialog.h b/editor/gui/editor_file_dialog.h index a151f2b38f74..8a07a2094372 100644 --- a/editor/gui/editor_file_dialog.h +++ b/editor/gui/editor_file_dialog.h @@ -44,6 +44,7 @@ class MenuButton; class OptionButton; class PopupMenu; class TextureRect; +class VSeparator; class EditorFileDialog : public ConfirmationDialog { GDCLASS(EditorFileDialog, ConfirmationDialog); @@ -89,6 +90,7 @@ class EditorFileDialog : public ConfirmationDialog { ConfirmationDialog *makedialog = nullptr; LineEdit *makedirname = nullptr; + VSeparator *makedir_sep = nullptr; Button *makedir = nullptr; Access access = ACCESS_RESOURCES; From 8a5eef95aa32ccaa7473498adb3f404d89f1e6d8 Mon Sep 17 00:00:00 2001 From: Wyv <181348033+wyvrtn@users.noreply.github.com> Date: Tue, 3 Dec 2024 11:37:38 -0500 Subject: [PATCH 24/33] Fix a very slight grammatical issue Changed the incorrect verb "have" into its correct form, "has." Fixes #99972 Co-authored-by: A Thousand Ships <96648715+AThousandShips@users.noreply.github.com> --- doc/classes/PackedFloat32Array.xml | 2 +- doc/classes/PackedFloat64Array.xml | 2 +- doc/classes/PackedInt32Array.xml | 2 +- doc/classes/PackedInt64Array.xml | 2 +- doc/classes/SurfaceTool.xml | 2 +- 5 files changed, 5 insertions(+), 5 deletions(-) diff --git a/doc/classes/PackedFloat32Array.xml b/doc/classes/PackedFloat32Array.xml index d421993b8fd9..237b659a5225 100644 --- a/doc/classes/PackedFloat32Array.xml +++ b/doc/classes/PackedFloat32Array.xml @@ -192,7 +192,7 @@ - Returns a copy of the data converted to a [PackedByteArray], where each element have been encoded as 4 bytes. + Returns a copy of the data converted to a [PackedByteArray], where each element has been encoded as 4 bytes. The size of the new array will be [code]float32_array.size() * 4[/code]. diff --git a/doc/classes/PackedFloat64Array.xml b/doc/classes/PackedFloat64Array.xml index 4622d632586c..56cc08d8c409 100644 --- a/doc/classes/PackedFloat64Array.xml +++ b/doc/classes/PackedFloat64Array.xml @@ -193,7 +193,7 @@ - Returns a copy of the data converted to a [PackedByteArray], where each element have been encoded as 8 bytes. + Returns a copy of the data converted to a [PackedByteArray], where each element has been encoded as 8 bytes. The size of the new array will be [code]float64_array.size() * 8[/code]. diff --git a/doc/classes/PackedInt32Array.xml b/doc/classes/PackedInt32Array.xml index 3a3596b2d02b..5bb2ea97ef95 100644 --- a/doc/classes/PackedInt32Array.xml +++ b/doc/classes/PackedInt32Array.xml @@ -186,7 +186,7 @@ - Returns a copy of the data converted to a [PackedByteArray], where each element have been encoded as 4 bytes. + Returns a copy of the data converted to a [PackedByteArray], where each element has been encoded as 4 bytes. The size of the new array will be [code]int32_array.size() * 4[/code]. diff --git a/doc/classes/PackedInt64Array.xml b/doc/classes/PackedInt64Array.xml index b82d0de350e3..2ed854895881 100644 --- a/doc/classes/PackedInt64Array.xml +++ b/doc/classes/PackedInt64Array.xml @@ -187,7 +187,7 @@ - Returns a copy of the data converted to a [PackedByteArray], where each element have been encoded as 8 bytes. + Returns a copy of the data converted to a [PackedByteArray], where each element has been encoded as 8 bytes. The size of the new array will be [code]int64_array.size() * 8[/code]. diff --git a/doc/classes/SurfaceTool.xml b/doc/classes/SurfaceTool.xml index 9265e6b3459c..258d68752ea6 100644 --- a/doc/classes/SurfaceTool.xml +++ b/doc/classes/SurfaceTool.xml @@ -147,7 +147,7 @@ - Generates a tangent vector for each vertex. Requires that each vertex have UVs and normals set already (see [method generate_normals]). + Generates a tangent vector for each vertex. Requires that each vertex already has UVs and normals set (see [method generate_normals]). From 06cae04b878c98e9cd89ed25e5f25421e973c9f0 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Pa=CC=84vels=20Nadtoc=CC=8Cajevs?= <7645683+bruvzg@users.noreply.github.com> Date: Thu, 5 Dec 2024 16:26:52 +0200 Subject: [PATCH 25/33] Change default Arabic font to Vazirmatn. --- COPYRIGHT.txt | 5 + editor/themes/editor_fonts.cpp | 4 +- tests/servers/test_text_server.h | 8 +- thirdparty/README.md | 8 +- thirdparty/fonts/LICENSE.Vazirmatn.txt | 93 ++++++++++++++++++ thirdparty/fonts/NotoNaskhArabicUI_Bold.woff2 | Bin 53860 -> 0 bytes .../fonts/NotoNaskhArabicUI_Regular.woff2 | Bin 53352 -> 0 bytes thirdparty/fonts/Vazirmatn_Bold.woff2 | Bin 0 -> 51184 bytes thirdparty/fonts/Vazirmatn_Regular.woff2 | Bin 0 -> 50596 bytes 9 files changed, 108 insertions(+), 10 deletions(-) create mode 100644 thirdparty/fonts/LICENSE.Vazirmatn.txt delete mode 100644 thirdparty/fonts/NotoNaskhArabicUI_Bold.woff2 delete mode 100644 thirdparty/fonts/NotoNaskhArabicUI_Regular.woff2 create mode 100644 thirdparty/fonts/Vazirmatn_Bold.woff2 create mode 100644 thirdparty/fonts/Vazirmatn_Regular.woff2 diff --git a/COPYRIGHT.txt b/COPYRIGHT.txt index 5cc6f7971b55..cff3642cb63e 100644 --- a/COPYRIGHT.txt +++ b/COPYRIGHT.txt @@ -240,6 +240,11 @@ Comment: Noto Sans font Copyright: 2012, Google Inc. License: OFL-1.1 +Files: ./thirdparty/fonts/Vazirmatn*.woff2 +Comment: Vazirmatn font +Copyright: 2015, The Vazirmatn Project Authors. +License: OFL-1.1 + Files: ./thirdparty/freetype/ Comment: The FreeType Project Copyright: 1996-2023, David Turner, Robert Wilhelm, and Werner Lemberg. diff --git a/editor/themes/editor_fonts.cpp b/editor/themes/editor_fonts.cpp index da35ed332ee8..8ae4889214b8 100644 --- a/editor/themes/editor_fonts.cpp +++ b/editor/themes/editor_fonts.cpp @@ -169,7 +169,7 @@ void editor_register_fonts(const Ref &p_theme) { } TypedArray fallbacks; - Ref arabic_font = load_internal_font(_font_NotoNaskhArabicUI_Regular, _font_NotoNaskhArabicUI_Regular_size, font_hinting, font_antialiasing, true, font_subpixel_positioning, font_disable_embedded_bitmaps, false, &fallbacks); + Ref arabic_font = load_internal_font(_font_Vazirmatn_Regular, _font_Vazirmatn_Regular_size, font_hinting, font_antialiasing, true, font_subpixel_positioning, font_disable_embedded_bitmaps, false, &fallbacks); Ref bengali_font = load_internal_font(_font_NotoSansBengaliUI_Regular, _font_NotoSansBengaliUI_Regular_size, font_hinting, font_antialiasing, true, font_subpixel_positioning, font_disable_embedded_bitmaps, false, &fallbacks); Ref devanagari_font = load_internal_font(_font_NotoSansDevanagariUI_Regular, _font_NotoSansDevanagariUI_Regular_size, font_hinting, font_antialiasing, true, font_subpixel_positioning, font_disable_embedded_bitmaps, false, &fallbacks); Ref georgian_font = load_internal_font(_font_NotoSansGeorgian_Regular, _font_NotoSansGeorgian_Regular_size, font_hinting, font_antialiasing, true, font_subpixel_positioning, font_disable_embedded_bitmaps, false, &fallbacks); @@ -192,7 +192,7 @@ void editor_register_fonts(const Ref &p_theme) { Ref default_font_bold_msdf = load_internal_font(_font_NotoSans_Bold, _font_NotoSans_Bold_size, font_hinting, font_antialiasing, true, font_subpixel_positioning, font_disable_embedded_bitmaps, font_allow_msdf); TypedArray fallbacks_bold; - Ref arabic_font_bold = load_internal_font(_font_NotoNaskhArabicUI_Bold, _font_NotoNaskhArabicUI_Bold_size, font_hinting, font_antialiasing, true, font_subpixel_positioning, font_disable_embedded_bitmaps, false, &fallbacks_bold); + Ref arabic_font_bold = load_internal_font(_font_Vazirmatn_Bold, _font_Vazirmatn_Bold_size, font_hinting, font_antialiasing, true, font_subpixel_positioning, font_disable_embedded_bitmaps, false, &fallbacks_bold); Ref bengali_font_bold = load_internal_font(_font_NotoSansBengaliUI_Bold, _font_NotoSansBengaliUI_Bold_size, font_hinting, font_antialiasing, true, font_subpixel_positioning, font_disable_embedded_bitmaps, false, &fallbacks_bold); Ref devanagari_font_bold = load_internal_font(_font_NotoSansDevanagariUI_Bold, _font_NotoSansDevanagariUI_Bold_size, font_hinting, font_antialiasing, true, font_subpixel_positioning, font_disable_embedded_bitmaps, false, &fallbacks_bold); Ref georgian_font_bold = load_internal_font(_font_NotoSansGeorgian_Bold, _font_NotoSansGeorgian_Bold_size, font_hinting, font_antialiasing, true, font_subpixel_positioning, font_disable_embedded_bitmaps, false, &fallbacks_bold); diff --git a/tests/servers/test_text_server.h b/tests/servers/test_text_server.h index 4190d3c23a4b..4e20d43efc29 100644 --- a/tests/servers/test_text_server.h +++ b/tests/servers/test_text_server.h @@ -124,7 +124,7 @@ TEST_SUITE("[TextServer]") { RID font1 = ts->create_font(); ts->font_set_data_ptr(font1, _font_NotoSans_Regular, _font_NotoSans_Regular_size); RID font2 = ts->create_font(); - ts->font_set_data_ptr(font2, _font_NotoNaskhArabicUI_Regular, _font_NotoNaskhArabicUI_Regular_size); + ts->font_set_data_ptr(font2, _font_Vazirmatn_Regular, _font_Vazirmatn_Regular_size); Array font; font.push_back(font1); @@ -180,7 +180,7 @@ TEST_SUITE("[TextServer]") { ts->font_set_data_ptr(font2, _font_NotoSansThai_Regular, _font_NotoSansThai_Regular_size); ts->font_set_allow_system_fallback(font2, false); RID font3 = ts->create_font(); - ts->font_set_data_ptr(font3, _font_NotoNaskhArabicUI_Regular, _font_NotoNaskhArabicUI_Regular_size); + ts->font_set_data_ptr(font3, _font_Vazirmatn_Regular, _font_Vazirmatn_Regular_size); ts->font_set_allow_system_fallback(font3, false); Array font; @@ -503,7 +503,7 @@ TEST_SUITE("[TextServer]") { { U" มีอุปกรณ์\nนี้", { 0, 11, 11, 19, 19, 22 } }, { U"الحمدا لحمدا لحمـــد", { 0, 13, 13, 20 } }, { U" الحمد test", { 0, 15, 15, 19 } }, - { U"الحمـد الرياضي العربي", { 0, 7, 7, 21 } }, + { U"الحمـد الرياضي العربي", { 0, 7, 7, 15, 15, 21 } }, }; for (size_t j = 0; j < sizeof(cases) / sizeof(TestCase); j++) { RID ctx = ts->create_shaped_text(); @@ -584,7 +584,7 @@ TEST_SUITE("[TextServer]") { RID font1 = ts->create_font(); ts->font_set_data_ptr(font1, _font_NotoSans_Regular, _font_NotoSans_Regular_size); RID font2 = ts->create_font(); - ts->font_set_data_ptr(font2, _font_NotoNaskhArabicUI_Regular, _font_NotoNaskhArabicUI_Regular_size); + ts->font_set_data_ptr(font2, _font_Vazirmatn_Regular, _font_Vazirmatn_Regular_size); Array font; font.push_back(font1); diff --git a/thirdparty/README.md b/thirdparty/README.md index 1e11c6c9725b..8f6736eec416 100644 --- a/thirdparty/README.md +++ b/thirdparty/README.md @@ -257,10 +257,6 @@ Files extracted from upstream source: * Upstream: https://github.com/JetBrains/JetBrainsMono * Version: 2.304 (cd5227bd1f61dff3bbd6c814ceaf7ffd95e947d9, 2023) * License: OFL-1.1 -- `NotoNaskhArabicUI*.woff2`: - * Upstream: https://github.com/notofonts/arabic - * Version: 2.014 (133ccaebf922ca080a7eef22998611ac3c242df9, 2022) - * License: OFL-1.1 - `NotoSans*.woff2`: * Upstream: https://github.com/notofonts/latin-greek-cyrillic * Version: 2.012 (9ea0c8d37bff0c0067b03777f40aa04f2bf78f99, 2023) @@ -309,6 +305,10 @@ Files extracted from upstream source: * Upstream: https://fonts.google.com/specimen/Open+Sans * Version: 1.10 (downloaded from Google Fonts in February 2021) * License: Apache 2.0 +- `Vazirmatn*.woff2`: + * Upstream: https://github.com/rastikerdar/vazirmatn + * Version: 33.003 (83629f877e8f084cc07b47030b5d3a0ff06c76ec, 2022) + * License: OFL-1.1 All fonts are converted from the unhinted `.ttf` sources using the `https://github.com/google/woff2` tool. diff --git a/thirdparty/fonts/LICENSE.Vazirmatn.txt b/thirdparty/fonts/LICENSE.Vazirmatn.txt new file mode 100644 index 000000000000..be66b3824e97 --- /dev/null +++ b/thirdparty/fonts/LICENSE.Vazirmatn.txt @@ -0,0 +1,93 @@ +Copyright 2015 The Vazirmatn Project Authors (https://github.com/rastikerdar/vazirmatn) + +This Font Software is licensed under the SIL Open Font License, Version 1.1. +This license is copied below, and is also available with a FAQ at: +http://scripts.sil.org/OFL + + +----------------------------------------------------------- +SIL OPEN FONT LICENSE Version 1.1 - 26 February 2007 +----------------------------------------------------------- + +PREAMBLE +The goals of the Open Font License (OFL) are to stimulate worldwide +development of collaborative font projects, to support the font creation +efforts of academic and linguistic communities, and to provide a free and +open framework in which fonts may be shared and improved in partnership +with others. + +The OFL allows the licensed fonts to be used, studied, modified and +redistributed freely as long as they are not sold by themselves. The +fonts, including any derivative works, can be bundled, embedded, +redistributed and/or sold with any software provided that any reserved +names are not used by derivative works. The fonts and derivatives, +however, cannot be released under any other type of license. The +requirement for fonts to remain under this license does not apply +to any document created using the fonts or their derivatives. + +DEFINITIONS +"Font Software" refers to the set of files released by the Copyright +Holder(s) under this license and clearly marked as such. This may +include source files, build scripts and documentation. + +"Reserved Font Name" refers to any names specified as such after the +copyright statement(s). + +"Original Version" refers to the collection of Font Software components as +distributed by the Copyright Holder(s). + +"Modified Version" refers to any derivative made by adding to, deleting, +or substituting -- in part or in whole -- any of the components of the +Original Version, by changing formats or by porting the Font Software to a +new environment. + +"Author" refers to any designer, engineer, programmer, technical +writer or other person who contributed to the Font Software. + +PERMISSION & CONDITIONS +Permission is hereby granted, free of charge, to any person obtaining +a copy of the Font Software, to use, study, copy, merge, embed, modify, +redistribute, and sell modified and unmodified copies of the Font +Software, subject to the following conditions: + +1) Neither the Font Software nor any of its individual components, +in Original or Modified Versions, may be sold by itself. + +2) Original or Modified Versions of the Font Software may be bundled, +redistributed and/or sold with any software, provided that each copy +contains the above copyright notice and this license. These can be +included either as stand-alone text files, human-readable headers or +in the appropriate machine-readable metadata fields within text or +binary files as long as those fields can be easily viewed by the user. + +3) No Modified Version of the Font Software may use the Reserved Font +Name(s) unless explicit written permission is granted by the corresponding +Copyright Holder. This restriction only applies to the primary font name as +presented to the users. + +4) The name(s) of the Copyright Holder(s) or the Author(s) of the Font +Software shall not be used to promote, endorse or advertise any +Modified Version, except to acknowledge the contribution(s) of the +Copyright Holder(s) and the Author(s) or with their explicit written +permission. + +5) The Font Software, modified or unmodified, in part or in whole, +must be distributed entirely under this license, and must not be +distributed under any other license. The requirement for fonts to +remain under this license does not apply to any document created +using the Font Software. + +TERMINATION +This license becomes null and void if any of the above conditions are +not met. + +DISCLAIMER +THE FONT SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, +EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO ANY WARRANTIES OF +MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT +OF COPYRIGHT, PATENT, TRADEMARK, OR OTHER RIGHT. IN NO EVENT SHALL THE +COPYRIGHT HOLDER BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, +INCLUDING ANY GENERAL, SPECIAL, INDIRECT, INCIDENTAL, OR CONSEQUENTIAL +DAMAGES, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING +FROM, OUT OF THE USE OR INABILITY TO USE THE FONT SOFTWARE OR FROM +OTHER DEALINGS IN THE FONT SOFTWARE. diff --git a/thirdparty/fonts/NotoNaskhArabicUI_Bold.woff2 b/thirdparty/fonts/NotoNaskhArabicUI_Bold.woff2 deleted file mode 100644 index e9a834ac6907b4cd564f66d8f2f8d64005c5187f..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 53860 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 z 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 diff --git a/thirdparty/fonts/Vazirmatn_Bold.woff2 b/thirdparty/fonts/Vazirmatn_Bold.woff2 new file mode 100644 index 0000000000000000000000000000000000000000..b9b85b8549c4fe74517e6e2d5ec7da1d82563814 GIT binary patch literal 51184 zcmV)PK()VjPew8T0RR910LSnE4*&oF0po}O0LPF30RR9100000000000000000000 z0000QgDM+=sS+HxP6l8Ak9H9V3WS3&?|NsC0 z{~gIjjMacT0J^$qW>!_z-Ti|Gy0A#}2n>Tts-(&WY&(PRo{!&fVKV?X(tYB05*AgL2y|D&rDb1=}lZehuw99d8w7yg%A! zDMUa7Cy6}3hnMhQWJ&-s^_K7@L<^^TL2McN+`KVx?3+NiC)9~K#{JGwHANvzqOAA^)1fEeH7!OwGj_#~LUeFe9 zVdV`E>-9fzF$BjiKu$2zb(oNRaaYfKyULub7sLEA<1(*O!}s0WQE&lLP`lg z1V%-hZ3W=|M8?>!e+bc%oH3Y!AzGsIn(liI5prHMDNpmwDiJ9&3oP;u8#{vtZr+2o z`5z<>I1FTr3E$V=Z{2<0>@G2>?+_Y^aHv%Vhl]Qv$BOO!xS2n}!U9C))upRa6J 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 Z^oFao>rMntlLHQea6Z2d5kX047ywQhkj4N2 literal 0 HcmV?d00001 From becf5d7ac3f28e96cbb3ae83db72083707a80ed5 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Andr=C3=A9s=20Botero?= <0xafbf@gmail.com> Date: Thu, 24 Oct 2024 20:50:43 -0500 Subject: [PATCH 26/33] Fix linux builds with debug symbols when there is a space in the path. --- platform/linuxbsd/platform_linuxbsd_builders.py | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/platform/linuxbsd/platform_linuxbsd_builders.py b/platform/linuxbsd/platform_linuxbsd_builders.py index 46fa1947e87d..1298a416e0cc 100644 --- a/platform/linuxbsd/platform_linuxbsd_builders.py +++ b/platform/linuxbsd/platform_linuxbsd_builders.py @@ -5,6 +5,6 @@ def make_debug_linuxbsd(target, source, env): dst = str(target[0]) - os.system("objcopy --only-keep-debug {0} {0}.debugsymbols".format(dst)) - os.system("strip --strip-debug --strip-unneeded {0}".format(dst)) - os.system("objcopy --add-gnu-debuglink={0}.debugsymbols {0}".format(dst)) + os.system('objcopy --only-keep-debug "{0}" "{0}.debugsymbols"'.format(dst)) + os.system('strip --strip-debug --strip-unneeded "{0}"'.format(dst)) + os.system('objcopy --add-gnu-debuglink="{0}.debugsymbols" "{0}"'.format(dst)) From e0693f8ad8f432b384acf2af809c4199caf05f6f Mon Sep 17 00:00:00 2001 From: Student Main Date: Sat, 5 Oct 2024 02:24:15 +0800 Subject: [PATCH 27/33] Add loongarch64 support --- core/config/engine.cpp | 3 +++ core/os/os.cpp | 4 ++++ editor/editor_property_name_processor.cpp | 1 + editor/plugins/gdextension_export_plugin.h | 1 + platform/linuxbsd/detect.py | 2 +- .../linuxbsd/doc_classes/EditorExportPlatformLinuxBSD.xml | 2 +- platform/linuxbsd/export/export_plugin.cpp | 4 +++- platform_methods.py | 3 ++- 8 files changed, 16 insertions(+), 4 deletions(-) diff --git a/core/config/engine.cpp b/core/config/engine.cpp index aac048e93f7e..250f39b08a32 100644 --- a/core/config/engine.cpp +++ b/core/config/engine.cpp @@ -248,6 +248,9 @@ String Engine::get_architecture_name() const { return "ppc"; #endif +#elif defined(__loongarch64) + return "loongarch64"; + #elif defined(__wasm__) #if defined(__wasm64__) return "wasm64"; diff --git a/core/os/os.cpp b/core/os/os.cpp index 4a833645f06e..ff713ab3027a 100644 --- a/core/os/os.cpp +++ b/core/os/os.cpp @@ -518,6 +518,10 @@ bool OS::has_feature(const String &p_feature) { if (p_feature == "wasm") { return true; } +#elif defined(__loongarch64) + if (p_feature == "loongarch64") { + return true; + } #endif #if defined(IOS_SIMULATOR) diff --git a/editor/editor_property_name_processor.cpp b/editor/editor_property_name_processor.cpp index ca8854f79702..3560174e6b51 100644 --- a/editor/editor_property_name_processor.cpp +++ b/editor/editor_property_name_processor.cpp @@ -235,6 +235,7 @@ EditorPropertyNameProcessor::EditorPropertyNameProcessor() { capitalize_string_remaps["linuxbsd"] = "Linux/*BSD"; capitalize_string_remaps["lod"] = "LOD"; capitalize_string_remaps["lods"] = "LODs"; + capitalize_string_remaps["loongarch64"] = "loongarch64"; capitalize_string_remaps["lowpass"] = "Low-pass"; capitalize_string_remaps["macos"] = "macOS"; capitalize_string_remaps["mb"] = "(MB)"; // Unit. diff --git a/editor/plugins/gdextension_export_plugin.h b/editor/plugins/gdextension_export_plugin.h index c8ed05c6b70e..9189441c2bae 100644 --- a/editor/plugins/gdextension_export_plugin.h +++ b/editor/plugins/gdextension_export_plugin.h @@ -73,6 +73,7 @@ void GDExtensionExportPlugin::_export_file(const String &p_path, const String &p all_archs.insert("ppc32"); all_archs.insert("ppc64"); all_archs.insert("wasm32"); + all_archs.insert("loongarch64"); all_archs.insert("universal"); HashSet archs; diff --git a/platform/linuxbsd/detect.py b/platform/linuxbsd/detect.py index c8202b147d25..6c9f33684901 100644 --- a/platform/linuxbsd/detect.py +++ b/platform/linuxbsd/detect.py @@ -73,7 +73,7 @@ def get_flags(): def configure(env: "SConsEnvironment"): # Validate arch. - supported_arches = ["x86_32", "x86_64", "arm32", "arm64", "rv64", "ppc32", "ppc64"] + supported_arches = ["x86_32", "x86_64", "arm32", "arm64", "rv64", "ppc32", "ppc64", "loongarch64"] validate_arch(env["arch"], get_name(), supported_arches) ## Build type diff --git a/platform/linuxbsd/doc_classes/EditorExportPlatformLinuxBSD.xml b/platform/linuxbsd/doc_classes/EditorExportPlatformLinuxBSD.xml index a44c86202e2d..7b50950646da 100644 --- a/platform/linuxbsd/doc_classes/EditorExportPlatformLinuxBSD.xml +++ b/platform/linuxbsd/doc_classes/EditorExportPlatformLinuxBSD.xml @@ -11,7 +11,7 @@ Application executable architecture. - Supported architectures: [code]x86_32[/code], [code]x86_64[/code], [code]arm64[/code], [code]arm32[/code], [code]rv64[/code], [code]ppc64[/code], and [code]ppc32[/code]. + Supported architectures: [code]x86_32[/code], [code]x86_64[/code], [code]arm64[/code], [code]arm32[/code], [code]rv64[/code], [code]ppc64[/code], [code]ppc32[/code], and [code]loongarch64[/code]. Official export templates include [code]x86_32[/code] and [code]x86_64[/code] binaries only. diff --git a/platform/linuxbsd/export/export_plugin.cpp b/platform/linuxbsd/export/export_plugin.cpp index 69ba742f72b1..7cd77dd5937c 100644 --- a/platform/linuxbsd/export/export_plugin.cpp +++ b/platform/linuxbsd/export/export_plugin.cpp @@ -180,7 +180,7 @@ bool EditorExportPlatformLinuxBSD::get_export_option_visibility(const EditorExpo void EditorExportPlatformLinuxBSD::get_export_options(List *r_options) const { EditorExportPlatformPC::get_export_options(r_options); - r_options->push_back(ExportOption(PropertyInfo(Variant::STRING, "binary_format/architecture", PROPERTY_HINT_ENUM, "x86_64,x86_32,arm64,arm32,rv64,ppc64,ppc32"), "x86_64")); + r_options->push_back(ExportOption(PropertyInfo(Variant::STRING, "binary_format/architecture", PROPERTY_HINT_ENUM, "x86_64,x86_32,arm64,arm32,rv64,ppc64,ppc32,loongarch64"), "x86_64")); String run_script = "#!/usr/bin/env bash\n" "export DISPLAY=:0\n" @@ -282,6 +282,8 @@ String EditorExportPlatformLinuxBSD::_get_exe_arch(const String &p_path) const { return "arm64"; case 0x00f3: return "rv64"; + case 0x0102: + return "loongarch64"; default: return "unknown"; } diff --git a/platform_methods.py b/platform_methods.py index 201df3c0b547..c8646a402267 100644 --- a/platform_methods.py +++ b/platform_methods.py @@ -16,7 +16,7 @@ } # CPU architecture options. -architectures = ["x86_32", "x86_64", "arm32", "arm64", "rv64", "ppc32", "ppc64", "wasm32"] +architectures = ["x86_32", "x86_64", "arm32", "arm64", "rv64", "ppc32", "ppc64", "wasm32", "loongarch64"] architecture_aliases = { "x86": "x86_32", "x64": "x86_64", @@ -31,6 +31,7 @@ "ppcle": "ppc32", "ppc": "ppc32", "ppc64le": "ppc64", + "loong64": "loongarch64", } From f39614aa5aa0bae2355e3cab13f8f50b58cddc58 Mon Sep 17 00:00:00 2001 From: Student Main Date: Sat, 5 Oct 2024 03:56:14 +0800 Subject: [PATCH 28/33] Ignore case when parse /proc/cpuinfo --- platform/linuxbsd/os_linuxbsd.cpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/platform/linuxbsd/os_linuxbsd.cpp b/platform/linuxbsd/os_linuxbsd.cpp index b309e8d8ebb7..9fba97d552a0 100644 --- a/platform/linuxbsd/os_linuxbsd.cpp +++ b/platform/linuxbsd/os_linuxbsd.cpp @@ -184,7 +184,7 @@ String OS_LinuxBSD::get_processor_name() const { while (!f->eof_reached()) { const String line = f->get_line(); - if (line.contains("model name")) { + if (line.to_lower().contains("model name")) { return line.split(":")[1].strip_edges(); } } From afd68d785b7148ed25b6d48d28072e7827e43ece Mon Sep 17 00:00:00 2001 From: DeeJayLSP Date: Wed, 4 Dec 2024 00:53:14 -0300 Subject: [PATCH 29/33] Use `qoa.c` and custom compress procedure --- scene/resources/SCsub | 8 ++- scene/resources/audio_stream_wav.cpp | 44 ++++++--------- scene/resources/audio_stream_wav.h | 31 +++++++++-- servers/audio/effects/audio_effect_record.cpp | 6 +++ thirdparty/README.md | 4 +- thirdparty/misc/patches/qoa-min-fix.patch | 53 ------------------- thirdparty/misc/qoa.c | 4 ++ thirdparty/misc/qoa.h | 24 +++++---- 8 files changed, 76 insertions(+), 98 deletions(-) delete mode 100644 thirdparty/misc/patches/qoa-min-fix.patch create mode 100644 thirdparty/misc/qoa.c diff --git a/scene/resources/SCsub b/scene/resources/SCsub index 46f6251b9131..ae2a9b8c3a79 100644 --- a/scene/resources/SCsub +++ b/scene/resources/SCsub @@ -7,7 +7,13 @@ Import("env") thirdparty_obj = [] -thirdparty_sources = "#thirdparty/misc/mikktspace.c" +thirdparty_dir = "#thirdparty/misc/" +thirdparty_sources = [ + "mikktspace.c", + "qoa.c" +] + +thirdparty_sources = [thirdparty_dir + file for file in thirdparty_sources] env_thirdparty = env.Clone() env_thirdparty.disable_warnings() diff --git a/scene/resources/audio_stream_wav.cpp b/scene/resources/audio_stream_wav.cpp index cea9af729eb0..15e3f0f806b5 100644 --- a/scene/resources/audio_stream_wav.cpp +++ b/scene/resources/audio_stream_wav.cpp @@ -1142,13 +1142,13 @@ Ref AudioStreamWAV::load_from_buffer(const Vector &p_fi is16 = false; } - Vector pcm_data; + Vector dst_data; AudioStreamWAV::Format dst_format; if (compression == 1) { dst_format = AudioStreamWAV::FORMAT_IMA_ADPCM; if (format_channels == 1) { - _compress_ima_adpcm(data, pcm_data); + _compress_ima_adpcm(data, dst_data); } else { //byte interleave Vector left; @@ -1170,9 +1170,9 @@ Ref AudioStreamWAV::load_from_buffer(const Vector &p_fi _compress_ima_adpcm(right, bright); int dl = bleft.size(); - pcm_data.resize(dl * 2); + dst_data.resize(dl * 2); - uint8_t *w = pcm_data.ptrw(); + uint8_t *w = dst_data.ptrw(); const uint8_t *rl = bleft.ptr(); const uint8_t *rr = bright.ptr(); @@ -1182,16 +1182,24 @@ Ref AudioStreamWAV::load_from_buffer(const Vector &p_fi } } + } else if (compression == 2) { + dst_format = AudioStreamWAV::FORMAT_QOA; + + qoa_desc desc = {}; + desc.samplerate = rate; + desc.samples = frames; + desc.channels = format_channels; + + _compress_qoa(data, dst_data, &desc); } else { dst_format = is16 ? AudioStreamWAV::FORMAT_16_BITS : AudioStreamWAV::FORMAT_8_BITS; - bool enforce16 = is16 || compression == 2; - pcm_data.resize(data.size() * (enforce16 ? 2 : 1)); + dst_data.resize(data.size() * (is16 ? 2 : 1)); { - uint8_t *w = pcm_data.ptrw(); + uint8_t *w = dst_data.ptrw(); int ds = data.size(); for (int i = 0; i < ds; i++) { - if (enforce16) { + if (is16) { int16_t v = CLAMP(data[i] * 32768, -32768, 32767); encode_uint16(v, &w[i * 2]); } else { @@ -1202,26 +1210,6 @@ Ref AudioStreamWAV::load_from_buffer(const Vector &p_fi } } - Vector dst_data; - if (compression == 2) { - dst_format = AudioStreamWAV::FORMAT_QOA; - qoa_desc desc = {}; - uint32_t qoa_len = 0; - - desc.samplerate = rate; - desc.samples = frames; - desc.channels = format_channels; - - void *encoded = qoa_encode((short *)pcm_data.ptr(), &desc, &qoa_len); - if (encoded) { - dst_data.resize(qoa_len); - memcpy(dst_data.ptrw(), encoded, qoa_len); - QOA_FREE(encoded); - } - } else { - dst_data = pcm_data; - } - Ref sample; sample.instantiate(); sample->set_data(dst_data); diff --git a/scene/resources/audio_stream_wav.h b/scene/resources/audio_stream_wav.h index 269ab1e05f12..e36d33cfa933 100644 --- a/scene/resources/audio_stream_wav.h +++ b/scene/resources/audio_stream_wav.h @@ -31,9 +31,6 @@ #ifndef AUDIO_STREAM_WAV_H #define AUDIO_STREAM_WAV_H -#define QOA_IMPLEMENTATION -#define QOA_NO_STDIO - #include "servers/audio/audio_stream.h" #include "thirdparty/misc/qoa.h" @@ -273,6 +270,34 @@ class AudioStreamWAV : public AudioStream { } } + static void _compress_qoa(const Vector &p_data, Vector &dst_data, qoa_desc *p_desc) { + uint32_t frames_len = (p_desc->samples + QOA_FRAME_LEN - 1) / QOA_FRAME_LEN * (QOA_LMS_LEN * 4 * p_desc->channels + 8); + uint32_t slices_len = (p_desc->samples + QOA_SLICE_LEN - 1) / QOA_SLICE_LEN * 8 * p_desc->channels; + dst_data.resize(8 + frames_len + slices_len); + + for (uint32_t c = 0; c < p_desc->channels; c++) { + memset(p_desc->lms[c].history, 0, sizeof(p_desc->lms[c].history)); + memset(p_desc->lms[c].weights, 0, sizeof(p_desc->lms[c].weights)); + p_desc->lms[c].weights[2] = -(1 << 13); + p_desc->lms[c].weights[3] = (1 << 14); + } + + LocalVector data16; + data16.resize(QOA_FRAME_LEN * p_desc->channels); + + uint8_t *dst_ptr = dst_data.ptrw(); + dst_ptr += qoa_encode_header(p_desc, dst_data.ptrw()); + + uint32_t frame_len = QOA_FRAME_LEN; + for (uint32_t s = 0; s < p_desc->samples; s += frame_len) { + frame_len = MIN(frame_len, p_desc->samples - s); + for (uint32_t i = 0; i < frame_len * p_desc->channels; i++) { + data16[i] = CLAMP(p_data[s * p_desc->channels + i] * 32767.0, -32768, 32767); + } + dst_ptr += qoa_encode_frame(data16.ptr(), p_desc, frame_len, dst_ptr); + } + } + AudioStreamWAV(); ~AudioStreamWAV(); }; diff --git a/servers/audio/effects/audio_effect_record.cpp b/servers/audio/effects/audio_effect_record.cpp index f82a6fa3afb2..b6bc5d00c4fe 100644 --- a/servers/audio/effects/audio_effect_record.cpp +++ b/servers/audio/effects/audio_effect_record.cpp @@ -250,6 +250,12 @@ Ref AudioEffectRecord::get_recording() const { w[i * 2 + 0] = rl[i]; w[i * 2 + 1] = rr[i]; } + } else if (dst_format == AudioStreamWAV::FORMAT_QOA) { + qoa_desc desc = {}; + desc.samples = current_instance->recording_data.size() / 2; + desc.samplerate = AudioServer::get_singleton()->get_mix_rate(); + desc.channels = 2; + AudioStreamWAV::_compress_qoa(current_instance->recording_data, dst_data, &desc); } else { ERR_PRINT("Format not implemented."); } diff --git a/thirdparty/README.md b/thirdparty/README.md index 1e11c6c9725b..7031a9ddf333 100644 --- a/thirdparty/README.md +++ b/thirdparty/README.md @@ -716,8 +716,8 @@ Collection of single-file libraries used in Godot components. * License: MIT - `qoa.h` * Upstream: https://github.com/phoboslab/qoa - * Version: git (e0c69447d4d3945c3c92ac1751e4cdc9803a8303, 2024) - * Modifications: Added a few modifiers to comply with C++ nature. + * Version: git (a2d927f8ce78a85e903676a33e0f956e53b89f7d, 2024) + * Modifications: Added implementation through `qoa.c`. * License: MIT - `r128.{c,h}` * Upstream: https://github.com/fahickman/r128 diff --git a/thirdparty/misc/patches/qoa-min-fix.patch b/thirdparty/misc/patches/qoa-min-fix.patch deleted file mode 100644 index 6008b5f8bcdc..000000000000 --- a/thirdparty/misc/patches/qoa-min-fix.patch +++ /dev/null @@ -1,53 +0,0 @@ -diff --git a/qoa.h b/qoa.h -index cfed266bef..23612bb0bf 100644 ---- a/qoa.h -+++ b/qoa.h -@@ -140,14 +140,14 @@ typedef struct { - #endif - } qoa_desc; - --unsigned int qoa_encode_header(qoa_desc *qoa, unsigned char *bytes); --unsigned int qoa_encode_frame(const short *sample_data, qoa_desc *qoa, unsigned int frame_len, unsigned char *bytes); --void *qoa_encode(const short *sample_data, qoa_desc *qoa, unsigned int *out_len); -+inline unsigned int qoa_encode_header(qoa_desc *qoa, unsigned char *bytes); -+inline unsigned int qoa_encode_frame(const short *sample_data, qoa_desc *qoa, unsigned int frame_len, unsigned char *bytes); -+inline void *qoa_encode(const short *sample_data, qoa_desc *qoa, unsigned int *out_len); - --unsigned int qoa_max_frame_size(qoa_desc *qoa); --unsigned int qoa_decode_header(const unsigned char *bytes, int size, qoa_desc *qoa); --unsigned int qoa_decode_frame(const unsigned char *bytes, unsigned int size, qoa_desc *qoa, short *sample_data, unsigned int *frame_len); --short *qoa_decode(const unsigned char *bytes, int size, qoa_desc *file); -+inline unsigned int qoa_max_frame_size(qoa_desc *qoa); -+inline unsigned int qoa_decode_header(const unsigned char *bytes, int size, qoa_desc *qoa); -+inline unsigned int qoa_decode_frame(const unsigned char *bytes, unsigned int size, qoa_desc *qoa, short *sample_data, unsigned int *frame_len); -+inline short *qoa_decode(const unsigned char *bytes, int size, qoa_desc *file); - - #ifndef QOA_NO_STDIO - -@@ -395,7 +395,7 @@ unsigned int qoa_encode_frame(const short *sample_data, qoa_desc *qoa, unsigned - qoa_uint64_t best_error = -1; - #endif - qoa_uint64_t best_slice = 0; -- qoa_lms_t best_lms; -+ qoa_lms_t best_lms = {}; - int best_scalefactor = 0; - - for (int sfi = 0; sfi < 16; sfi++) { -@@ -500,7 +500,7 @@ void *qoa_encode(const short *sample_data, qoa_desc *qoa, unsigned int *out_len) - num_frames * QOA_LMS_LEN * 4 * qoa->channels + /* 4 * 4 bytes lms state per channel */ - num_slices * 8 * qoa->channels; /* 8 byte slices */ - -- unsigned char *bytes = QOA_MALLOC(encoded_size); -+ unsigned char *bytes = (unsigned char *)QOA_MALLOC(encoded_size); - - for (unsigned int c = 0; c < qoa->channels; c++) { - /* Set the initial LMS weights to {0, 0, -1, 2}. This helps with the -@@ -655,7 +655,7 @@ short *qoa_decode(const unsigned char *bytes, int size, qoa_desc *qoa) { - - /* Calculate the required size of the sample buffer and allocate */ - int total_samples = qoa->samples * qoa->channels; -- short *sample_data = QOA_MALLOC(total_samples * sizeof(short)); -+ short *sample_data = (short *)QOA_MALLOC(total_samples * sizeof(short)); - - unsigned int sample_index = 0; - unsigned int frame_len; diff --git a/thirdparty/misc/qoa.c b/thirdparty/misc/qoa.c new file mode 100644 index 000000000000..7f7d366dfa1c --- /dev/null +++ b/thirdparty/misc/qoa.c @@ -0,0 +1,4 @@ +#define QOA_IMPLEMENTATION +#define QOA_NO_STDIO + +#include "qoa.h" diff --git a/thirdparty/misc/qoa.h b/thirdparty/misc/qoa.h index 23612bb0bfb4..f0f44214d813 100644 --- a/thirdparty/misc/qoa.h +++ b/thirdparty/misc/qoa.h @@ -140,14 +140,14 @@ typedef struct { #endif } qoa_desc; -inline unsigned int qoa_encode_header(qoa_desc *qoa, unsigned char *bytes); -inline unsigned int qoa_encode_frame(const short *sample_data, qoa_desc *qoa, unsigned int frame_len, unsigned char *bytes); -inline void *qoa_encode(const short *sample_data, qoa_desc *qoa, unsigned int *out_len); +unsigned int qoa_encode_header(qoa_desc *qoa, unsigned char *bytes); +unsigned int qoa_encode_frame(const short *sample_data, qoa_desc *qoa, unsigned int frame_len, unsigned char *bytes); +void *qoa_encode(const short *sample_data, qoa_desc *qoa, unsigned int *out_len); -inline unsigned int qoa_max_frame_size(qoa_desc *qoa); -inline unsigned int qoa_decode_header(const unsigned char *bytes, int size, qoa_desc *qoa); -inline unsigned int qoa_decode_frame(const unsigned char *bytes, unsigned int size, qoa_desc *qoa, short *sample_data, unsigned int *frame_len); -inline short *qoa_decode(const unsigned char *bytes, int size, qoa_desc *file); +unsigned int qoa_max_frame_size(qoa_desc *qoa); +unsigned int qoa_decode_header(const unsigned char *bytes, int size, qoa_desc *qoa); +unsigned int qoa_decode_frame(const unsigned char *bytes, unsigned int size, qoa_desc *qoa, short *sample_data, unsigned int *frame_len); +short *qoa_decode(const unsigned char *bytes, int size, qoa_desc *file); #ifndef QOA_NO_STDIO @@ -395,7 +395,7 @@ unsigned int qoa_encode_frame(const short *sample_data, qoa_desc *qoa, unsigned qoa_uint64_t best_error = -1; #endif qoa_uint64_t best_slice = 0; - qoa_lms_t best_lms = {}; + qoa_lms_t best_lms; int best_scalefactor = 0; for (int sfi = 0; sfi < 16; sfi++) { @@ -500,7 +500,7 @@ void *qoa_encode(const short *sample_data, qoa_desc *qoa, unsigned int *out_len) num_frames * QOA_LMS_LEN * 4 * qoa->channels + /* 4 * 4 bytes lms state per channel */ num_slices * 8 * qoa->channels; /* 8 byte slices */ - unsigned char *bytes = (unsigned char *)QOA_MALLOC(encoded_size); + unsigned char *bytes = QOA_MALLOC(encoded_size); for (unsigned int c = 0; c < qoa->channels; c++) { /* Set the initial LMS weights to {0, 0, -1, 2}. This helps with the @@ -626,12 +626,14 @@ unsigned int qoa_decode_frame(const unsigned char *bytes, unsigned int size, qoa qoa_uint64_t slice = qoa_read_u64(bytes, &p); int scalefactor = (slice >> 60) & 0xf; + slice <<= 4; + int slice_start = sample_index * channels + c; int slice_end = qoa_clamp(sample_index + QOA_SLICE_LEN, 0, samples) * channels + c; for (int si = slice_start; si < slice_end; si += channels) { int predicted = qoa_lms_predict(&qoa->lms[c]); - int quantized = (slice >> 57) & 0x7; + int quantized = (slice >> 61) & 0x7; int dequantized = qoa_dequant_tab[scalefactor][quantized]; int reconstructed = qoa_clamp_s16(predicted + dequantized); @@ -655,7 +657,7 @@ short *qoa_decode(const unsigned char *bytes, int size, qoa_desc *qoa) { /* Calculate the required size of the sample buffer and allocate */ int total_samples = qoa->samples * qoa->channels; - short *sample_data = (short *)QOA_MALLOC(total_samples * sizeof(short)); + short *sample_data = QOA_MALLOC(total_samples * sizeof(short)); unsigned int sample_index = 0; unsigned int frame_len; From 6bc80effbb9302c9aa14e0006826fc91fd863e30 Mon Sep 17 00:00:00 2001 From: MBCX <63003002+MBCX@users.noreply.github.com> Date: Thu, 5 Dec 2024 10:04:25 -0400 Subject: [PATCH 30/33] Make Godot compile on FreeBSD --- modules/camera/config.py | 4 ++++ modules/openxr/SCsub | 6 +++++- platform/linuxbsd/wayland/wayland_thread.cpp | 6 +++++- thirdparty/linuxbsd_headers/alsa/asoundlib.h | 4 ++++ .../alsa/patches/freebsd_endian.diff | 16 ++++++++++++++++ 5 files changed, 34 insertions(+), 2 deletions(-) create mode 100644 thirdparty/linuxbsd_headers/alsa/patches/freebsd_endian.diff diff --git a/modules/camera/config.py b/modules/camera/config.py index 7b368d219368..fa229ef2f5a8 100644 --- a/modules/camera/config.py +++ b/modules/camera/config.py @@ -1,4 +1,8 @@ def can_build(env, platform): + import sys + + if sys.platform.startswith("freebsd"): + return False return platform == "macos" or platform == "windows" or platform == "linuxbsd" diff --git a/modules/openxr/SCsub b/modules/openxr/SCsub index 48c87bcd59cd..edfe0bbfad6f 100644 --- a/modules/openxr/SCsub +++ b/modules/openxr/SCsub @@ -1,6 +1,8 @@ #!/usr/bin/env python from misc.utility.scons_hints import * +import sys + Import("env") Import("env_modules") @@ -29,7 +31,9 @@ elif env["platform"] == "linuxbsd": env_openxr.AppendUnique(CPPDEFINES=["XR_USE_PLATFORM_EGL"]) # FIXME: Review what needs to be set for Android and macOS. - env_openxr.AppendUnique(CPPDEFINES=["HAVE_SECURE_GETENV"]) + # FreeBSD uses non-standard getenv functions. + if not sys.platform.startswith("freebsd"): + env_openxr.AppendUnique(CPPDEFINES=["HAVE_SECURE_GETENV"]) elif env["platform"] == "windows": env_openxr.AppendUnique(CPPDEFINES=["XR_OS_WINDOWS", "NOMINMAX", "XR_USE_PLATFORM_WIN32"]) elif env["platform"] == "macos": diff --git a/platform/linuxbsd/wayland/wayland_thread.cpp b/platform/linuxbsd/wayland/wayland_thread.cpp index f1aac4f0696c..8e3dc7558184 100644 --- a/platform/linuxbsd/wayland/wayland_thread.cpp +++ b/platform/linuxbsd/wayland/wayland_thread.cpp @@ -32,8 +32,12 @@ #ifdef WAYLAND_ENABLED -// FIXME: Does this cause issues with *BSDs? +#ifdef __FreeBSD__ +#include +#else +// Assume Linux. #include +#endif // For the actual polling thread. #include diff --git a/thirdparty/linuxbsd_headers/alsa/asoundlib.h b/thirdparty/linuxbsd_headers/alsa/asoundlib.h index a5461943825c..598175403cda 100644 --- a/thirdparty/linuxbsd_headers/alsa/asoundlib.h +++ b/thirdparty/linuxbsd_headers/alsa/asoundlib.h @@ -38,7 +38,11 @@ #include #include #include +#ifdef __FreeBSD__ +#include +#else #include +#endif // __FreeBSD__ #ifndef __GNUC__ #define __inline__ inline diff --git a/thirdparty/linuxbsd_headers/alsa/patches/freebsd_endian.diff b/thirdparty/linuxbsd_headers/alsa/patches/freebsd_endian.diff new file mode 100644 index 000000000000..f104d9df8547 --- /dev/null +++ b/thirdparty/linuxbsd_headers/alsa/patches/freebsd_endian.diff @@ -0,0 +1,16 @@ +diff --git a/thirdparty/linuxbsd_headers/alsa/asoundlib.h b/thirdparty/linuxbsd_headers/alsa/asoundlib.h +index a546194382..598175403c 100644 +--- a/thirdparty/linuxbsd_headers/alsa/asoundlib.h ++++ b/thirdparty/linuxbsd_headers/alsa/asoundlib.h +@@ -38,7 +38,11 @@ + #include + #include + #include ++#ifdef __FreeBSD__ ++#include ++#else + #include ++#endif // __FreeBSD__ + + #ifndef __GNUC__ + #define __inline__ inline From 329d25b1fafccf5e8059c494b6210f1805b0b4bb Mon Sep 17 00:00:00 2001 From: Rudolph Bester Date: Thu, 5 Dec 2024 18:54:13 +0200 Subject: [PATCH 31/33] Fixed occlusion culling buffer getting overwritten in larger scenes --- modules/raycast/raycast_occlusion_cull.cpp | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/modules/raycast/raycast_occlusion_cull.cpp b/modules/raycast/raycast_occlusion_cull.cpp index b136d54520a1..fb2d0023b60f 100644 --- a/modules/raycast/raycast_occlusion_cull.cpp +++ b/modules/raycast/raycast_occlusion_cull.cpp @@ -366,7 +366,7 @@ void RaycastOcclusionCull::Scenario::_transform_vertices_thread(uint32_t p_threa } void RaycastOcclusionCull::Scenario::_transform_vertices_range(const Vector3 *p_read, float *p_write, const Transform3D &p_xform, int p_from, int p_to) { - float *floats_w = p_write; + float *floats_w = p_write + 3 * p_from; for (int i = p_from; i < p_to; i++) { const Vector3 p = p_xform.xform(p_read[i]); floats_w[0] = p.x; From d7d6251c30a9d54da6842d23e3bd6db4df8f8d8b Mon Sep 17 00:00:00 2001 From: Dario Date: Thu, 5 Dec 2024 13:55:57 -0300 Subject: [PATCH 32/33] Add texture limits for D3D12 Driver. --- drivers/d3d12/rendering_device_driver_d3d12.cpp | 10 ++++++++++ 1 file changed, 10 insertions(+) diff --git a/drivers/d3d12/rendering_device_driver_d3d12.cpp b/drivers/d3d12/rendering_device_driver_d3d12.cpp index b72a1932f830..f78233b9b33f 100644 --- a/drivers/d3d12/rendering_device_driver_d3d12.cpp +++ b/drivers/d3d12/rendering_device_driver_d3d12.cpp @@ -6144,6 +6144,16 @@ uint64_t RenderingDeviceDriverD3D12::limit_get(Limit p_limit) { switch (p_limit) { case LIMIT_MAX_BOUND_UNIFORM_SETS: return safe_unbounded; + case LIMIT_MAX_TEXTURE_ARRAY_LAYERS: + return D3D12_REQ_TEXTURE2D_ARRAY_AXIS_DIMENSION; + case LIMIT_MAX_TEXTURE_SIZE_1D: + return D3D12_REQ_TEXTURE1D_U_DIMENSION; + case LIMIT_MAX_TEXTURE_SIZE_2D: + return D3D12_REQ_TEXTURE2D_U_OR_V_DIMENSION; + case LIMIT_MAX_TEXTURE_SIZE_3D: + return D3D12_REQ_TEXTURE3D_U_V_OR_W_DIMENSION; + case LIMIT_MAX_TEXTURE_SIZE_CUBE: + return D3D12_REQ_TEXTURECUBE_DIMENSION; case LIMIT_MAX_TEXTURES_PER_SHADER_STAGE: return device_limits.max_srvs_per_shader_stage; case LIMIT_MAX_UNIFORM_BUFFER_SIZE: From 1b5a15d5d1696099bf06bee5775606a7a50f3267 Mon Sep 17 00:00:00 2001 From: clayjohn Date: Thu, 5 Dec 2024 11:08:59 -0800 Subject: [PATCH 33/33] Correctly check time since shadow was allocated in atlas to avoid unnecessary re-allocations Co-authored-by: jitspoe --- drivers/gles3/storage/light_storage.cpp | 2 +- servers/rendering/renderer_rd/storage_rd/light_storage.cpp | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/drivers/gles3/storage/light_storage.cpp b/drivers/gles3/storage/light_storage.cpp index 9b81430d45c0..886918a2f76e 100644 --- a/drivers/gles3/storage/light_storage.cpp +++ b/drivers/gles3/storage/light_storage.cpp @@ -1405,7 +1405,7 @@ bool LightStorage::shadow_atlas_update_light(RID p_atlas, RID p_light_instance, old_shadow = old_key & SHADOW_INDEX_MASK; // Only re-allocate if a better option is available, and enough time has passed. - should_realloc = shadow_atlas->quadrants[old_quadrant].subdivision != (uint32_t)best_subdiv && (shadow_atlas->quadrants[old_quadrant].shadows[old_shadow].alloc_tick - tick > shadow_atlas_realloc_tolerance_msec); + should_realloc = shadow_atlas->quadrants[old_quadrant].subdivision != (uint32_t)best_subdiv && (tick - shadow_atlas->quadrants[old_quadrant].shadows[old_shadow].alloc_tick > shadow_atlas_realloc_tolerance_msec); should_redraw = shadow_atlas->quadrants[old_quadrant].shadows[old_shadow].version != p_light_version; if (!should_realloc) { diff --git a/servers/rendering/renderer_rd/storage_rd/light_storage.cpp b/servers/rendering/renderer_rd/storage_rd/light_storage.cpp index 9de37050f0a7..1d309a80063d 100644 --- a/servers/rendering/renderer_rd/storage_rd/light_storage.cpp +++ b/servers/rendering/renderer_rd/storage_rd/light_storage.cpp @@ -2328,7 +2328,7 @@ bool LightStorage::shadow_atlas_update_light(RID p_atlas, RID p_light_instance, old_quadrant = (old_key >> QUADRANT_SHIFT) & 0x3; old_shadow = old_key & SHADOW_INDEX_MASK; - should_realloc = shadow_atlas->quadrants[old_quadrant].subdivision != (uint32_t)best_subdiv && (shadow_atlas->quadrants[old_quadrant].shadows[old_shadow].alloc_tick - tick > shadow_atlas_realloc_tolerance_msec); + should_realloc = shadow_atlas->quadrants[old_quadrant].subdivision != (uint32_t)best_subdiv && (tick - shadow_atlas->quadrants[old_quadrant].shadows[old_shadow].alloc_tick > shadow_atlas_realloc_tolerance_msec); should_redraw = shadow_atlas->quadrants[old_quadrant].shadows[old_shadow].version != p_light_version; if (!should_realloc) {