| 
									
										
										
										
											2022-02-11 09:07:11 +11:00
										 |  |  | /* SPDX-License-Identifier: GPL-2.0-or-later */ | 
					
						
							| 
									
										
										
										
											2020-06-22 15:50:08 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | #pragma once
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** \file
 | 
					
						
							|  |  |  |  * \ingroup fn | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This file contains several utilities to create multi-functions with less redundant code. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <functional>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "FN_multi_function.hh"
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 14:25:20 +02:00
										 |  |  | namespace blender::fn { | 
					
						
							| 
									
										
										
										
											2020-06-22 15:50:08 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  |  * These presets determine what code is generated for a #CustomMF. Different presets make different | 
					
						
							|  |  |  |  * trade-offs between run-time performance and compile-time/binary size. | 
					
						
							| 
									
										
										
										
											2020-06-22 15:50:08 +02:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  | namespace CustomMF_presets { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /** Method to execute a function in case devirtualization was not possible. */ | 
					
						
							|  |  |  | enum class FallbackMode { | 
					
						
							|  |  |  |   /** Access all elements in virtual arrays through virtual function calls. */ | 
					
						
							|  |  |  |   Simple, | 
					
						
							|  |  |  |   /** Process elements in chunks to reduce virtual function call overhead. */ | 
					
						
							|  |  |  |   Materialized, | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2020-06-22 15:50:08 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * The "naive" method for executing a #CustomMF. Every element is processed separately and input | 
					
						
							|  |  |  |  * values are retrieved from the virtual arrays one by one. This generates the least amount of | 
					
						
							|  |  |  |  * code, but is also the slowest method. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | struct Simple { | 
					
						
							|  |  |  |   static constexpr bool use_devirtualization = false; | 
					
						
							|  |  |  |   static constexpr FallbackMode fallback_mode = FallbackMode::Simple; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * This is an improvement over the #Simple method. It still generates a relatively small amount of | 
					
						
							|  |  |  |  * code, because the function is only instantiated once. It's generally faster than #Simple, | 
					
						
							|  |  |  |  * because inputs are retrieved from the virtual arrays in chunks, reducing virtual method call | 
					
						
							|  |  |  |  * overhead. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | struct Materialized { | 
					
						
							|  |  |  |   static constexpr bool use_devirtualization = false; | 
					
						
							|  |  |  |   static constexpr FallbackMode fallback_mode = FallbackMode::Materialized; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * The most efficient preset, but also potentially generates a lot of code (exponential in the | 
					
						
							|  |  |  |  * number of inputs of the function). It generates separate optimized loops for all combinations of | 
					
						
							|  |  |  |  * inputs. This should be used for small functions of which all inputs are likely to be single | 
					
						
							|  |  |  |  * values or spans, and the number of inputs is relatively small. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | struct AllSpanOrSingle { | 
					
						
							|  |  |  |   static constexpr bool use_devirtualization = true; | 
					
						
							|  |  |  |   static constexpr FallbackMode fallback_mode = FallbackMode::Materialized; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-07 12:55:48 +01:00
										 |  |  |   template<typename... ParamTags, typename... LoadedParams, size_t... I> | 
					
						
							|  |  |  |   auto create_devirtualizers(TypeSequence<ParamTags...> /*param_tags*/, | 
					
						
							|  |  |  |                              std::index_sequence<I...> /*indices*/, | 
					
						
							|  |  |  |                              const IndexMask &mask, | 
					
						
							|  |  |  |                              const std::tuple<LoadedParams...> &loaded_params) | 
					
						
							| 
									
										
										
										
											2020-06-22 15:50:08 +02:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2023-01-07 12:55:48 +01:00
										 |  |  |     return std::make_tuple(IndexMaskDevirtualizer<true, true>{mask}, [&]() { | 
					
						
							|  |  |  |       typedef ParamTags ParamTag; | 
					
						
							|  |  |  |       typedef typename ParamTag::base_type T; | 
					
						
							|  |  |  |       if constexpr (ParamTag::category == MFParamCategory::SingleInput) { | 
					
						
							|  |  |  |         const GVArrayImpl &varray_impl = *std::get<I>(loaded_params); | 
					
						
							|  |  |  |         return GVArrayDevirtualizer<T, true, true>{varray_impl}; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       else if constexpr (ParamTag::category == MFParamCategory::SingleOutput) { | 
					
						
							|  |  |  |         T *ptr = std::get<I>(loaded_params); | 
					
						
							|  |  |  |         return BasicDevirtualizer<T *>{ptr}; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     }()...); | 
					
						
							| 
									
										
										
										
											2020-06-22 15:50:08 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2020-06-22 15:50:08 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  | /**
 | 
					
						
							| 
									
										
										
										
											2022-04-28 14:03:49 +10:00
										 |  |  |  * A slightly weaker variant of #AllSpanOrSingle. It generates less code, because it assumes that | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  |  * some of the inputs are most likely single values. It should be used for small functions which | 
					
						
							|  |  |  |  * have too many inputs to make #AllSingleOrSpan a reasonable choice. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | template<size_t... Indices> struct SomeSpanOrSingle { | 
					
						
							|  |  |  |   static constexpr bool use_devirtualization = true; | 
					
						
							|  |  |  |   static constexpr FallbackMode fallback_mode = FallbackMode::Materialized; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-07 12:55:48 +01:00
										 |  |  |   template<typename... ParamTags, typename... LoadedParams, size_t... I> | 
					
						
							|  |  |  |   auto create_devirtualizers(TypeSequence<ParamTags...> /*param_tags*/, | 
					
						
							|  |  |  |                              std::index_sequence<I...> /*indices*/, | 
					
						
							|  |  |  |                              const IndexMask &mask, | 
					
						
							|  |  |  |                              const std::tuple<LoadedParams...> &loaded_params) | 
					
						
							| 
									
										
										
										
											2020-06-22 15:50:08 +02:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2023-01-07 12:55:48 +01:00
										 |  |  |     return std::make_tuple(IndexMaskDevirtualizer<true, true>{mask}, [&]() { | 
					
						
							|  |  |  |       typedef ParamTags ParamTag; | 
					
						
							|  |  |  |       typedef typename ParamTag::base_type T; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if constexpr (ParamTag::category == MFParamCategory::SingleInput) { | 
					
						
							|  |  |  |         constexpr bool UseSpan = ValueSequence<size_t, Indices...>::template contains<I>(); | 
					
						
							|  |  |  |         const GVArrayImpl &varray_impl = *std::get<I>(loaded_params); | 
					
						
							|  |  |  |         return GVArrayDevirtualizer<T, true, UseSpan>{varray_impl}; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       else if constexpr (ParamTag::category == MFParamCategory::SingleOutput) { | 
					
						
							|  |  |  |         T *ptr = std::get<I>(loaded_params); | 
					
						
							|  |  |  |         return BasicDevirtualizer<T *>{ptr}; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     }()...); | 
					
						
							| 
									
										
										
										
											2020-06-22 15:50:08 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2020-06-22 15:50:08 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  | }  // namespace CustomMF_presets
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | namespace detail { | 
					
						
							| 
									
										
										
										
											2022-04-07 18:48:14 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Executes #element_fn for all indices in the mask. The passed in #args contain the input as well | 
					
						
							|  |  |  |  * as output parameters. Usually types in #args are devirtualized (e.g. a `Span<int>` is passed in | 
					
						
							|  |  |  |  * instead of a `VArray<int>`). | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | template<typename MaskT, typename... Args, typename... ParamTags, size_t... I, typename ElementFn> | 
					
						
							| 
									
										
										
										
											2023-01-07 12:55:48 +01:00
										 |  |  | void execute_array(TypeSequence<ParamTags...> /*param_tags*/, | 
					
						
							|  |  |  |                    std::index_sequence<I...> /*indices*/, | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  |                    ElementFn element_fn, | 
					
						
							|  |  |  |                    MaskT mask, | 
					
						
							|  |  |  |                    /* Use restrict to tell the compiler that pointer inputs do not alias each
 | 
					
						
							|  |  |  |                     * other. This is important for some compiler optimizations. */ | 
					
						
							|  |  |  |                    Args &&__restrict... args) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   for (const int64_t i : mask) { | 
					
						
							|  |  |  |     element_fn([&]() -> decltype(auto) { | 
					
						
							|  |  |  |       using ParamTag = typename TypeSequence<ParamTags...>::template at_index<I>; | 
					
						
							|  |  |  |       if constexpr (ParamTag::category == MFParamCategory::SingleInput) { | 
					
						
							|  |  |  |         /* For inputs, pass the value (or a reference to it) to the function. */ | 
					
						
							|  |  |  |         return args[i]; | 
					
						
							| 
									
										
										
										
											2022-04-07 18:48:14 +02:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  |       else if constexpr (ParamTag::category == MFParamCategory::SingleOutput) { | 
					
						
							|  |  |  |         /* For outputs, pass a pointer to the function. This is done instead of passing a
 | 
					
						
							|  |  |  |          * reference, because the pointer points to uninitialized memory. */ | 
					
						
							| 
									
										
										
										
											2023-01-07 12:55:48 +01:00
										 |  |  |         return args + i; | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  |       } | 
					
						
							|  |  |  |     }()...); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2022-04-07 18:48:14 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  | }  // namespace detail
 | 
					
						
							| 
									
										
										
										
											2022-04-07 18:48:14 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  | namespace materialize_detail { | 
					
						
							| 
									
										
										
										
											2022-04-07 18:48:14 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  | enum class ArgMode { | 
					
						
							|  |  |  |   Unknown, | 
					
						
							|  |  |  |   Single, | 
					
						
							|  |  |  |   Span, | 
					
						
							|  |  |  |   Materialized, | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2022-04-07 18:48:14 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  | template<typename ParamTag> struct ArgInfo { | 
					
						
							|  |  |  |   ArgMode mode = ArgMode::Unknown; | 
					
						
							|  |  |  |   Span<typename ParamTag::base_type> internal_span; | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2022-04-07 18:48:14 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Similar to #execute_array but accepts two mask inputs, one for inputs and one for outputs. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | template<typename... ParamTags, typename ElementFn, typename... Chunks> | 
					
						
							| 
									
										
										
										
											2023-01-07 12:55:48 +01:00
										 |  |  | void execute_materialized_impl(TypeSequence<ParamTags...> /*param_tags*/, | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  |                                const ElementFn element_fn, | 
					
						
							|  |  |  |                                const IndexRange in_mask, | 
					
						
							|  |  |  |                                const IndexMask out_mask, | 
					
						
							|  |  |  |                                Chunks &&__restrict... chunks) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |   BLI_assert(in_mask.size() == out_mask.size()); | 
					
						
							|  |  |  |   for (const int64_t i : IndexRange(in_mask.size())) { | 
					
						
							|  |  |  |     const int64_t in_i = in_mask[i]; | 
					
						
							|  |  |  |     const int64_t out_i = out_mask[i]; | 
					
						
							|  |  |  |     element_fn([&]() -> decltype(auto) { | 
					
						
							|  |  |  |       using ParamTag = ParamTags; | 
					
						
							|  |  |  |       if constexpr (ParamTag::category == MFParamCategory::SingleInput) { | 
					
						
							|  |  |  |         return chunks[in_i]; | 
					
						
							| 
									
										
										
										
											2022-04-07 18:48:14 +02:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  |       else if constexpr (ParamTag::category == MFParamCategory::SingleOutput) { | 
					
						
							|  |  |  |         /* For outputs, a pointer is passed, because the memory is uninitialized. */ | 
					
						
							| 
									
										
										
										
											2023-01-07 12:55:48 +01:00
										 |  |  |         return chunks + out_i; | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  |       } | 
					
						
							|  |  |  |     }()...); | 
					
						
							| 
									
										
										
										
											2022-04-04 11:57:39 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  | } | 
					
						
							| 
									
										
										
										
											2022-04-04 11:57:39 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Executes #element_fn for all indices in #mask. However, instead of processing every element | 
					
						
							|  |  |  |  * separately, processing happens in chunks. This allows retrieving from input virtual arrays in | 
					
						
							|  |  |  |  * chunks, which reduces virtual function call overhead. | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2023-01-07 12:55:48 +01:00
										 |  |  | template<typename... ParamTags, size_t... I, typename ElementFn, typename... LoadedParams> | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  | void execute_materialized(TypeSequence<ParamTags...> /* param_tags */, | 
					
						
							|  |  |  |                           std::index_sequence<I...> /* indices */, | 
					
						
							|  |  |  |                           const ElementFn element_fn, | 
					
						
							|  |  |  |                           const IndexMask mask, | 
					
						
							| 
									
										
										
										
											2023-01-07 12:55:48 +01:00
										 |  |  |                           const std::tuple<LoadedParams...> &loaded_params) | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  | { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* In theory, all elements could be processed in one chunk. However, that has the disadvantage
 | 
					
						
							|  |  |  |    * that large temporary arrays are needed. Using small chunks allows using small arrays, which | 
					
						
							|  |  |  |    * are reused multiple times, which improves cache efficiency. The chunk size also shouldn't be | 
					
						
							|  |  |  |    * too small, because then overhead of the outer loop over chunks becomes significant again. */ | 
					
						
							|  |  |  |   static constexpr int64_t MaxChunkSize = 32; | 
					
						
							|  |  |  |   const int64_t mask_size = mask.size(); | 
					
						
							|  |  |  |   const int64_t buffer_size = std::min(mask_size, MaxChunkSize); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Local buffers that are used to temporarily store values retrieved from virtual arrays. */ | 
					
						
							|  |  |  |   std::tuple<TypedBuffer<typename ParamTags::base_type, MaxChunkSize>...> buffers_owner; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* A span for each parameter which is either empty or points to memory in #buffers_owner. */ | 
					
						
							|  |  |  |   std::tuple<MutableSpan<typename ParamTags::base_type>...> buffers; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Information about every parameter. */ | 
					
						
							|  |  |  |   std::tuple<ArgInfo<ParamTags>...> args_info; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   ( | 
					
						
							|  |  |  |       /* Setup information for all parameters. */ | 
					
						
							|  |  |  |       [&] { | 
					
						
							| 
									
										
										
										
											2022-09-20 10:42:25 +02:00
										 |  |  |         /* Use `typedef` instead of `using` to work around a compiler bug. */ | 
					
						
							| 
									
										
										
										
											2022-05-12 13:38:22 +02:00
										 |  |  |         typedef ParamTags ParamTag; | 
					
						
							|  |  |  |         typedef typename ParamTag::base_type T; | 
					
						
							| 
									
										
										
										
											2022-05-12 13:03:12 +02:00
										 |  |  |         [[maybe_unused]] ArgInfo<ParamTags> &arg_info = std::get<I>(args_info); | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  |         if constexpr (ParamTag::category == MFParamCategory::SingleInput) { | 
					
						
							| 
									
										
										
										
											2023-01-07 12:55:48 +01:00
										 |  |  |           const GVArrayImpl &varray_impl = *std::get<I>(loaded_params); | 
					
						
							|  |  |  |           const CommonVArrayInfo common_info = varray_impl.common_info(); | 
					
						
							|  |  |  |           if (common_info.type == CommonVArrayInfo::Type::Single) { | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  |             /* If an input #VArray is a single value, we have to fill the buffer with that value
 | 
					
						
							|  |  |  |              * only once. The same unchanged buffer can then be reused in every chunk. */ | 
					
						
							|  |  |  |             MutableSpan<T> in_chunk{std::get<I>(buffers_owner).ptr(), buffer_size}; | 
					
						
							| 
									
										
										
										
											2023-01-07 12:55:48 +01:00
										 |  |  |             const T &in_single = *static_cast<const T *>(common_info.data); | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  |             uninitialized_fill_n(in_chunk.data(), in_chunk.size(), in_single); | 
					
						
							|  |  |  |             std::get<I>(buffers) = in_chunk; | 
					
						
							|  |  |  |             arg_info.mode = ArgMode::Single; | 
					
						
							|  |  |  |           } | 
					
						
							| 
									
										
										
										
											2023-01-07 12:55:48 +01:00
										 |  |  |           else if (common_info.type == CommonVArrayInfo::Type::Span) { | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  |             /* Remember the span so that it doesn't have to be retrieved in every iteration. */ | 
					
						
							| 
									
										
										
										
											2023-01-07 12:55:48 +01:00
										 |  |  |             const T *ptr = static_cast<const T *>(common_info.data); | 
					
						
							|  |  |  |             arg_info.internal_span = Span<T>(ptr, varray_impl.size()); | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       }(), | 
					
						
							|  |  |  |       ...); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   /* Outer loop over all chunks. */ | 
					
						
							|  |  |  |   for (int64_t chunk_start = 0; chunk_start < mask_size; chunk_start += MaxChunkSize) { | 
					
						
							| 
									
										
										
										
											2022-11-23 11:35:59 -06:00
										 |  |  |     const IndexMask sliced_mask = mask.slice_safe(chunk_start, MaxChunkSize); | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  |     const int64_t chunk_size = sliced_mask.size(); | 
					
						
							|  |  |  |     const bool sliced_mask_is_range = sliced_mask.is_range(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     execute_materialized_impl( | 
					
						
							|  |  |  |         TypeSequence<ParamTags...>(), | 
					
						
							|  |  |  |         element_fn, | 
					
						
							|  |  |  |         /* Inputs are "compressed" into contiguous arrays without gaps. */ | 
					
						
							|  |  |  |         IndexRange(chunk_size), | 
					
						
							|  |  |  |         /* Outputs are written directly into the correct place in the output arrays. */ | 
					
						
							|  |  |  |         sliced_mask, | 
					
						
							|  |  |  |         /* Prepare every parameter for this chunk. */ | 
					
						
							|  |  |  |         [&] { | 
					
						
							|  |  |  |           using ParamTag = ParamTags; | 
					
						
							|  |  |  |           using T = typename ParamTag::base_type; | 
					
						
							| 
									
										
										
										
											2022-05-12 13:03:12 +02:00
										 |  |  |           [[maybe_unused]] ArgInfo<ParamTags> &arg_info = std::get<I>(args_info); | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  |           if constexpr (ParamTag::category == MFParamCategory::SingleInput) { | 
					
						
							|  |  |  |             if (arg_info.mode == ArgMode::Single) { | 
					
						
							|  |  |  |               /* The single value has been filled into a buffer already reused for every chunk. */ | 
					
						
							|  |  |  |               return Span<T>(std::get<I>(buffers)); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             else { | 
					
						
							|  |  |  |               if (sliced_mask_is_range) { | 
					
						
							|  |  |  |                 if (!arg_info.internal_span.is_empty()) { | 
					
						
							|  |  |  |                   /* In this case we can just use an existing span instead of "compressing" it into
 | 
					
						
							|  |  |  |                    * a new temporary buffer. */ | 
					
						
							|  |  |  |                   const IndexRange sliced_mask_range = sliced_mask.as_range(); | 
					
						
							|  |  |  |                   arg_info.mode = ArgMode::Span; | 
					
						
							|  |  |  |                   return arg_info.internal_span.slice(sliced_mask_range); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |               } | 
					
						
							| 
									
										
										
										
											2023-01-07 12:55:48 +01:00
										 |  |  |               const GVArrayImpl &varray_impl = *std::get<I>(loaded_params); | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  |               /* As a fallback, do a virtual function call to retrieve all elements in the current
 | 
					
						
							|  |  |  |                * chunk. The elements are stored in a temporary buffer reused for every chunk. */ | 
					
						
							|  |  |  |               MutableSpan<T> in_chunk{std::get<I>(buffers_owner).ptr(), chunk_size}; | 
					
						
							| 
									
										
										
										
											2023-01-07 12:55:48 +01:00
										 |  |  |               varray_impl.materialize_compressed_to_uninitialized(sliced_mask, in_chunk.data()); | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  |               /* Remember that this parameter has been materialized, so that the values are
 | 
					
						
							|  |  |  |                * destructed properly when the chunk is done. */ | 
					
						
							|  |  |  |               arg_info.mode = ArgMode::Materialized; | 
					
						
							|  |  |  |               return Span<T>(in_chunk); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |           else if constexpr (ParamTag::category == MFParamCategory::SingleOutput) { | 
					
						
							|  |  |  |             /* For outputs, just pass a pointer. This is important so that `__restrict` works. */ | 
					
						
							| 
									
										
										
										
											2023-01-07 12:55:48 +01:00
										 |  |  |             return std::get<I>(loaded_params); | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  |           } | 
					
						
							|  |  |  |         }()...); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ( | 
					
						
							|  |  |  |         /* Destruct values that have been materialized before. */ | 
					
						
							|  |  |  |         [&] { | 
					
						
							| 
									
										
										
										
											2022-09-20 10:42:25 +02:00
										 |  |  |           /* Use `typedef` instead of `using` to work around a compiler bug. */ | 
					
						
							| 
									
										
										
										
											2022-05-12 13:38:22 +02:00
										 |  |  |           typedef ParamTags ParamTag; | 
					
						
							|  |  |  |           typedef typename ParamTag::base_type T; | 
					
						
							| 
									
										
										
										
											2022-05-12 13:03:12 +02:00
										 |  |  |           [[maybe_unused]] ArgInfo<ParamTags> &arg_info = std::get<I>(args_info); | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  |           if constexpr (ParamTag::category == MFParamCategory::SingleInput) { | 
					
						
							|  |  |  |             if (arg_info.mode == ArgMode::Materialized) { | 
					
						
							|  |  |  |               T *in_chunk = std::get<I>(buffers_owner).ptr(); | 
					
						
							|  |  |  |               destruct_n(in_chunk, chunk_size); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         }(), | 
					
						
							|  |  |  |         ...); | 
					
						
							| 
									
										
										
										
											2022-04-07 18:48:14 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  |   ( | 
					
						
							|  |  |  |       /* Destruct buffers for single value inputs. */ | 
					
						
							|  |  |  |       [&] { | 
					
						
							| 
									
										
										
										
											2022-09-20 10:42:25 +02:00
										 |  |  |         /* Use `typedef` instead of `using` to work around a compiler bug. */ | 
					
						
							| 
									
										
										
										
											2022-05-12 13:38:22 +02:00
										 |  |  |         typedef ParamTags ParamTag; | 
					
						
							|  |  |  |         typedef typename ParamTag::base_type T; | 
					
						
							| 
									
										
										
										
											2022-05-12 13:03:12 +02:00
										 |  |  |         [[maybe_unused]] ArgInfo<ParamTags> &arg_info = std::get<I>(args_info); | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  |         if constexpr (ParamTag::category == MFParamCategory::SingleInput) { | 
					
						
							|  |  |  |           if (arg_info.mode == ArgMode::Single) { | 
					
						
							|  |  |  |             MutableSpan<T> in_chunk = std::get<I>(buffers); | 
					
						
							|  |  |  |             destruct_n(in_chunk.data(), in_chunk.size()); | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       }(), | 
					
						
							|  |  |  |       ...); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | }  // namespace materialize_detail
 | 
					
						
							| 
									
										
										
										
											2020-06-22 15:50:08 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  | template<typename... ParamTags> class CustomMF : public MultiFunction { | 
					
						
							| 
									
										
										
										
											2020-06-22 15:50:08 +02:00
										 |  |  |  private: | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  |   std::function<void(IndexMask mask, MFParams params)> fn_; | 
					
						
							| 
									
										
										
										
											2021-03-22 11:57:24 +01:00
										 |  |  |   MFSignature signature_; | 
					
						
							| 
									
										
										
										
											2020-06-22 15:50:08 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  |   using TagsSequence = TypeSequence<ParamTags...>; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-22 15:50:08 +02:00
										 |  |  |  public: | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  |   template<typename ElementFn, typename ExecPreset = CustomMF_presets::Materialized> | 
					
						
							|  |  |  |   CustomMF(const char *name, | 
					
						
							|  |  |  |            ElementFn element_fn, | 
					
						
							|  |  |  |            ExecPreset exec_preset = CustomMF_presets::Materialized()) | 
					
						
							| 
									
										
										
										
											2020-06-22 15:50:08 +02:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2021-03-22 11:57:24 +01:00
										 |  |  |     MFSignatureBuilder signature{name}; | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  |     add_signature_parameters(signature, std::make_index_sequence<TagsSequence::size()>()); | 
					
						
							| 
									
										
										
										
											2021-03-22 11:57:24 +01:00
										 |  |  |     signature_ = signature.build(); | 
					
						
							|  |  |  |     this->set_signature(&signature_); | 
					
						
							| 
									
										
										
										
											2020-06-22 15:50:08 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  |     fn_ = [element_fn, exec_preset](IndexMask mask, MFParams params) { | 
					
						
							|  |  |  |       execute( | 
					
						
							|  |  |  |           element_fn, exec_preset, mask, params, std::make_index_sequence<TagsSequence::size()>()); | 
					
						
							|  |  |  |     }; | 
					
						
							| 
									
										
										
										
											2020-06-22 15:50:08 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  |   template<typename ElementFn, typename ExecPreset, size_t... I> | 
					
						
							|  |  |  |   static void execute(ElementFn element_fn, | 
					
						
							|  |  |  |                       ExecPreset exec_preset, | 
					
						
							|  |  |  |                       IndexMask mask, | 
					
						
							|  |  |  |                       MFParams params, | 
					
						
							| 
									
										
										
										
											2023-01-07 12:55:48 +01:00
										 |  |  |                       std::index_sequence<I...> /*indices*/) | 
					
						
							| 
									
										
										
										
											2020-06-22 15:50:08 +02:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2023-01-07 12:55:48 +01:00
										 |  |  |     /* Contains `const GVArrayImpl *` for inputs and `T *` for outputs. */ | 
					
						
							|  |  |  |     const auto loaded_params = std::make_tuple([&]() { | 
					
						
							|  |  |  |       /* Use `typedef` instead of `using` to work around a compiler bug. */ | 
					
						
							|  |  |  |       typedef typename TagsSequence::template at_index<I> ParamTag; | 
					
						
							|  |  |  |       typedef typename ParamTag::base_type T; | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-01-07 12:55:48 +01:00
										 |  |  |       if constexpr (ParamTag::category == MFParamCategory::SingleInput) { | 
					
						
							|  |  |  |         return params.readonly_single_input(I).get_implementation(); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       else if constexpr (ParamTag::category == MFParamCategory::SingleOutput) { | 
					
						
							|  |  |  |         return static_cast<T *>(params.uninitialized_single_output(I).data()); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     }()...); | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     /* First try devirtualized execution, since this is the most efficient. */ | 
					
						
							|  |  |  |     bool executed_devirtualized = false; | 
					
						
							|  |  |  |     if constexpr (ExecPreset::use_devirtualization) { | 
					
						
							| 
									
										
										
										
											2023-01-07 12:55:48 +01:00
										 |  |  |       const auto devirtualizers = exec_preset.create_devirtualizers( | 
					
						
							|  |  |  |           TagsSequence(), std::index_sequence<I...>(), mask, loaded_params); | 
					
						
							|  |  |  |       executed_devirtualized = call_with_devirtualized_parameters( | 
					
						
							|  |  |  |           devirtualizers, [&](auto &&...args) { | 
					
						
							|  |  |  |             detail::execute_array(TagsSequence(), | 
					
						
							|  |  |  |                                   std::index_sequence<I...>(), | 
					
						
							|  |  |  |                                   element_fn, | 
					
						
							|  |  |  |                                   std::forward<decltype(args)>(args)...); | 
					
						
							|  |  |  |           }); | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     /* If devirtualized execution was disabled or not possible, use a fallback method which is
 | 
					
						
							|  |  |  |      * slower but always works. */ | 
					
						
							|  |  |  |     if (!executed_devirtualized) { | 
					
						
							|  |  |  |       if constexpr (ExecPreset::fallback_mode == CustomMF_presets::FallbackMode::Materialized) { | 
					
						
							|  |  |  |         materialize_detail::execute_materialized( | 
					
						
							| 
									
										
										
										
											2023-01-07 12:55:48 +01:00
										 |  |  |             TagsSequence(), std::index_sequence<I...>(), element_fn, mask, loaded_params); | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  |       } | 
					
						
							|  |  |  |       else { | 
					
						
							| 
									
										
										
										
											2023-01-07 12:55:48 +01:00
										 |  |  |         detail::execute_array( | 
					
						
							|  |  |  |             TagsSequence(), std::index_sequence<I...>(), element_fn, mask, [&]() { | 
					
						
							|  |  |  |               /* Use `typedef` instead of `using` to work around a compiler bug. */ | 
					
						
							|  |  |  |               typedef typename TagsSequence::template at_index<I> ParamTag; | 
					
						
							|  |  |  |               typedef typename ParamTag::base_type T; | 
					
						
							|  |  |  |               if constexpr (ParamTag::category == MFParamCategory::SingleInput) { | 
					
						
							|  |  |  |                 const GVArrayImpl &varray_impl = *std::get<I>(loaded_params); | 
					
						
							|  |  |  |                 return GVArray(&varray_impl).typed<T>(); | 
					
						
							|  |  |  |               } | 
					
						
							|  |  |  |               else if constexpr (ParamTag::category == MFParamCategory::SingleOutput) { | 
					
						
							|  |  |  |                 T *ptr = std::get<I>(loaded_params); | 
					
						
							|  |  |  |                 return ptr; | 
					
						
							|  |  |  |               } | 
					
						
							|  |  |  |             }()...); | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2020-06-22 15:50:08 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  |   template<size_t... I> | 
					
						
							|  |  |  |   static void add_signature_parameters(MFSignatureBuilder &signature, | 
					
						
							|  |  |  |                                        std::index_sequence<I...> /* indices */) | 
					
						
							| 
									
										
										
										
											2022-04-04 11:57:39 +02:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  |     ( | 
					
						
							|  |  |  |         /* Loop over all parameter types and add an entry for each in the signature. */ | 
					
						
							|  |  |  |         [&] { | 
					
						
							| 
									
										
										
										
											2022-09-20 10:42:25 +02:00
										 |  |  |           /* Use `typedef` instead of `using` to work around a compiler bug. */ | 
					
						
							| 
									
										
										
										
											2022-05-12 13:38:22 +02:00
										 |  |  |           typedef typename TagsSequence::template at_index<I> ParamTag; | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  |           signature.add(ParamTag(), ""); | 
					
						
							|  |  |  |         }(), | 
					
						
							|  |  |  |         ...); | 
					
						
							| 
									
										
										
										
											2022-04-04 11:57:39 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-03 17:37:25 -05:00
										 |  |  |   void call(IndexMask mask, MFParams params, MFContext /*context*/) const override | 
					
						
							| 
									
										
										
										
											2020-06-22 15:50:08 +02:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  |     fn_(mask, params); | 
					
						
							| 
									
										
										
										
											2020-06-22 15:50:08 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-30 17:59:33 +02:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Generates a multi-function with the following parameters: | 
					
						
							|  |  |  |  * 1. single input (SI) of type In1 | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  |  * 2. single output (SO) of type Out1 | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * This example creates a function that adds 10 to the incoming values: | 
					
						
							|  |  |  |  * `CustomMF_SI_SO<int, int> fn("add 10", [](int value) { return value + 10; });` | 
					
						
							| 
									
										
										
										
											2020-06-30 17:59:33 +02:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  | template<typename In1, typename Out1> | 
					
						
							|  |  |  | class CustomMF_SI_SO : public CustomMF<MFParamTag<MFParamCategory::SingleInput, In1>, | 
					
						
							|  |  |  |                                        MFParamTag<MFParamCategory::SingleOutput, Out1>> { | 
					
						
							| 
									
										
										
										
											2020-06-30 17:59:33 +02:00
										 |  |  |  public: | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  |   template<typename ElementFn, typename ExecPreset = CustomMF_presets::Materialized> | 
					
						
							|  |  |  |   CustomMF_SI_SO(const char *name, | 
					
						
							|  |  |  |                  ElementFn element_fn, | 
					
						
							|  |  |  |                  ExecPreset exec_preset = CustomMF_presets::Materialized()) | 
					
						
							|  |  |  |       : CustomMF<MFParamTag<MFParamCategory::SingleInput, In1>, | 
					
						
							|  |  |  |                  MFParamTag<MFParamCategory::SingleOutput, Out1>>( | 
					
						
							|  |  |  |             name, | 
					
						
							|  |  |  |             [element_fn](const In1 &in1, Out1 *out1) { new (out1) Out1(element_fn(in1)); }, | 
					
						
							|  |  |  |             exec_preset) | 
					
						
							| 
									
										
										
										
											2020-06-30 17:59:33 +02:00
										 |  |  |   { | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2020-06-30 17:59:33 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Generates a multi-function with the following parameters: | 
					
						
							|  |  |  |  * 1. single input (SI) of type In1 | 
					
						
							|  |  |  |  * 2. single input (SI) of type In2 | 
					
						
							|  |  |  |  * 3. single output (SO) of type Out1 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | template<typename In1, typename In2, typename Out1> | 
					
						
							|  |  |  | class CustomMF_SI_SI_SO : public CustomMF<MFParamTag<MFParamCategory::SingleInput, In1>, | 
					
						
							|  |  |  |                                           MFParamTag<MFParamCategory::SingleInput, In2>, | 
					
						
							|  |  |  |                                           MFParamTag<MFParamCategory::SingleOutput, Out1>> { | 
					
						
							|  |  |  |  public: | 
					
						
							|  |  |  |   template<typename ElementFn, typename ExecPreset = CustomMF_presets::Materialized> | 
					
						
							|  |  |  |   CustomMF_SI_SI_SO(const char *name, | 
					
						
							|  |  |  |                     ElementFn element_fn, | 
					
						
							|  |  |  |                     ExecPreset exec_preset = CustomMF_presets::Materialized()) | 
					
						
							|  |  |  |       : CustomMF<MFParamTag<MFParamCategory::SingleInput, In1>, | 
					
						
							|  |  |  |                  MFParamTag<MFParamCategory::SingleInput, In2>, | 
					
						
							|  |  |  |                  MFParamTag<MFParamCategory::SingleOutput, Out1>>( | 
					
						
							|  |  |  |             name, | 
					
						
							|  |  |  |             [element_fn](const In1 &in1, const In2 &in2, Out1 *out1) { | 
					
						
							|  |  |  |               new (out1) Out1(element_fn(in1, in2)); | 
					
						
							|  |  |  |             }, | 
					
						
							|  |  |  |             exec_preset) | 
					
						
							| 
									
										
										
										
											2022-04-07 11:51:31 +02:00
										 |  |  |   { | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  | }; | 
					
						
							| 
									
										
										
										
											2022-04-07 11:51:31 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Generates a multi-function with the following parameters: | 
					
						
							|  |  |  |  * 1. single input (SI) of type In1 | 
					
						
							|  |  |  |  * 2. single input (SI) of type In2 | 
					
						
							|  |  |  |  * 3. single input (SI) of type In3 | 
					
						
							|  |  |  |  * 4. single output (SO) of type Out1 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | template<typename In1, typename In2, typename In3, typename Out1> | 
					
						
							|  |  |  | class CustomMF_SI_SI_SI_SO : public CustomMF<MFParamTag<MFParamCategory::SingleInput, In1>, | 
					
						
							|  |  |  |                                              MFParamTag<MFParamCategory::SingleInput, In2>, | 
					
						
							|  |  |  |                                              MFParamTag<MFParamCategory::SingleInput, In3>, | 
					
						
							|  |  |  |                                              MFParamTag<MFParamCategory::SingleOutput, Out1>> { | 
					
						
							|  |  |  |  public: | 
					
						
							|  |  |  |   template<typename ElementFn, typename ExecPreset = CustomMF_presets::Materialized> | 
					
						
							|  |  |  |   CustomMF_SI_SI_SI_SO(const char *name, | 
					
						
							|  |  |  |                        ElementFn element_fn, | 
					
						
							|  |  |  |                        ExecPreset exec_preset = CustomMF_presets::Materialized()) | 
					
						
							|  |  |  |       : CustomMF<MFParamTag<MFParamCategory::SingleInput, In1>, | 
					
						
							|  |  |  |                  MFParamTag<MFParamCategory::SingleInput, In2>, | 
					
						
							|  |  |  |                  MFParamTag<MFParamCategory::SingleInput, In3>, | 
					
						
							|  |  |  |                  MFParamTag<MFParamCategory::SingleOutput, Out1>>( | 
					
						
							|  |  |  |             name, | 
					
						
							|  |  |  |             [element_fn](const In1 &in1, const In2 &in2, const In3 &in3, Out1 *out1) { | 
					
						
							|  |  |  |               new (out1) Out1(element_fn(in1, in2, in3)); | 
					
						
							|  |  |  |             }, | 
					
						
							|  |  |  |             exec_preset) | 
					
						
							| 
									
										
										
										
											2020-06-30 17:59:33 +02:00
										 |  |  |   { | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-08 11:37:37 +01:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Generates a multi-function with the following parameters: | 
					
						
							|  |  |  |  * 1. single input (SI) of type In1 | 
					
						
							|  |  |  |  * 2. single input (SI) of type In2 | 
					
						
							|  |  |  |  * 3. single input (SI) of type In3 | 
					
						
							|  |  |  |  * 4. single input (SI) of type In4 | 
					
						
							|  |  |  |  * 5. single output (SO) of type Out1 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | template<typename In1, typename In2, typename In3, typename In4, typename Out1> | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  | class CustomMF_SI_SI_SI_SI_SO : public CustomMF<MFParamTag<MFParamCategory::SingleInput, In1>, | 
					
						
							|  |  |  |                                                 MFParamTag<MFParamCategory::SingleInput, In2>, | 
					
						
							|  |  |  |                                                 MFParamTag<MFParamCategory::SingleInput, In3>, | 
					
						
							|  |  |  |                                                 MFParamTag<MFParamCategory::SingleInput, In4>, | 
					
						
							|  |  |  |                                                 MFParamTag<MFParamCategory::SingleOutput, Out1>> { | 
					
						
							| 
									
										
										
										
											2021-03-08 11:37:37 +01:00
										 |  |  |  public: | 
					
						
							| 
									
										
										
										
											2022-04-26 17:12:34 +02:00
										 |  |  |   template<typename ElementFn, typename ExecPreset = CustomMF_presets::Materialized> | 
					
						
							|  |  |  |   CustomMF_SI_SI_SI_SI_SO(const char *name, | 
					
						
							|  |  |  |                           ElementFn element_fn, | 
					
						
							|  |  |  |                           ExecPreset exec_preset = CustomMF_presets::Materialized()) | 
					
						
							|  |  |  |       : CustomMF<MFParamTag<MFParamCategory::SingleInput, In1>, | 
					
						
							|  |  |  |                  MFParamTag<MFParamCategory::SingleInput, In2>, | 
					
						
							|  |  |  |                  MFParamTag<MFParamCategory::SingleInput, In3>, | 
					
						
							|  |  |  |                  MFParamTag<MFParamCategory::SingleInput, In4>, | 
					
						
							|  |  |  |                  MFParamTag<MFParamCategory::SingleOutput, Out1>>( | 
					
						
							|  |  |  |             name, | 
					
						
							|  |  |  |             [element_fn]( | 
					
						
							|  |  |  |                 const In1 &in1, const In2 &in2, const In3 &in3, const In4 &in4, Out1 *out1) { | 
					
						
							|  |  |  |               new (out1) Out1(element_fn(in1, in2, in3, in4)); | 
					
						
							|  |  |  |             }, | 
					
						
							|  |  |  |             exec_preset) | 
					
						
							| 
									
										
										
										
											2021-03-08 11:37:37 +01:00
										 |  |  |   { | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-06-22 15:50:08 +02:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Generates a multi-function with the following parameters: | 
					
						
							|  |  |  |  * 1. single mutable (SM) of type Mut1 | 
					
						
							|  |  |  |  */ | 
					
						
							| 
									
										
										
										
											2020-06-30 17:59:33 +02:00
										 |  |  | template<typename Mut1> class CustomMF_SM : public MultiFunction { | 
					
						
							| 
									
										
										
										
											2020-06-22 15:50:08 +02:00
										 |  |  |  private: | 
					
						
							|  |  |  |   using FunctionT = std::function<void(IndexMask, MutableSpan<Mut1>)>; | 
					
						
							| 
									
										
										
										
											2020-07-03 14:20:42 +02:00
										 |  |  |   FunctionT function_; | 
					
						
							| 
									
										
										
										
											2021-03-22 11:57:24 +01:00
										 |  |  |   MFSignature signature_; | 
					
						
							| 
									
										
										
										
											2020-06-22 15:50:08 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |  public: | 
					
						
							| 
									
										
										
										
											2021-11-21 12:37:04 +01:00
										 |  |  |   CustomMF_SM(const char *name, FunctionT function) : function_(std::move(function)) | 
					
						
							| 
									
										
										
										
											2020-06-22 15:50:08 +02:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2021-03-22 11:57:24 +01:00
										 |  |  |     MFSignatureBuilder signature{name}; | 
					
						
							| 
									
										
										
										
											2020-06-22 15:50:08 +02:00
										 |  |  |     signature.single_mutable<Mut1>("Mut1"); | 
					
						
							| 
									
										
										
										
											2021-03-22 11:57:24 +01:00
										 |  |  |     signature_ = signature.build(); | 
					
						
							|  |  |  |     this->set_signature(&signature_); | 
					
						
							| 
									
										
										
										
											2020-06-22 15:50:08 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   template<typename ElementFuncT> | 
					
						
							| 
									
										
										
										
											2021-11-21 12:37:04 +01:00
										 |  |  |   CustomMF_SM(const char *name, ElementFuncT element_fn) | 
					
						
							| 
									
										
										
										
											2020-06-30 17:59:33 +02:00
										 |  |  |       : CustomMF_SM(name, CustomMF_SM::create_function(element_fn)) | 
					
						
							| 
									
										
										
										
											2020-06-22 15:50:08 +02:00
										 |  |  |   { | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   template<typename ElementFuncT> static FunctionT create_function(ElementFuncT element_fn) | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     return [=](IndexMask mask, MutableSpan<Mut1> mut1) { | 
					
						
							| 
									
										
										
										
											2022-03-29 10:11:49 +02:00
										 |  |  |       mask.to_best_mask_type([&](const auto &mask) { | 
					
						
							|  |  |  |         for (const int64_t i : mask) { | 
					
						
							|  |  |  |           element_fn(mut1[i]); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |       }); | 
					
						
							| 
									
										
										
										
											2020-06-22 15:50:08 +02:00
										 |  |  |     }; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-03 17:37:25 -05:00
										 |  |  |   void call(IndexMask mask, MFParams params, MFContext /*context*/) const override | 
					
						
							| 
									
										
										
										
											2020-06-22 15:50:08 +02:00
										 |  |  |   { | 
					
						
							|  |  |  |     MutableSpan<Mut1> mut1 = params.single_mutable<Mut1>(0); | 
					
						
							| 
									
										
										
										
											2020-07-03 14:20:42 +02:00
										 |  |  |     function_(mask, mut1); | 
					
						
							| 
									
										
										
										
											2020-06-22 15:50:08 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-07 19:34:35 +02:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * A multi-function that outputs the same value every time. The value is not owned by an instance | 
					
						
							| 
									
										
										
										
											2021-09-11 13:05:20 +02:00
										 |  |  |  * of this function. If #make_value_copy is false, the caller is responsible for destructing and | 
					
						
							|  |  |  |  * freeing the value. | 
					
						
							| 
									
										
										
										
											2020-07-07 19:34:35 +02:00
										 |  |  |  */ | 
					
						
							|  |  |  | class CustomMF_GenericConstant : public MultiFunction { | 
					
						
							|  |  |  |  private: | 
					
						
							|  |  |  |   const CPPType &type_; | 
					
						
							|  |  |  |   const void *value_; | 
					
						
							| 
									
										
										
										
											2021-03-22 11:57:24 +01:00
										 |  |  |   MFSignature signature_; | 
					
						
							| 
									
										
										
										
											2021-09-11 13:05:20 +02:00
										 |  |  |   bool owns_value_; | 
					
						
							| 
									
										
										
										
											2020-07-07 19:34:35 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-08 15:04:28 +02:00
										 |  |  |   template<typename T> friend class CustomMF_Constant; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-07 19:34:35 +02:00
										 |  |  |  public: | 
					
						
							| 
									
										
										
										
											2021-09-11 13:05:20 +02:00
										 |  |  |   CustomMF_GenericConstant(const CPPType &type, const void *value, bool make_value_copy); | 
					
						
							|  |  |  |   ~CustomMF_GenericConstant(); | 
					
						
							| 
									
										
										
										
											2020-07-07 19:34:35 +02:00
										 |  |  |   void call(IndexMask mask, MFParams params, MFContext context) const override; | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  |   uint64_t hash() const override; | 
					
						
							| 
									
										
										
										
											2020-07-08 15:04:28 +02:00
										 |  |  |   bool equals(const MultiFunction &other) const override; | 
					
						
							| 
									
										
										
										
											2020-07-07 19:34:35 +02:00
										 |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | /**
 | 
					
						
							|  |  |  |  * A multi-function that outputs the same array every time. The array is not owned by in instance | 
					
						
							|  |  |  |  * of this function. The caller is responsible for destructing and freeing the values. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | class CustomMF_GenericConstantArray : public MultiFunction { | 
					
						
							|  |  |  |  private: | 
					
						
							|  |  |  |   GSpan array_; | 
					
						
							| 
									
										
										
										
											2021-03-22 11:57:24 +01:00
										 |  |  |   MFSignature signature_; | 
					
						
							| 
									
										
										
										
											2020-07-07 19:34:35 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |  public: | 
					
						
							|  |  |  |   CustomMF_GenericConstantArray(GSpan array); | 
					
						
							|  |  |  |   void call(IndexMask mask, MFParams params, MFContext context) const override; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-08 15:04:28 +02:00
										 |  |  | /**
 | 
					
						
							|  |  |  |  * Generates a multi-function that outputs a constant value. | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | template<typename T> class CustomMF_Constant : public MultiFunction { | 
					
						
							|  |  |  |  private: | 
					
						
							|  |  |  |   T value_; | 
					
						
							| 
									
										
										
										
											2021-03-22 11:57:24 +01:00
										 |  |  |   MFSignature signature_; | 
					
						
							| 
									
										
										
										
											2020-07-08 15:04:28 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |  public: | 
					
						
							|  |  |  |   template<typename U> CustomMF_Constant(U &&value) : value_(std::forward<U>(value)) | 
					
						
							|  |  |  |   { | 
					
						
							| 
									
										
										
										
											2021-03-22 11:57:24 +01:00
										 |  |  |     MFSignatureBuilder signature{"Constant"}; | 
					
						
							| 
									
										
										
										
											2021-11-21 12:37:04 +01:00
										 |  |  |     signature.single_output<T>("Value"); | 
					
						
							| 
									
										
										
										
											2021-03-22 11:57:24 +01:00
										 |  |  |     signature_ = signature.build(); | 
					
						
							|  |  |  |     this->set_signature(&signature_); | 
					
						
							| 
									
										
										
										
											2020-07-08 15:04:28 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-03 17:37:25 -05:00
										 |  |  |   void call(IndexMask mask, MFParams params, MFContext /*context*/) const override | 
					
						
							| 
									
										
										
										
											2020-07-08 15:04:28 +02:00
										 |  |  |   { | 
					
						
							|  |  |  |     MutableSpan<T> output = params.uninitialized_single_output<T>(0); | 
					
						
							| 
									
										
										
										
											2022-03-29 10:11:49 +02:00
										 |  |  |     mask.to_best_mask_type([&](const auto &mask) { | 
					
						
							|  |  |  |       for (const int64_t i : mask) { | 
					
						
							|  |  |  |         new (&output[i]) T(value_); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2020-07-08 15:04:28 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  |   uint64_t hash() const override | 
					
						
							| 
									
										
										
										
											2020-07-08 15:04:28 +02:00
										 |  |  |   { | 
					
						
							| 
									
										
										
										
											2021-03-25 16:01:28 +01:00
										 |  |  |     return get_default_hash(value_); | 
					
						
							| 
									
										
										
										
											2020-07-08 15:04:28 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   bool equals(const MultiFunction &other) const override | 
					
						
							|  |  |  |   { | 
					
						
							|  |  |  |     const CustomMF_Constant *other1 = dynamic_cast<const CustomMF_Constant *>(&other); | 
					
						
							|  |  |  |     if (other1 != nullptr) { | 
					
						
							|  |  |  |       return value_ == other1->value_; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     const CustomMF_GenericConstant *other2 = dynamic_cast<const CustomMF_GenericConstant *>( | 
					
						
							|  |  |  |         &other); | 
					
						
							|  |  |  |     if (other2 != nullptr) { | 
					
						
							| 
									
										
										
										
											2020-07-10 12:56:57 +02:00
										 |  |  |       const CPPType &type = CPPType::get<T>(); | 
					
						
							|  |  |  |       if (type == other2->type_) { | 
					
						
							| 
									
										
										
										
											2021-06-28 13:13:52 +02:00
										 |  |  |         return type.is_equal_or_false(static_cast<const void *>(&value_), other2->value_); | 
					
						
							| 
									
										
										
										
											2020-07-08 15:04:28 +02:00
										 |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-16 13:38:23 +02:00
										 |  |  | class CustomMF_DefaultOutput : public MultiFunction { | 
					
						
							|  |  |  |  private: | 
					
						
							| 
									
										
										
										
											2020-07-20 12:16:20 +02:00
										 |  |  |   int output_amount_; | 
					
						
							| 
									
										
										
										
											2021-03-22 11:57:24 +01:00
										 |  |  |   MFSignature signature_; | 
					
						
							| 
									
										
										
										
											2020-07-16 13:38:23 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |  public: | 
					
						
							| 
									
										
										
										
											2021-11-21 12:37:04 +01:00
										 |  |  |   CustomMF_DefaultOutput(Span<MFDataType> input_types, Span<MFDataType> output_types); | 
					
						
							| 
									
										
										
										
											2020-07-16 13:38:23 +02:00
										 |  |  |   void call(IndexMask mask, MFParams params, MFContext context) const override; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-09-09 12:54:20 +02:00
										 |  |  | class CustomMF_GenericCopy : public MultiFunction { | 
					
						
							|  |  |  |  private: | 
					
						
							|  |  |  |   MFSignature signature_; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |  public: | 
					
						
							| 
									
										
										
										
											2021-11-21 12:37:04 +01:00
										 |  |  |   CustomMF_GenericCopy(MFDataType data_type); | 
					
						
							| 
									
										
										
										
											2021-09-09 12:54:20 +02:00
										 |  |  |   void call(IndexMask mask, MFParams params, MFContext context) const override; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2020-07-03 14:25:20 +02:00
										 |  |  | }  // namespace blender::fn
 |