This moves `PointCloud` to use the recently added `AttributeStorage` at runtime. Mainly this involves implementing the higher level attribute API on top, and implementing the RNA API as well. The attribute RNA type is now backed by either CustomDataLayer or bke::Attribute. For now the new code is specific to point clouds but next steps can reuse it for Grease Pencil layer attributes, curves, and eventually meshes. Point cloud attributes no longer have a name length limit. Internally, the `AttributeStorage` API is extended with a few additions: - The data structs have static constructors for convenience. - A few functions give index-based access to attributes - A "rename" function is added. The `Attribute` RNA type now exposes a `storage_type` property. For now the "single value" option is still unused at runtime, and accessing the single value data isn't implemented yet. Pull Request: https://projects.blender.org/blender/blender/pulls/139165
510 lines
17 KiB
C++
510 lines
17 KiB
C++
/* SPDX-FileCopyrightText: 2023 Blender Authors
|
|
*
|
|
* SPDX-License-Identifier: GPL-2.0-or-later */
|
|
|
|
/** \file
|
|
* \ingroup bke
|
|
*/
|
|
|
|
#include <optional>
|
|
|
|
#include "MEM_guardedalloc.h"
|
|
|
|
#include "DNA_defaults.h"
|
|
#include "DNA_material_types.h"
|
|
#include "DNA_object_types.h"
|
|
#include "DNA_pointcloud_types.h"
|
|
|
|
#include "BLI_bounds.hh"
|
|
#include "BLI_index_range.hh"
|
|
#include "BLI_resource_scope.hh"
|
|
#include "BLI_span.hh"
|
|
#include "BLI_utildefines.h"
|
|
#include "BLI_vector.hh"
|
|
|
|
#include "BKE_anim_data.hh"
|
|
#include "BKE_attribute_legacy_convert.hh"
|
|
#include "BKE_attribute_storage.hh"
|
|
#include "BKE_attribute_storage_blend_write.hh"
|
|
#include "BKE_bake_data_block_id.hh"
|
|
#include "BKE_customdata.hh"
|
|
#include "BKE_geometry_set.hh"
|
|
#include "BKE_idtype.hh"
|
|
#include "BKE_lib_id.hh"
|
|
#include "BKE_lib_query.hh"
|
|
#include "BKE_modifier.hh"
|
|
#include "BKE_object.hh"
|
|
#include "BKE_object_types.hh"
|
|
#include "BKE_pointcloud.hh"
|
|
|
|
#include "BLT_translation.hh"
|
|
|
|
#include "DEG_depsgraph_query.hh"
|
|
|
|
#include "BLO_read_write.hh"
|
|
|
|
using blender::CPPType;
|
|
using blender::float3;
|
|
using blender::IndexRange;
|
|
using blender::MutableSpan;
|
|
using blender::Span;
|
|
using blender::StringRef;
|
|
using blender::VArray;
|
|
using blender::Vector;
|
|
|
|
constexpr StringRef ATTR_POSITION = "position";
|
|
|
|
static void pointcloud_init_data(ID *id)
|
|
{
|
|
PointCloud *pointcloud = (PointCloud *)id;
|
|
BLI_assert(MEMCMP_STRUCT_AFTER_IS_ZERO(pointcloud, id));
|
|
|
|
MEMCPY_STRUCT_AFTER(pointcloud, DNA_struct_default_get(PointCloud), id);
|
|
|
|
new (&pointcloud->attribute_storage.wrap()) blender::bke::AttributeStorage();
|
|
pointcloud->runtime = new blender::bke::PointCloudRuntime();
|
|
}
|
|
|
|
static void pointcloud_copy_data(Main * /*bmain*/,
|
|
std::optional<Library *> /*owner_library*/,
|
|
ID *id_dst,
|
|
const ID *id_src,
|
|
const int /*flag*/)
|
|
{
|
|
PointCloud *pointcloud_dst = (PointCloud *)id_dst;
|
|
const PointCloud *pointcloud_src = (const PointCloud *)id_src;
|
|
pointcloud_dst->mat = static_cast<Material **>(MEM_dupallocN(pointcloud_src->mat));
|
|
|
|
new (&pointcloud_dst->attribute_storage.wrap())
|
|
blender::bke::AttributeStorage(pointcloud_src->attribute_storage.wrap());
|
|
|
|
pointcloud_dst->runtime = new blender::bke::PointCloudRuntime();
|
|
pointcloud_dst->runtime->bounds_cache = pointcloud_src->runtime->bounds_cache;
|
|
pointcloud_dst->runtime->bounds_with_radius_cache =
|
|
pointcloud_src->runtime->bounds_with_radius_cache;
|
|
pointcloud_dst->runtime->bvh_cache = pointcloud_src->runtime->bvh_cache;
|
|
if (pointcloud_src->runtime->bake_materials) {
|
|
pointcloud_dst->runtime->bake_materials =
|
|
std::make_unique<blender::bke::bake::BakeMaterialsList>(
|
|
*pointcloud_src->runtime->bake_materials);
|
|
}
|
|
|
|
pointcloud_dst->batch_cache = nullptr;
|
|
}
|
|
|
|
static void pointcloud_free_data(ID *id)
|
|
{
|
|
PointCloud *pointcloud = (PointCloud *)id;
|
|
BKE_animdata_free(&pointcloud->id, false);
|
|
BKE_pointcloud_batch_cache_free(pointcloud);
|
|
pointcloud->attribute_storage.wrap().~AttributeStorage();
|
|
MEM_SAFE_FREE(pointcloud->mat);
|
|
delete pointcloud->runtime;
|
|
}
|
|
|
|
static void pointcloud_foreach_id(ID *id, LibraryForeachIDData *data)
|
|
{
|
|
PointCloud *pointcloud = (PointCloud *)id;
|
|
for (int i = 0; i < pointcloud->totcol; i++) {
|
|
BKE_LIB_FOREACHID_PROCESS_IDSUPER(data, pointcloud->mat[i], IDWALK_CB_USER);
|
|
}
|
|
}
|
|
|
|
static void pointcloud_blend_write(BlendWriter *writer, ID *id, const void *id_address)
|
|
{
|
|
using namespace blender;
|
|
using namespace blender::bke;
|
|
PointCloud *pointcloud = (PointCloud *)id;
|
|
|
|
ResourceScope scope;
|
|
bke::AttributeStorage::BlendWriteData attribute_data{scope};
|
|
attribute_storage_blend_write_prepare(pointcloud->attribute_storage.wrap(), {}, attribute_data);
|
|
BLI_assert(pointcloud->pdata_legacy.totlayer == 0);
|
|
pointcloud->attribute_storage.dna_attributes = attribute_data.attributes.data();
|
|
pointcloud->attribute_storage.dna_attributes_num = attribute_data.attributes.size();
|
|
|
|
/* Write LibData */
|
|
BLO_write_id_struct(writer, PointCloud, id_address, &pointcloud->id);
|
|
BKE_id_blend_write(writer, &pointcloud->id);
|
|
|
|
/* Direct data */
|
|
pointcloud->attribute_storage.wrap().blend_write(*writer, attribute_data);
|
|
|
|
BLO_write_pointer_array(writer, pointcloud->totcol, pointcloud->mat);
|
|
}
|
|
|
|
static void pointcloud_blend_read_data(BlendDataReader *reader, ID *id)
|
|
{
|
|
PointCloud *pointcloud = (PointCloud *)id;
|
|
|
|
/* Geometry */
|
|
CustomData_blend_read(reader, &pointcloud->pdata_legacy, pointcloud->totpoint);
|
|
pointcloud->attribute_storage.wrap().blend_read(*reader);
|
|
|
|
/* Materials */
|
|
BLO_read_pointer_array(reader, pointcloud->totcol, (void **)&pointcloud->mat);
|
|
|
|
pointcloud->runtime = new blender::bke::PointCloudRuntime();
|
|
}
|
|
|
|
IDTypeInfo IDType_ID_PT = {
|
|
/*id_code*/ PointCloud::id_type,
|
|
/*id_filter*/ FILTER_ID_PT,
|
|
/*dependencies_id_types*/ FILTER_ID_MA,
|
|
/*main_listbase_index*/ INDEX_ID_PT,
|
|
/*struct_size*/ sizeof(PointCloud),
|
|
/*name*/ "PointCloud",
|
|
/*name_plural*/ N_("pointclouds"),
|
|
/*translation_context*/ BLT_I18NCONTEXT_ID_POINTCLOUD,
|
|
/*flags*/ IDTYPE_FLAGS_APPEND_IS_REUSABLE,
|
|
/*asset_type_info*/ nullptr,
|
|
|
|
/*init_data*/ pointcloud_init_data,
|
|
/*copy_data*/ pointcloud_copy_data,
|
|
/*free_data*/ pointcloud_free_data,
|
|
/*make_local*/ nullptr,
|
|
/*foreach_id*/ pointcloud_foreach_id,
|
|
/*foreach_cache*/ nullptr,
|
|
/*foreach_path*/ nullptr,
|
|
/*owner_pointer_get*/ nullptr,
|
|
|
|
/*blend_write*/ pointcloud_blend_write,
|
|
/*blend_read_data*/ pointcloud_blend_read_data,
|
|
/*blend_read_after_liblink*/ nullptr,
|
|
|
|
/*blend_read_undo_preserve*/ nullptr,
|
|
|
|
/*lib_override_apply_post*/ nullptr,
|
|
};
|
|
|
|
template<typename T>
|
|
static VArray<T> get_varray_attribute(const PointCloud &pointcloud,
|
|
const StringRef name,
|
|
const T default_value)
|
|
{
|
|
using namespace blender;
|
|
const bke::Attribute *attr = pointcloud.attribute_storage.wrap().lookup(name);
|
|
if (!attr) {
|
|
return VArray<T>::ForSingle(default_value, pointcloud.totpoint);
|
|
}
|
|
switch (attr->storage_type()) {
|
|
case bke::AttrStorageType::Array: {
|
|
const auto &data = std::get<bke::Attribute::ArrayData>(attr->data());
|
|
const Span span(static_cast<const T *>(data.data), data.size);
|
|
BLI_assert(data.size == pointcloud.totpoint);
|
|
return VArray<T>::ForSpan(span);
|
|
}
|
|
case bke::AttrStorageType::Single: {
|
|
const auto &data = std::get<bke::Attribute::SingleData>(attr->data());
|
|
return VArray<T>::ForSingle(*static_cast<const T *>(data.value), pointcloud.totpoint);
|
|
}
|
|
}
|
|
return VArray<T>::ForSingle(default_value, pointcloud.totpoint);
|
|
}
|
|
|
|
template<typename T>
|
|
static Span<T> get_span_attribute(const PointCloud &pointcloud, const StringRef name)
|
|
{
|
|
using namespace blender;
|
|
const bke::Attribute *attr = pointcloud.attribute_storage.wrap().lookup(name);
|
|
if (!attr) {
|
|
return {};
|
|
}
|
|
if (const auto *array_data = std::get_if<bke::Attribute::ArrayData>(&attr->data())) {
|
|
BLI_assert(array_data->size == pointcloud.totpoint);
|
|
return Span(static_cast<const T *>(array_data->data), array_data->size);
|
|
}
|
|
return {};
|
|
}
|
|
|
|
template<typename T>
|
|
static MutableSpan<T> get_mutable_attribute(PointCloud &pointcloud,
|
|
const StringRef name,
|
|
const T default_value = T())
|
|
{
|
|
using namespace blender;
|
|
if (pointcloud.totpoint <= 0) {
|
|
return {};
|
|
}
|
|
const bke::AttrType type = bke::cpp_type_to_attribute_type(CPPType::get<T>());
|
|
if (bke::Attribute *attr = pointcloud.attribute_storage.wrap().lookup(name)) {
|
|
if (attr->data_type() == type) {
|
|
if (const auto *single_data = std::get_if<bke::Attribute::SingleData>(&attr->data())) {
|
|
/* Convert single value storage to array storage. */
|
|
const GPointer g_value(CPPType::get<T>(), single_data->value);
|
|
attr->data_for_write() = bke::Attribute::ArrayData::ForValue(g_value, pointcloud.totpoint);
|
|
}
|
|
auto &array_data = std::get<bke::Attribute::ArrayData>(attr->data_for_write());
|
|
BLI_assert(array_data.size == pointcloud.totpoint);
|
|
return MutableSpan(static_cast<T *>(array_data.data), pointcloud.totpoint);
|
|
}
|
|
/* The attribute has the wrong type. This shouldn't happen for builtin attributes, but just in
|
|
* case, remove it. */
|
|
pointcloud.attribute_storage.wrap().remove(name);
|
|
}
|
|
bke::Attribute &attr = pointcloud.attribute_storage.wrap().add(
|
|
name,
|
|
bke::AttrDomain::Point,
|
|
type,
|
|
bke::Attribute::ArrayData::ForValue({CPPType::get<T>(), &default_value},
|
|
pointcloud.totpoint));
|
|
auto &array_data = std::get<bke::Attribute::ArrayData>(attr.data_for_write());
|
|
BLI_assert(array_data.size == pointcloud.totpoint);
|
|
return MutableSpan(static_cast<T *>(array_data.data), pointcloud.totpoint);
|
|
}
|
|
|
|
Span<float3> PointCloud::positions() const
|
|
{
|
|
return get_span_attribute<float3>(*this, "position");
|
|
}
|
|
MutableSpan<float3> PointCloud::positions_for_write()
|
|
{
|
|
return get_mutable_attribute<float3>(*this, "position");
|
|
}
|
|
|
|
VArray<float> PointCloud::radius() const
|
|
{
|
|
return get_varray_attribute<float>(*this, "radius", 0.01f);
|
|
}
|
|
MutableSpan<float> PointCloud::radius_for_write()
|
|
{
|
|
return get_mutable_attribute<float>(*this, "radius", 0.01f);
|
|
}
|
|
|
|
PointCloud *BKE_pointcloud_add(Main *bmain, const char *name)
|
|
{
|
|
PointCloud *pointcloud = BKE_id_new<PointCloud>(bmain, name);
|
|
|
|
return pointcloud;
|
|
}
|
|
|
|
PointCloud *BKE_pointcloud_new_nomain(const int totpoint)
|
|
{
|
|
PointCloud *pointcloud = static_cast<PointCloud *>(BKE_libblock_alloc(
|
|
nullptr, ID_PT, BKE_idtype_idcode_to_name(ID_PT), LIB_ID_CREATE_LOCALIZE));
|
|
|
|
BKE_libblock_init_empty(&pointcloud->id);
|
|
|
|
pointcloud->totpoint = totpoint;
|
|
|
|
pointcloud->attributes_for_write().add<float3>(
|
|
"position", blender::bke::AttrDomain::Point, blender::bke::AttributeInitConstruct());
|
|
|
|
return pointcloud;
|
|
}
|
|
|
|
void BKE_pointcloud_nomain_to_pointcloud(PointCloud *pointcloud_src, PointCloud *pointcloud_dst)
|
|
{
|
|
BLI_assert(pointcloud_src->id.tag & ID_TAG_NO_MAIN);
|
|
|
|
pointcloud_dst->totpoint = pointcloud_src->totpoint;
|
|
pointcloud_dst->attribute_storage.wrap() = std::move(pointcloud_src->attribute_storage.wrap());
|
|
pointcloud_dst->runtime->bounds_cache = pointcloud_src->runtime->bounds_cache;
|
|
pointcloud_dst->runtime->bounds_with_radius_cache =
|
|
pointcloud_src->runtime->bounds_with_radius_cache;
|
|
pointcloud_dst->runtime->bvh_cache = pointcloud_src->runtime->bvh_cache;
|
|
BKE_id_free(nullptr, pointcloud_src);
|
|
}
|
|
|
|
std::optional<blender::Bounds<float3>> PointCloud::bounds_min_max(const bool use_radius) const
|
|
{
|
|
using namespace blender;
|
|
using namespace blender::bke;
|
|
if (this->totpoint == 0) {
|
|
return std::nullopt;
|
|
}
|
|
if (use_radius) {
|
|
this->runtime->bounds_with_radius_cache.ensure([&](Bounds<float3> &r_bounds) {
|
|
const VArray<float> radius = this->radius();
|
|
if (const std::optional radius_single = radius.get_if_single()) {
|
|
r_bounds = *this->bounds_min_max(false);
|
|
r_bounds.pad(*radius_single);
|
|
return;
|
|
}
|
|
const Span radius_span = radius.get_internal_span();
|
|
r_bounds = *bounds::min_max_with_radii(this->positions(), radius_span);
|
|
});
|
|
}
|
|
else {
|
|
this->runtime->bounds_cache.ensure(
|
|
[&](Bounds<float3> &r_bounds) { r_bounds = *bounds::min_max(this->positions()); });
|
|
}
|
|
return use_radius ? this->runtime->bounds_with_radius_cache.data() :
|
|
this->runtime->bounds_cache.data();
|
|
}
|
|
|
|
std::optional<int> PointCloud::material_index_max() const
|
|
{
|
|
if (this->totpoint == 0) {
|
|
return std::nullopt;
|
|
}
|
|
std::optional<int> max_material_index = blender::bounds::max<int>(
|
|
this->attributes()
|
|
.lookup_or_default<int>("material_index", blender::bke::AttrDomain::Point, 0)
|
|
.varray);
|
|
if (max_material_index.has_value()) {
|
|
max_material_index = std::clamp(*max_material_index, 0, MAXMAT);
|
|
}
|
|
return max_material_index;
|
|
}
|
|
|
|
void PointCloud::count_memory(blender::MemoryCounter &memory) const
|
|
{
|
|
this->attribute_storage.wrap().count_memory(memory);
|
|
}
|
|
|
|
blender::bke::AttributeAccessor PointCloud::attributes() const
|
|
{
|
|
return blender::bke::AttributeAccessor(this,
|
|
blender::bke::pointcloud_attribute_accessor_functions());
|
|
}
|
|
|
|
blender::bke::MutableAttributeAccessor PointCloud::attributes_for_write()
|
|
{
|
|
return blender::bke::MutableAttributeAccessor(
|
|
this, blender::bke::pointcloud_attribute_accessor_functions());
|
|
}
|
|
|
|
bool BKE_pointcloud_attribute_required(const PointCloud * /*pointcloud*/,
|
|
const blender::StringRef name)
|
|
{
|
|
return name == ATTR_POSITION;
|
|
}
|
|
|
|
void pointcloud_copy_parameters(const PointCloud &src, PointCloud &dst)
|
|
{
|
|
dst.flag = src.flag;
|
|
MEM_SAFE_FREE(dst.mat);
|
|
dst.mat = MEM_malloc_arrayN<Material *>(src.totcol, __func__);
|
|
dst.totcol = src.totcol;
|
|
MutableSpan(dst.mat, dst.totcol).copy_from(Span(src.mat, src.totcol));
|
|
}
|
|
|
|
/* Dependency Graph */
|
|
|
|
PointCloud *BKE_pointcloud_copy_for_eval(const PointCloud *pointcloud_src)
|
|
{
|
|
return reinterpret_cast<PointCloud *>(
|
|
BKE_id_copy_ex(nullptr, &pointcloud_src->id, nullptr, LIB_ID_COPY_LOCALIZE));
|
|
}
|
|
|
|
static void pointcloud_evaluate_modifiers(Depsgraph *depsgraph,
|
|
Scene *scene,
|
|
Object *object,
|
|
blender::bke::GeometrySet &geometry_set)
|
|
{
|
|
/* Modifier evaluation modes. */
|
|
const bool use_render = (DEG_get_mode(depsgraph) == DAG_EVAL_RENDER);
|
|
const int required_mode = use_render ? eModifierMode_Render : eModifierMode_Realtime;
|
|
ModifierApplyFlag apply_flag = use_render ? MOD_APPLY_RENDER : MOD_APPLY_USECACHE;
|
|
const ModifierEvalContext mectx = {depsgraph, object, apply_flag};
|
|
|
|
BKE_modifiers_clear_errors(object);
|
|
|
|
/* Get effective list of modifiers to execute. Some effects like shape keys
|
|
* are added as virtual modifiers before the user created modifiers. */
|
|
VirtualModifierData virtual_modifier_data;
|
|
ModifierData *md = BKE_modifiers_get_virtual_modifierlist(object, &virtual_modifier_data);
|
|
|
|
/* Evaluate modifiers. */
|
|
for (; md; md = md->next) {
|
|
const ModifierTypeInfo *mti = BKE_modifier_get_info((ModifierType)md->type);
|
|
|
|
if (!BKE_modifier_is_enabled(scene, md, required_mode)) {
|
|
continue;
|
|
}
|
|
|
|
blender::bke::ScopedModifierTimer modifier_timer{*md};
|
|
|
|
if (mti->modify_geometry_set) {
|
|
mti->modify_geometry_set(md, &mectx, &geometry_set);
|
|
}
|
|
}
|
|
}
|
|
|
|
static PointCloud *take_pointcloud_ownership_from_geometry_set(
|
|
blender::bke::GeometrySet &geometry_set)
|
|
{
|
|
if (!geometry_set.has<blender::bke::PointCloudComponent>()) {
|
|
return nullptr;
|
|
}
|
|
blender::bke::PointCloudComponent &pointcloud_component =
|
|
geometry_set.get_component_for_write<blender::bke::PointCloudComponent>();
|
|
PointCloud *pointcloud = pointcloud_component.release();
|
|
if (pointcloud != nullptr) {
|
|
/* Add back, but as read-only non-owning component. */
|
|
pointcloud_component.replace(pointcloud, blender::bke::GeometryOwnershipType::ReadOnly);
|
|
}
|
|
else {
|
|
/* The component was empty, we can also remove it. */
|
|
geometry_set.remove<blender::bke::PointCloudComponent>();
|
|
}
|
|
return pointcloud;
|
|
}
|
|
|
|
void BKE_pointcloud_data_update(Depsgraph *depsgraph, Scene *scene, Object *object)
|
|
{
|
|
/* Free any evaluated data and restore original data. */
|
|
BKE_object_free_derived_caches(object);
|
|
|
|
/* Evaluate modifiers. */
|
|
PointCloud *pointcloud = static_cast<PointCloud *>(object->data);
|
|
blender::bke::GeometrySet geometry_set = blender::bke::GeometrySet::from_pointcloud(
|
|
pointcloud, blender::bke::GeometryOwnershipType::ReadOnly);
|
|
pointcloud_evaluate_modifiers(depsgraph, scene, object, geometry_set);
|
|
|
|
PointCloud *pointcloud_eval = take_pointcloud_ownership_from_geometry_set(geometry_set);
|
|
|
|
/* If the geometry set did not contain a point cloud, we still create an empty one. */
|
|
if (pointcloud_eval == nullptr) {
|
|
pointcloud_eval = BKE_pointcloud_new_nomain(0);
|
|
}
|
|
|
|
/* Assign evaluated object. */
|
|
const bool eval_is_owned = pointcloud_eval != pointcloud;
|
|
BKE_object_eval_assign_data(object, &pointcloud_eval->id, eval_is_owned);
|
|
object->runtime->geometry_set_eval = new blender::bke::GeometrySet(std::move(geometry_set));
|
|
}
|
|
|
|
void PointCloud::tag_positions_changed()
|
|
{
|
|
this->runtime->bounds_cache.tag_dirty();
|
|
this->runtime->bounds_with_radius_cache.tag_dirty();
|
|
this->runtime->bvh_cache.tag_dirty();
|
|
}
|
|
|
|
void PointCloud::tag_radii_changed()
|
|
{
|
|
this->runtime->bounds_with_radius_cache.tag_dirty();
|
|
}
|
|
|
|
/* Draw Cache */
|
|
|
|
void (*BKE_pointcloud_batch_cache_dirty_tag_cb)(PointCloud *pointcloud, int mode) = nullptr;
|
|
void (*BKE_pointcloud_batch_cache_free_cb)(PointCloud *pointcloud) = nullptr;
|
|
|
|
void BKE_pointcloud_batch_cache_dirty_tag(PointCloud *pointcloud, int mode)
|
|
{
|
|
if (pointcloud->batch_cache) {
|
|
BKE_pointcloud_batch_cache_dirty_tag_cb(pointcloud, mode);
|
|
}
|
|
}
|
|
|
|
void BKE_pointcloud_batch_cache_free(PointCloud *pointcloud)
|
|
{
|
|
if (pointcloud->batch_cache) {
|
|
BKE_pointcloud_batch_cache_free_cb(pointcloud);
|
|
}
|
|
}
|
|
|
|
namespace blender::bke {
|
|
|
|
PointCloud *pointcloud_new_no_attributes(int totpoint)
|
|
{
|
|
PointCloud *pointcloud = BKE_id_new_nomain<PointCloud>(nullptr);
|
|
pointcloud->totpoint = totpoint;
|
|
return pointcloud;
|
|
}
|
|
|
|
} // namespace blender::bke
|