[Bf-blender-cvs] [4e10b196ac1] master: Functions: make copying virtual arrays to span more efficient

Jacques Lucke noreply at git.blender.org
Thu Apr 29 12:59:52 CEST 2021


Commit: 4e10b196ac15339cfded8d5615f04ac40c93e19b
Author: Jacques Lucke
Date:   Thu Apr 29 12:59:44 2021 +0200
Branches: master
https://developer.blender.org/rB4e10b196ac15339cfded8d5615f04ac40c93e19b

Functions: make copying virtual arrays to span more efficient

Sometimes functions expect a span instead of a virtual array.
If the virtual array is a span internally already, great. But if it is
not (e.g. the position attribute on a mesh), the elements have
to be copied over to a span.

This patch makes the copying process more efficient by giving
the compiler more opportunity for optimization.

===================================================================

M	source/blender/blenlib/BLI_virtual_array.hh
M	source/blender/functions/FN_generic_virtual_array.hh
M	source/blender/functions/intern/generic_virtual_array.cc

===================================================================

diff --git a/source/blender/blenlib/BLI_virtual_array.hh b/source/blender/blenlib/BLI_virtual_array.hh
index eae15f0300c..1c02bce8411 100644
--- a/source/blender/blenlib/BLI_virtual_array.hh
+++ b/source/blender/blenlib/BLI_virtual_array.hh
@@ -38,6 +38,7 @@
  */
 
 #include "BLI_array.hh"
+#include "BLI_index_mask.hh"
 #include "BLI_span.hh"
 
 namespace blender {
@@ -127,14 +128,25 @@ template<typename T> class VArray {
   /* Copy the entire virtual array into a span. */
   void materialize(MutableSpan<T> r_span) const
   {
-    BLI_assert(size_ == r_span.size());
-    this->materialize_impl(r_span);
+    this->materialize(IndexMask(size_), r_span);
+  }
+
+  /* Copy some indices of the virtual array into a span. */
+  void materialize(IndexMask mask, MutableSpan<T> r_span) const
+  {
+    BLI_assert(mask.min_array_size() <= size_);
+    this->materialize_impl(mask, r_span);
   }
 
   void materialize_to_uninitialized(MutableSpan<T> r_span) const
   {
-    BLI_assert(size_ == r_span.size());
-    this->materialize_to_uninitialized_impl(r_span);
+    this->materialize_to_uninitialized(IndexMask(size_), r_span);
+  }
+
+  void materialize_to_uninitialized(IndexMask mask, MutableSpan<T> r_span) const
+  {
+    BLI_assert(mask.min_array_size() <= size_);
+    this->materialize_to_uninitialized_impl(mask, r_span);
   }
 
  protected:
@@ -164,40 +176,35 @@ template<typename T> class VArray {
     return T();
   }
 
-  virtual void materialize_impl(MutableSpan<T> r_span) const
+  virtual void materialize_impl(IndexMask mask, MutableSpan<T> r_span) const
   {
+    T *dst = r_span.data();
     if (this->is_span()) {
-      const Span<T> span = this->get_internal_span();
-      initialized_copy_n(span.data(), size_, r_span.data());
+      const T *src = this->get_internal_span().data();
+      mask.foreach_index([&](const int64_t i) { dst[i] = src[i]; });
     }
     else if (this->is_single()) {
       const T single = this->get_internal_single();
-      initialized_fill_n(r_span.data(), size_, single);
+      mask.foreach_index([&](const int64_t i) { dst[i] = single; });
     }
     else {
-      const int64_t size = size_;
-      for (int64_t i = 0; i < size; i++) {
-        r_span[i] = this->get(i);
-      }
+      mask.foreach_index([&](const int64_t i) { dst[i] = this->get(i); });
     }
   }
 
-  virtual void materialize_to_uninitialized_impl(MutableSpan<T> r_span) const
+  virtual void materialize_to_uninitialized_impl(IndexMask mask, MutableSpan<T> r_span) const
   {
+    T *dst = r_span.data();
     if (this->is_span()) {
-      const Span<T> span = this->get_internal_span();
-      uninitialized_copy_n(span.data(), size_, r_span.data());
+      const T *src = this->get_internal_span().data();
+      mask.foreach_index([&](const int64_t i) { new (dst + i) T(src[i]); });
     }
     else if (this->is_single()) {
       const T single = this->get_internal_single();
-      uninitialized_fill_n(r_span.data(), size_, single);
+      mask.foreach_index([&](const int64_t i) { new (dst + i) T(single); });
     }
     else {
-      const int64_t size = size_;
-      T *dst = r_span.data();
-      for (int64_t i = 0; i < size; i++) {
-        new (dst + i) T(this->get(i));
-      }
+      mask.foreach_index([&](const int64_t i) { new (dst + i) T(this->get(i)); });
     }
   }
 };
@@ -494,6 +501,18 @@ template<typename T, typename GetFunc> class VArray_For_Func final : public VArr
   {
     return get_func_(index);
   }
+
+  void materialize_impl(IndexMask mask, MutableSpan<T> r_span) const override
+  {
+    T *dst = r_span.data();
+    mask.foreach_index([&](const int64_t i) { dst[i] = get_func_(i); });
+  }
+
+  void materialize_to_uninitialized_impl(IndexMask mask, MutableSpan<T> r_span) const override
+  {
+    T *dst = r_span.data();
+    mask.foreach_index([&](const int64_t i) { new (dst + i) T(get_func_(i)); });
+  }
 };
 
 template<typename StructT, typename ElemT, ElemT (*GetFunc)(const StructT &)>
