From 49478e7f5c558480b4dff01cc6ff1c7711f24914 Mon Sep 17 00:00:00 2001 From: Sander Mertens Date: Fri, 15 Sep 2023 13:09:34 -0700 Subject: [PATCH] Fix double free issue in C++ when iterating terms with allocated resources --- flecs.h | 2 ++ .../flecs/addons/cpp/mixins/filter/impl.hpp | 1 + .../flecs/addons/cpp/mixins/query/impl.hpp | 1 + test/cpp_api/project.json | 7 +++++-- test/cpp_api/src/Filter.cpp | 16 ++++++++++++++ test/cpp_api/src/Query.cpp | 15 +++++++++++++ test/cpp_api/src/RuleBuilder.cpp | 16 ++++++++++++++ test/cpp_api/src/main.cpp | 21 ++++++++++++++++--- 8 files changed, 74 insertions(+), 5 deletions(-) diff --git a/flecs.h b/flecs.h index 46e03fbcfb..16fa76c159 100644 --- a/flecs.h +++ b/flecs.h @@ -27512,6 +27512,7 @@ struct filter_base { for (int i = 0; i < m_filter_ptr->term_count; i ++) { flecs::term t(m_world, m_filter_ptr->terms[i]); func(t); + t.reset(); // prevent freeing resources } } @@ -28001,6 +28002,7 @@ struct query_base { for (int i = 0; i < f->term_count; i ++) { flecs::term t(m_world, f->terms[i]); func(t); + t.reset(); // prevent freeing resources } } diff --git a/include/flecs/addons/cpp/mixins/filter/impl.hpp b/include/flecs/addons/cpp/mixins/filter/impl.hpp index cf30c9a202..33b7260d54 100644 --- a/include/flecs/addons/cpp/mixins/filter/impl.hpp +++ b/include/flecs/addons/cpp/mixins/filter/impl.hpp @@ -106,6 +106,7 @@ struct filter_base { for (int i = 0; i < m_filter_ptr->term_count; i ++) { flecs::term t(m_world, m_filter_ptr->terms[i]); func(t); + t.reset(); // prevent freeing resources } } diff --git a/include/flecs/addons/cpp/mixins/query/impl.hpp b/include/flecs/addons/cpp/mixins/query/impl.hpp index 7f7f3c2367..4f43b362d3 100644 --- a/include/flecs/addons/cpp/mixins/query/impl.hpp +++ b/include/flecs/addons/cpp/mixins/query/impl.hpp @@ -103,6 +103,7 @@ struct query_base { for (int i = 0; i < f->term_count; i ++) { flecs::term t(m_world, f->terms[i]); func(t); + t.reset(); // prevent freeing resources } } diff --git a/test/cpp_api/project.json b/test/cpp_api/project.json index f20b49eb7f..4aac59ca2c 100644 --- a/test/cpp_api/project.json +++ b/test/cpp_api/project.json @@ -539,6 +539,7 @@ "test_no_defer_iter", "inspect_terms", "inspect_terms_w_each", + "inspect_terms_w_expr", "comp_to_str", "pair_to_str", "oper_not_to_str", @@ -806,7 +807,8 @@ "is_valid", "unresolved_by_name", "scope", - "iter_w_stage" + "iter_w_stage", + "inspect_terms_w_expr" ] }, { "id": "SystemBuilder", @@ -892,7 +894,8 @@ "invalid_each_w_no_this", "named_filter", "named_scoped_filter", - "set_this_var" + "set_this_var", + "inspect_terms_w_expr" ] }, { "id": "ComponentLifecycle", diff --git a/test/cpp_api/src/Filter.cpp b/test/cpp_api/src/Filter.cpp index 90a032d8a5..4ab875f3e2 100644 --- a/test/cpp_api/src/Filter.cpp +++ b/test/cpp_api/src/Filter.cpp @@ -456,3 +456,19 @@ void Filter_set_this_var(void) { }); test_int(count, 1); } + +void Filter_inspect_terms_w_expr(void) { + flecs::world ecs; + + flecs::filter<> f = ecs.filter_builder() + .expr("(ChildOf,0)") + .build(); + + int32_t count = 0; + f.each_term([&](flecs::term &term) { + test_assert(term.id().is_pair()); + count ++; + }); + + test_int(count, 1); +} diff --git a/test/cpp_api/src/Query.cpp b/test/cpp_api/src/Query.cpp index 2ed6301872..dcfea33dd7 100644 --- a/test/cpp_api/src/Query.cpp +++ b/test/cpp_api/src/Query.cpp @@ -828,6 +828,21 @@ void Query_inspect_terms_w_each(void) { test_int(count, 3); } +void Query_inspect_terms_w_expr(void) { + flecs::world ecs; + + flecs::filter<> f = ecs.filter_builder() + .expr("(ChildOf,0)") + .build(); + + int32_t count = 0; + f.each_term([&](flecs::term &term) { + test_assert(term.id().is_pair()); + count ++; + }); + + test_int(count, 1); +} void Query_comp_to_str(void) { flecs::world ecs; diff --git a/test/cpp_api/src/RuleBuilder.cpp b/test/cpp_api/src/RuleBuilder.cpp index b4da91ce41..bcd396f8ee 100644 --- a/test/cpp_api/src/RuleBuilder.cpp +++ b/test/cpp_api/src/RuleBuilder.cpp @@ -797,3 +797,19 @@ void RuleBuilder_iter_w_stage(void) { q.destruct(); } + +void RuleBuilder_inspect_terms_w_expr(void) { + flecs::world ecs; + + flecs::filter<> f = ecs.filter_builder() + .expr("(ChildOf,0)") + .build(); + + int32_t count = 0; + f.each_term([&](flecs::term &term) { + test_assert(term.id().is_pair()); + count ++; + }); + + test_int(count, 1); +} diff --git a/test/cpp_api/src/main.cpp b/test/cpp_api/src/main.cpp index aa6b8cd936..2a7755e097 100644 --- a/test/cpp_api/src/main.cpp +++ b/test/cpp_api/src/main.cpp @@ -515,6 +515,7 @@ void Query_test_no_defer_each(void); void Query_test_no_defer_iter(void); void Query_inspect_terms(void); void Query_inspect_terms_w_each(void); +void Query_inspect_terms_w_expr(void); void Query_comp_to_str(void); void Query_pair_to_str(void); void Query_oper_not_to_str(void); @@ -777,6 +778,7 @@ void RuleBuilder_is_valid(void); void RuleBuilder_unresolved_by_name(void); void RuleBuilder_scope(void); void RuleBuilder_iter_w_stage(void); +void RuleBuilder_inspect_terms_w_expr(void); // Testsuite 'SystemBuilder' void SystemBuilder_builder_assign_same_type(void); @@ -857,6 +859,7 @@ void Filter_invalid_each_w_no_this(void); void Filter_named_filter(void); void Filter_named_scoped_filter(void); void Filter_set_this_var(void); +void Filter_inspect_terms_w_expr(void); // Testsuite 'ComponentLifecycle' void ComponentLifecycle_ctor_on_add(void); @@ -3257,6 +3260,10 @@ bake_test_case Query_testcases[] = { "inspect_terms_w_each", Query_inspect_terms_w_each }, + { + "inspect_terms_w_expr", + Query_inspect_terms_w_expr + }, { "comp_to_str", Query_comp_to_str @@ -4289,6 +4296,10 @@ bake_test_case RuleBuilder_testcases[] = { { "iter_w_stage", RuleBuilder_iter_w_stage + }, + { + "inspect_terms_w_expr", + RuleBuilder_inspect_terms_w_expr } }; @@ -4594,6 +4605,10 @@ bake_test_case Filter_testcases[] = { { "set_this_var", Filter_set_this_var + }, + { + "inspect_terms_w_expr", + Filter_inspect_terms_w_expr } }; @@ -6277,7 +6292,7 @@ static bake_test_suite suites[] = { "Query", NULL, NULL, - 80, + 81, Query_testcases }, { @@ -6298,7 +6313,7 @@ static bake_test_suite suites[] = { "RuleBuilder", NULL, NULL, - 29, + 30, RuleBuilder_testcases }, { @@ -6319,7 +6334,7 @@ static bake_test_suite suites[] = { "Filter", NULL, NULL, - 22, + 23, Filter_testcases }, {