/* Apache License, Version 2.0 */ #include "testing/testing.h" #include "FN_multi_function.hh" #include "FN_multi_function_builder.hh" namespace blender::fn::tests { namespace { class AddFunction : public MultiFunction { public: AddFunction() { static MFSignature signature = create_signature(); this->set_signature(&signature); } static MFSignature create_signature() { MFSignatureBuilder signature("Add"); signature.single_input("A"); signature.single_input("B"); signature.single_output("Result"); return signature.build(); } void call(IndexMask mask, MFParams params, MFContext UNUSED(context)) const override { const VArray &a = params.readonly_single_input(0, "A"); const VArray &b = params.readonly_single_input(1, "B"); MutableSpan result = params.uninitialized_single_output(2, "Result"); for (int64_t i : mask) { result[i] = a[i] + b[i]; } } }; TEST(multi_function, AddFunction) { AddFunction fn; Array input1 = {4, 5, 6}; Array input2 = {10, 20, 30}; Array output(3, -1); MFParamsBuilder params(fn, 3); params.add_readonly_single_input(input1.as_span()); params.add_readonly_single_input(input2.as_span()); params.add_uninitialized_single_output(output.as_mutable_span()); MFContextBuilder context; fn.call({0, 2}, params, context); EXPECT_EQ(output[0], 14); EXPECT_EQ(output[1], -1); EXPECT_EQ(output[2], 36); } class AddPrefixFunction : public MultiFunction { public: AddPrefixFunction() { static MFSignature signature = create_signature(); this->set_signature(&signature); } static MFSignature create_signature() { MFSignatureBuilder signature{"Add Prefix"}; signature.single_input("Prefix"); signature.single_mutable("Strings"); return signature.build(); } void call(IndexMask mask, MFParams params, MFContext UNUSED(context)) const override { const VArray &prefixes = params.readonly_single_input(0, "Prefix"); MutableSpan strings = params.single_mutable(1, "Strings"); for (int64_t i : mask) { strings[i] = prefixes[i] + strings[i]; } } }; TEST(multi_function, AddPrefixFunction) { AddPrefixFunction fn; Array strings = { "Hello", "World", "This is a test", "Another much longer string to trigger an allocation", }; std::string prefix = "AB"; MFParamsBuilder params(fn, strings.size()); params.add_readonly_single_input(&prefix); params.add_single_mutable(strings.as_mutable_span()); MFContextBuilder context; fn.call({0, 2, 3}, params, context); EXPECT_EQ(strings[0], "ABHello"); EXPECT_EQ(strings[1], "World"); EXPECT_EQ(strings[2], "ABThis is a test"); EXPECT_EQ(strings[3], "ABAnother much longer string to trigger an allocation"); } class CreateRangeFunction : public MultiFunction { public: CreateRangeFunction() { static MFSignature signature = create_signature(); this->set_signature(&signature); } static MFSignature create_signature() { MFSignatureBuilder signature{"Create Range"}; signature.single_input("Size"); signature.vector_output("Range"); return signature.build(); } void call(IndexMask mask, MFParams params, MFContext UNUSED(context)) const override { const VArray &sizes = params.readonly_single_input(0, "Size"); GVectorArray &ranges = params.vector_output(1, "Range"); for (int64_t i : mask) { uint size = sizes[i]; for (uint j : IndexRange(size)) { ranges.append(i, &j); } } } }; TEST(multi_function, CreateRangeFunction) { CreateRangeFunction fn; GVectorArray ranges(CPPType::get(), 5); GVectorArray_TypedMutableRef ranges_ref{ranges}; Array sizes = {3, 0, 6, 1, 4}; MFParamsBuilder params(fn, ranges.size()); params.add_readonly_single_input(sizes.as_span()); params.add_vector_output(ranges); MFContextBuilder context; fn.call({0, 1, 2, 3}, params, context); EXPECT_EQ(ranges[0].size(), 3); EXPECT_EQ(ranges[1].size(), 0); EXPECT_EQ(ranges[2].size(), 6); EXPECT_EQ(ranges[3].size(), 1); EXPECT_EQ(ranges[4].size(), 0); EXPECT_EQ(ranges_ref[0][0], 0); EXPECT_EQ(ranges_ref[0][1], 1); EXPECT_EQ(ranges_ref[0][2], 2); EXPECT_EQ(ranges_ref[2][0], 0); EXPECT_EQ(ranges_ref[2][1], 1); } class GenericAppendFunction : public MultiFunction { private: MFSignature signature_; public: GenericAppendFunction(const CPPType &type) { MFSignatureBuilder signature{"Append"}; signature.vector_mutable("Vector", type); signature.single_input("Value", type); signature_ = signature.build(); this->set_signature(&signature_); } void call(IndexMask mask, MFParams params, MFContext UNUSED(context)) const override { GVectorArray &vectors = params.vector_mutable(0, "Vector"); const GVArray &values = params.readonly_single_input(1, "Value"); for (int64_t i : mask) { BUFFER_FOR_CPP_TYPE_VALUE(values.type(), buffer); values.get(i, buffer); vectors.append(i, buffer); values.type().destruct(buffer); } } }; TEST(multi_function, GenericAppendFunction) { GenericAppendFunction fn(CPPType::get()); GVectorArray vectors(CPPType::get(), 4); GVectorArray_TypedMutableRef vectors_ref{vectors}; vectors_ref.append(0, 1); vectors_ref.append(0, 2); vectors_ref.append(2, 6); Array values = {5, 7, 3, 1}; MFParamsBuilder params(fn, vectors.size()); params.add_vector_mutable(vectors); params.add_readonly_single_input(values.as_span()); MFContextBuilder context; fn.call(IndexRange(vectors.size()), params, context); EXPECT_EQ(vectors[0].size(), 3); EXPECT_EQ(vectors[1].size(), 1); EXPECT_EQ(vectors[2].size(), 2); EXPECT_EQ(vectors[3].size(), 1); EXPECT_EQ(vectors_ref[0][0], 1); EXPECT_EQ(vectors_ref[0][1], 2); EXPECT_EQ(vectors_ref[0][2], 5); EXPECT_EQ(vectors_ref[1][0], 7); EXPECT_EQ(vectors_ref[2][0], 6); EXPECT_EQ(vectors_ref[2][1], 3); EXPECT_EQ(vectors_ref[3][0], 1); } TEST(multi_function, CustomMF_SI_SO) { CustomMF_SI_SO fn("strlen", [](const std::string &str) { return str.size(); }); Array strings = {"hello", "world", "test", "another test"}; Array sizes(strings.size(), 0); MFParamsBuilder params(fn, strings.size()); params.add_readonly_single_input(strings.as_span()); params.add_uninitialized_single_output(sizes.as_mutable_span()); MFContextBuilder context; fn.call(IndexRange(strings.size()), params, context); EXPECT_EQ(sizes[0], 5); EXPECT_EQ(sizes[1], 5); EXPECT_EQ(sizes[2], 4); EXPECT_EQ(sizes[3], 12); } TEST(multi_function, CustomMF_SI_SI_SO) { CustomMF_SI_SI_SO fn("mul", [](int a, int b) { return a * b; }); Array values_a = {4, 6, 8, 9}; int value_b = 10; Array outputs(values_a.size(), -1); MFParamsBuilder params(fn, values_a.size()); params.add_readonly_single_input(values_a.as_span()); params.add_readonly_single_input(&value_b); params.add_uninitialized_single_output(outputs.as_mutable_span()); MFContextBuilder context; fn.call({0, 1, 3}, params, context); EXPECT_EQ(outputs[0], 40); EXPECT_EQ(outputs[1], 60); EXPECT_EQ(outputs[2], -1); EXPECT_EQ(outputs[3], 90); } TEST(multi_function, CustomMF_SI_SI_SI_SO) { CustomMF_SI_SI_SI_SO fn{ "custom", [](int a, const std::string &b, bool c) { return (uint)((uint)a + b.size() + (uint)c); }}; Array values_a = {5, 7, 3, 8}; Array values_b = {"hello", "world", "another", "test"}; Array values_c = {true, false, false, true}; Array outputs(values_a.size(), 0); MFParamsBuilder params(fn, values_a.size()); params.add_readonly_single_input(values_a.as_span()); params.add_readonly_single_input(values_b.as_span()); params.add_readonly_single_input(values_c.as_span()); params.add_uninitialized_single_output(outputs.as_mutable_span()); MFContextBuilder context; fn.call({1, 2, 3}, params, context); EXPECT_EQ(outputs[0], 0); EXPECT_EQ(outputs[1], 12); EXPECT_EQ(outputs[2], 10); EXPECT_EQ(outputs[3], 13); } TEST(multi_function, CustomMF_SM) { CustomMF_SM fn("AddSuffix", [](std::string &value) { value += " test"; }); Array values = {"a", "b", "c", "d", "e"}; MFParamsBuilder params(fn, values.size()); params.add_single_mutable(values.as_mutable_span()); MFContextBuilder context; fn.call({1, 2, 3}, params, context); EXPECT_EQ(values[0], "a"); EXPECT_EQ(values[1], "b test"); EXPECT_EQ(values[2], "c test"); EXPECT_EQ(values[3], "d test"); EXPECT_EQ(values[4], "e"); } TEST(multi_function, CustomMF_Constant) { CustomMF_Constant fn{42}; Array outputs(4, 0); MFParamsBuilder params(fn, outputs.size()); params.add_uninitialized_single_output(outputs.as_mutable_span()); MFContextBuilder context; fn.call({0, 2, 3}, params, context); EXPECT_EQ(outputs[0], 42); EXPECT_EQ(outputs[1], 0); EXPECT_EQ(outputs[2], 42); EXPECT_EQ(outputs[3], 42); } TEST(multi_function, CustomMF_GenericConstant) { int value = 42; CustomMF_GenericConstant fn{CPPType::get(), (const void *)&value}; EXPECT_EQ(fn.param_name(0), "42"); Array outputs(4, 0); MFParamsBuilder params(fn, outputs.size()); params.add_uninitialized_single_output(outputs.as_mutable_span()); MFContextBuilder context; fn.call({0, 1, 2}, params, context); EXPECT_EQ(outputs[0], 42); EXPECT_EQ(outputs[1], 42); EXPECT_EQ(outputs[2], 42); EXPECT_EQ(outputs[3], 0); } TEST(multi_function, CustomMF_GenericConstantArray) { std::array values = {3, 4, 5, 6}; CustomMF_GenericConstantArray fn{GSpan(Span(values))}; EXPECT_EQ(fn.param_name(0), "[3, 4, 5, 6, ]"); GVectorArray vector_array{CPPType::get(), 4}; GVectorArray_TypedMutableRef vector_array_ref{vector_array}; MFParamsBuilder params(fn, vector_array.size()); params.add_vector_output(vector_array); MFContextBuilder context; fn.call({1, 2, 3}, params, context); EXPECT_EQ(vector_array[0].size(), 0); EXPECT_EQ(vector_array[1].size(), 4); EXPECT_EQ(vector_array[2].size(), 4); EXPECT_EQ(vector_array[3].size(), 4); for (int i = 1; i < 4; i++) { EXPECT_EQ(vector_array_ref[i][0], 3); EXPECT_EQ(vector_array_ref[i][1], 4); EXPECT_EQ(vector_array_ref[i][2], 5); EXPECT_EQ(vector_array_ref[i][3], 6); } } TEST(multi_function, CustomMF_Convert) { CustomMF_Convert fn; Array inputs = {5.4f, 7.1f, 9.0f}; Array outputs(inputs.size(), 0); MFParamsBuilder params(fn, inputs.size()); params.add_readonly_single_input(inputs.as_span()); params.add_uninitialized_single_output(outputs.as_mutable_span()); MFContextBuilder context; fn.call({0, 2}, params, context); EXPECT_EQ(outputs[0], 5); EXPECT_EQ(outputs[1], 0); EXPECT_EQ(outputs[2], 9); } } // namespace } // namespace blender::fn::tests