@@ -511,6 +530,18 @@ class VArray_For_DerivedSpan : public VArray<ElemT> {
   {
     return GetFunc(data_[index]);
   }
+
+  void materialize_impl(IndexMask mask, MutableSpan<ElemT> r_span) const override
+  {
+    ElemT *dst = r_span.data();
+    mask.foreach_index([&](const int64_t i) { dst[i] = GetFunc(data_[i]); });
+  }
+
+  void materialize_to_uninitialized_impl(IndexMask mask, MutableSpan<ElemT> r_span) const override
+  {
+    ElemT *dst = r_span.data();
+    mask.foreach_index([&](const int64_t i) { new (dst + i) ElemT(GetFunc(data_[i])); });
+  }
 };
 
 template<typename StructT,
@@ -537,6 +568,18 @@ class VMutableArray_For_DerivedSpan : public VMutableArray<ElemT> {
   {
     SetFunc(data_[index], std::move(value));
   }
+
+  void materialize_impl(IndexMask mask, MutableSpan<ElemT> r_span) const override
+  {
+    ElemT *dst = r_span.data();
+    mask.foreach_index([&](const int64_t i) { dst[i] = GetFunc(data_[i]); });
+  }
+
+  void materialize_to_uninitialized_impl(IndexMask mask, MutableSpan<ElemT> r_span) const override
+  {
+    ElemT *dst = r_span.data();
+    mask.foreach_index([&](const int64_t i) { new (dst + i) ElemT(GetFunc(data_[i])); });
+  }
 };
 
 /**
diff --git a/source/blender/functions/FN_generic_virtual_array.hh b/source/blender/functions/FN_generic_virtual_array.hh
index c1af00fd4cd..848deb6bc04 100644
--- a/source/blender/functions/FN_generic_virtual_array.hh
+++ b/source/blender/functions/FN_generic_virtual_array.hh
@@ -128,6 +128,7 @@ class GVArray {
     this->get_internal_single(r_value);
   }
 
+  void materialize_to_uninitialized(void *dst) const;
   void materialize_to_uninitialized(const IndexMask mask, void *dst) const;
 
   template<typename T> const VArray<T> *try_get_internal_varray() const
@@ -152,6 +153,8 @@ class GVArray {
   virtual bool is_single_impl() const;
   virtual void get_internal_single_impl(void *UNUSED(r_value)) const;
 
+  virtual void materialize_to_uninitialized_impl(const IndexMask mask, void *dst) const;
+
   virtual const void *try_get_internal_varray_impl() const;
 };
 
@@ -361,6 +364,11 @@ template<typename T> class GVArray_For_VArray : public GVArray {
     *(T *)r_value = varray_->get_internal_single();
   }
 
+  void materialize_to_uninitialized_impl(const IndexMask mask, void *dst) const override
+  {
+    varray_->materialize_to_uninitialized(mask, MutableSpan((T *)dst, mask.min_array_size()));
+  }
+
   const void *try_get_internal_varray_impl() const override
   {
     return varray_;
diff --git a/source/blender/functions/intern/generic_virtual_array.cc b/source/blender/functions/intern/generic_virtual_array.cc
index e11501828f8..754a2156a65 100644
--- a/source/blender/functions/intern/generic_virtual_array.cc
+++ b/source/blender/functions/intern/generic_virtual_array.cc
@@ -22,7 +22,18 @@ namespace blender::fn {
  * GVArray.
  */
 
+void GVArray::materialize_to_uninitialized(void *dst) const
+{
+  this->materialize_to_uninitialized(IndexMask(size_), dst);
+}
+
 void GVArray::materialize_to_uninitialized(const IndexMask mask, void *dst) const
+{
+  BLI_assert(mask.min_array_size() <= size_);
+  this->materialize_to_uninitialized_impl(mask, dst);
+}
+
+void GVArray::materialize_to_uninitialized_impl(const IndexMask mask, void *dst) const
 {
   for (const int64_t i : mask) {
     void *elem_dst = POINTER_OFFSET(dst, type_->size() * i);



More information about the Bf-blender-cvs mailing list