Graphite Version 3
An experimental 3D geometry processing program
Loading...
Searching...
No Matches
attributes.h
Go to the documentation of this file.
1/*
2 * Copyright (c) 2000-2022 Inria
3 * All rights reserved.
4 *
5 * Redistribution and use in source and binary forms, with or without
6 * modification, are permitted provided that the following conditions are met:
7 *
8 * * Redistributions of source code must retain the above copyright notice,
9 * this list of conditions and the following disclaimer.
10 * * Redistributions in binary form must reproduce the above copyright notice,
11 * this list of conditions and the following disclaimer in the documentation
12 * and/or other materials provided with the distribution.
13 * * Neither the name of the ALICE Project-Team nor the names of its
14 * contributors may be used to endorse or promote products derived from this
15 * software without specific prior written permission.
16 *
17 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
18 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
19 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
20 * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE
21 * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
22 * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF
23 * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS
24 * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN
25 * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
26 * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE
27 * POSSIBILITY OF SUCH DAMAGE.
28 *
29 * Contact: Bruno Levy
30 *
31 * https://www.inria.fr/fr/bruno-levy
32 *
33 * Inria,
34 * Domaine de Voluceau,
35 * 78150 Le Chesnay - Rocquencourt
36 * FRANCE
37 *
38 */
39
40
41#ifndef GEOGRAM_BASIC_ATTRIBUTES
42#define GEOGRAM_BASIC_ATTRIBUTES
43
44
50#include <geogram/basic/logger.h>
51
52#include <map>
53#include <typeinfo>
54#include <set>
55#include <type_traits>
56
62namespace GEO {
63
64 class AttributeStore;
65
70 class GEOGRAM_API AttributeStoreObserver {
71 public:
72
77 base_addr_(nullptr), size_(0), dimension_(0),
78 disconnected_(false) {
79 }
80
88 void notify(
89 Memory::pointer base_addr, index_t size, index_t dim
90 ) {
91 base_addr_ = base_addr;
92 size_ = size;
93 dimension_ = dim;
94 }
95
100 index_t size() const {
101 return size_;
102 }
103
109 return dimension_;
110 }
111
119 return size_ * dimension_;
120 }
121
127
133
141 void disconnect() {
142 base_addr_ = nullptr;
143 size_ = 0;
144 dimension_ = 0;
145 disconnected_ = true;
146 }
147
148 protected:
149 Memory::pointer base_addr_;
150 index_t size_;
151 index_t dimension_;
152 bool disconnected_;
153 };
154
155
156 /*********************************************************************/
157
158
163 class GEOGRAM_API AttributeStoreCreator : public Counted {
164 public:
165
170
177
178
179 private:
180 std::string element_type_name_;
181 std::string element_typeid_name_;
182 };
183
189
195 class GEOGRAM_API AttributeStore {
196 public:
206 AttributeStore(size_t elemsize, index_t dim=1);
207
212
213
224 const std::string& type_name
225 ) const = 0;
226
232 virtual std::string element_typeid_name() const = 0;
233
238 index_t size() const {
239 return cached_size_;
240 }
241
248 return cached_capacity_;
249 }
250
255 virtual void resize(index_t new_size) = 0;
256
262 virtual void reserve(index_t new_capacity) = 0;
263
269 virtual void clear(bool keep_memory = false) = 0;
270
276 bool has_observers() const {
277 return !observers_.empty();
278 }
279
286 return dimension_;
287 }
288
298 virtual void redim(index_t dim) = 0;
299
300
318 virtual void apply_permutation(
319 const vector<index_t>& permutation
320 );
321
339 virtual void compress(const vector<index_t>& old2new);
340
348 virtual void zero();
349
355 virtual AttributeStore* clone() const = 0;
356
357
363 void copy_item(index_t to, index_t from) {
364 geo_debug_assert(from < cached_size_);
365 geo_debug_assert(to < cached_size_);
366 size_t item_size = element_size_ * dimension_;
367 Memory::copy(
368 cached_base_addr_+to*item_size,
369 cached_base_addr_+from*item_size,
370 item_size
371 );
372 }
373
379 geo_debug_assert(to < cached_size_);
380 size_t item_size = element_size_ * dimension_;
381 Memory::clear(
382 cached_base_addr_+to*item_size,
383 item_size
384 );
385 }
386
394 virtual void scale_item(index_t to, double s);
395
404 virtual void madd_item(index_t to, double s, index_t from);
405
411
416 void* data() {
417 return cached_base_addr_;
418 }
419
424 const void* data() const {
425 return cached_base_addr_;
426 }
427
432 size_t element_size() const {
433 return element_size_;
434 }
435
445 const std::string& element_type_name
446 ) {
447 return (
448 type_name_to_creator_.find(element_type_name) !=
449 type_name_to_creator_.end()
450 );
451 }
452
462 const std::string& element_typeid_name
463 ) {
464 return (
465 typeid_name_to_type_name_.find(element_typeid_name) !=
466 typeid_name_to_type_name_.end()
467 );
468 }
469
477 const std::string& element_type_name,
478 index_t dimension
479 ) {
480 geo_assert(element_type_name_is_known(element_type_name));
481 return type_name_to_creator_[element_type_name]->
482 create_attribute_store(dimension);
483 }
484
493 const std::string& element_typeid_name
494 ) {
495 geo_assert(element_typeid_name_is_known(element_typeid_name));
496 return typeid_name_to_type_name_[element_typeid_name];
497 }
498
508 const std::string& element_type_name
509 ) {
510 geo_assert(element_type_name_is_known(element_type_name));
511 return type_name_to_typeid_name_[element_type_name];
512 }
513
525 AttributeStoreCreator* creator,
526 const std::string& element_type_name,
527 const std::string& element_typeid_name
528 ) {
529 if(element_type_name_is_known(element_type_name)) {
530 Logger::warn("Attributes") << element_type_name
531 << " already registered"
532 << std::endl;
533 if(element_typeid_name_is_known(element_typeid_name)) {
534 bool already_registered_attribute_has_same_type = (
535 type_name_to_typeid_name_[element_type_name] ==
536 element_typeid_name
537 );
538 geo_assert(already_registered_attribute_has_same_type);
539 }
540 }
541 type_name_to_creator_[element_type_name] = creator;
542 typeid_name_to_type_name_[element_typeid_name] = element_type_name;
543 type_name_to_typeid_name_[element_type_name] = element_typeid_name;
544 }
545
546 protected:
555 virtual void notify(
556 Memory::pointer base_addr, index_t size, index_t dim
557 );
558
568
577
578
579 protected:
580 size_t element_size_;
581 index_t dimension_;
582 Memory::pointer cached_base_addr_;
583 index_t cached_size_;
584 index_t cached_capacity_;
585 std::set<AttributeStoreObserver*> observers_;
586 Process::spinlock lock_;
587
588 static std::map<std::string, AttributeStoreCreator_var>
589 type_name_to_creator_;
590
591 static std::map<std::string, std::string>
592 typeid_name_to_type_name_;
593
594 static std::map<std::string, std::string>
595 type_name_to_typeid_name_;
596
597 friend class AttributeStoreObserver;
598 };
599
600 /*********************************************************************/
601
607 template <class T> class TypedAttributeStore : public AttributeStore {
608 public:
609
617 AttributeStore(sizeof(T),dim) {
618 }
619
620 void resize(index_t new_size) override {
621 store_.resize(new_size*dimension_);
622 notify(
623 store_.empty() ? nullptr : Memory::pointer(store_.data()),
624 new_size,
625 dimension_
626 );
627 }
628
629 void reserve(index_t new_capacity) override {
630 if(new_capacity > capacity()) {
631 store_.reserve(new_capacity*dimension_);
632 cached_capacity_ = new_capacity;
633 notify(
634 store_.empty() ? nullptr : Memory::pointer(store_.data()),
635 size(),
636 dimension_
637 );
638 }
639 }
640
641 void clear(bool keep_memory=false) override {
642 if(keep_memory) {
643 store_.resize(0);
644 } else {
645 store_.clear();
646 }
647 notify(nullptr, 0, dimension_);
648 }
649
650
651 void redim(index_t dim) override {
652 if(dim == dimension()) {
653 return;
654 }
655 vector<T> new_store(size()*dim);
656 new_store.reserve(capacity()*dim);
657 index_t copy_dim = std::min(dim, dimension());
658 for(index_t i = 0; i < size(); ++i) {
659 for(index_t c = 0; c < copy_dim; ++c) {
660 new_store[dim * i + c] = store_[dimension_ * i + c];
661 }
662 }
663 store_.swap(new_store);
664 notify(
665 store_.empty() ? nullptr : Memory::pointer(store_.data()),
666 size(),
667 dim
668 );
669 }
670
671 bool elements_type_matches(const std::string& type_name) const override {
672 return type_name == typeid(T).name();
673 }
674
675 std::string element_typeid_name() const override {
676 return typeid(T).name();
677 }
678
679 AttributeStore* clone() const override {
680 TypedAttributeStore<T>* result =
682 result->resize(size());
683 result->store_ = store_;
684 return result;
685 }
686
687 vector<T>& get_vector() {
688 return store_;
689 }
690
691 void scale_item(index_t to, double s) override {
692 geo_assert(to < size());
693 for(index_t i=0; i<dimension_; ++i) {
694 scale_value(store_[to*dimension_+i], s);
695 }
696 }
697
698 void madd_item(index_t to, double s, index_t from) override {
699 geo_assert(from < size());
700 geo_assert(to < size());
701 for(index_t i=0; i<dimension_; ++i) {
702 madd_value(
703 store_[to*dimension_+i], s, store_[from*dimension_+i]
704 );
705 }
706 }
707
708 protected:
709 void notify(
710 Memory::pointer base_addr, index_t size, index_t dim
711 ) override {
712 AttributeStore::notify(base_addr, size, dim);
713 geo_assert(size*dim <= store_.size());
714 }
715
716 template<class TT> static void scale_value(TT& to, double s) {
717 geo_argused(to);
718 geo_argused(s);
719 }
720
721 static void scale_value(uint8_t& to, double s) {
722 to = uint8_t(double(to)*s != 0.0);
723 }
724
725 static void scale_value(int32_t& to, double s) {
726 to = int32_t(double(to)*s);
727 }
728
729 static void scale_value(uint32_t& to, double s) {
730 to = uint32_t(double(to)*s);
731 }
732
733 static void scale_value(float& to, double s) {
734 to = float(double(to)*s);
735 }
736
737 static void scale_value(double& to, double s) {
738 to *= s;
739 }
740
741 template<class TT> static void madd_value(TT& to, double s, TT& from) {
742 geo_argused(to);
743 geo_argused(s);
744 geo_argused(from);
745 }
746
747 static void madd_value(uint8_t& to, double s, uint8_t& from) {
748 to = uint8_t(double(to) + s*double(from) != 0.0);
749 }
750
751 static void madd_value(int32_t& to, double s, int32_t& from) {
752 to = int32_t(double(to) + s*double(from));
753 }
754
755 static void madd_value(uint32_t& to, double s, uint32_t& from) {
756 to = uint32_t(double(to) + s*double(from));
757 }
758
759 static void madd_value(float& to, double s, float& from) {
760 to = float(double(to) + s*double(from));
761 }
762
763 static void madd_value(double& to, double s, double& from) {
764 to += s*from;
765 }
766
767
768 private:
769 vector<T> store_;
770 };
771
772 /*********************************************************************/
773
778 template <class T>
780 public:
785 return new TypedAttributeStore<T>(dim);
786 }
787 };
788
789 /*********************************************************************/
790
795 template <class T> class geo_register_attribute_type {
796 public:
806 geo_register_attribute_type(const std::string& type_name) {
808 new TypedAttributeStoreCreator<T>, type_name, typeid(T).name()
809 );
810 if(type_name == "bool") {
812 type_name,
815 );
816 } else {
818 type_name,
819 read_ascii_attribute<T>,
820 write_ascii_attribute<T>
821 );
822 }
823 }
824 };
825
826 /*********************************************************************/
827
832 class GEOGRAM_API AttributesManager {
833 public:
838
839
844
850 index_t nb() const {
851 return index_t(attributes_.size());
852 }
853
860
867 index_t size() const {
868 return size_;
869 }
870
877 return capacity_;
878 }
879
886 void resize(index_t new_size);
887
894 void reserve(index_t new_capacity);
895
904 void clear(bool keep_attributes, bool keep_memory = false);
905
906
910 void zero();
911
920 void bind_attribute_store(const std::string& name, AttributeStore* as);
921
928 AttributeStore* find_attribute_store(const std::string& name);
929
937 const std::string& name
938 ) const;
939
940
947 bool is_defined(const std::string& name) const {
948 return (find_attribute_store(name) != nullptr);
949 }
950
956 void delete_attribute_store(const std::string& name);
957
964
981 const vector<index_t>& permutation
982 );
983
999 void compress(const vector<index_t>& old2new);
1000
1005 void copy(const AttributesManager& rhs);
1006
1007
1014 void copy_item(index_t to, index_t from);
1015
1022
1023
1029
1037 void scale_item(index_t to, double s);
1038
1047 void madd_item(index_t to, double s, index_t from);
1048
1058 const std::string& name, const std::string& new_name
1059 );
1060
1070 const std::string& old_name, const std::string& new_name
1071 );
1072
1073 private:
1082
1090 const AttributesManager& operator=(const AttributesManager& rhs);
1091
1092 private:
1093 index_t size_;
1094 index_t capacity_;
1095 std::map<std::string, AttributeStore*> attributes_;
1096 } ;
1097
1098
1099 /*********************************************************************/
1100
1101
1106 template <class T> class AttributeBase : public AttributeStoreObserver {
1107 public:
1108
1113 manager_(nullptr),
1114 store_(nullptr) {
1115 }
1116
1126 AttributeBase(AttributesManager& manager, const std::string& name) :
1127 manager_(nullptr),
1128 store_(nullptr) {
1129 bind(manager, name);
1130 }
1131
1137 bool is_bound() const {
1138 return (store_ != nullptr && !disconnected_);
1139 }
1140
1145 void unbind() {
1147 // If the AttributesManager was destroyed before, do not
1148 // do anything. This can occur in Lua scripting when using
1149 // Attribute wrapper objects.
1150 if(!disconnected_) {
1151 unregister_me(store_);
1152 }
1153 manager_ = nullptr;
1154 store_ = nullptr;
1155 }
1156
1166 void bind(AttributesManager& manager, const std::string& name) {
1167 geo_assert(!is_bound());
1168 manager_ = &manager;
1169 store_ = manager_->find_attribute_store(name);
1170 if(store_ == nullptr) {
1171 store_ = new TypedAttributeStore<T>();
1172 manager_->bind_attribute_store(name,store_);
1173 } else {
1174 geo_assert(store_->elements_type_matches(typeid(T).name()));
1175 }
1176 register_me(store_);
1177 }
1178
1179
1189 AttributesManager& manager, const std::string& name
1190 ) {
1191 geo_assert(!is_bound());
1192 manager_ = &manager;
1193 store_ = manager_->find_attribute_store(name);
1194 if(store_ != nullptr) {
1195 geo_assert(store_->elements_type_matches(typeid(T).name()));
1196 register_me(store_);
1197 return true;
1198 }
1199 return false;
1200 }
1201
1211 AttributesManager& manager, const std::string& name
1212 ) {
1213 if( is_bound() ) {
1214 unbind();
1215 }
1216 store_ = manager.find_attribute_store(name);
1217 if(store_ != nullptr) {
1218 if( !store_->elements_type_matches(typeid(T).name()) ) {
1219 store_ = nullptr;
1220 return false;
1221 }
1222 manager_ = &manager;
1223 register_me(store_);
1224 return true;
1225 }
1226 return false;
1227 }
1228
1237 const std::string& name,
1239 ) {
1240 geo_assert(!is_bound());
1241 manager_ = &manager;
1242 geo_assert(manager_->find_attribute_store(name) == nullptr);
1243 store_ = new TypedAttributeStore<T>(dimension);
1244 manager_->bind_attribute_store(name,store_);
1245 register_me(store_);
1246 }
1247
1254 void destroy() {
1256 unregister_me(store_);
1257 manager_->delete_attribute_store(store_);
1258 store_ = nullptr;
1259 manager_ = nullptr;
1260 }
1261
1271 void redim(index_t new_dim) {
1273 store_->redim(new_dim);
1274 }
1275
1284 if(is_bound()) {
1285 unbind();
1286 }
1287 }
1288
1289
1297 static bool is_defined(
1298 AttributesManager& manager, const std::string& name,
1299 index_t dim = 0
1300 ) {
1302 return (
1303 store != nullptr &&
1304 store->elements_type_matches(typeid(T).name()) &&
1305 ((dim == 0) || (store->dimension() == dim))
1306 );
1307 }
1308
1313 index_t size() const {
1314 return size_;
1315 }
1316
1321 void zero() {
1323 store_->zero();
1324 }
1325
1336 return(
1337 dynamic_cast<TypedAttributeStore<T>*>(store_) != nullptr
1338 );
1339 }
1340
1352 TypedAttributeStore<T>* typed_store =
1353 dynamic_cast<TypedAttributeStore<T>*>(store_);
1354 geo_assert(typed_store != nullptr);
1355 return typed_store->get_vector();
1356 }
1357
1364 const vector<T>& get_vector() const {
1365 TypedAttributeStore<T>* typed_store =
1366 dynamic_cast<TypedAttributeStore<T>*>(store_);
1367 geo_assert(typed_store != nullptr);
1368 return typed_store->get_vector();
1369 }
1370
1376 return manager_;
1377 }
1378
1379 protected:
1380 AttributesManager* manager_;
1381 AttributeStore* store_;
1382 } ;
1383
1384 /*********************************************************************/
1385
1392 template <class T> class Attribute : public AttributeBase<T> {
1393 public:
1395
1401 static void static_test_type() {
1402 // Attributes are only implemented for classes that
1403 // can be copied with memcpy() and read/written to
1404 // files using fread()/fwrite()
1405#if __GNUG__ && __GNUC__ < 5 && !__clang__
1406 static_assert(
1407 __has_trivial_copy(T),
1408 "Attribute only implemented for types that can be copied with memcpy()"
1409 );
1410#else
1411 static_assert(
1412 std::is_trivially_copyable<T>::value,
1413 "Attribute only implemented for types that can be copied with memcpy()"
1414 );
1415#endif
1416 }
1417
1423 }
1424
1434 Attribute(AttributesManager& manager, const std::string& name) :
1435 superclass(manager, name) {
1437 }
1438
1446 return ((T*)(void*)superclass::base_addr_)[i];
1447 }
1448
1454 const T& operator[](index_t i) const {
1456 return ((const T*)(void*)superclass::base_addr_)[i];
1457 }
1458
1464 void fill(const T& val) {
1465 for(index_t i=0; i<superclass::nb_elements(); ++i) {
1466 (*this)[i] = val;
1467 }
1468 }
1469
1476 void copy(const Attribute<T>& rhs) {
1477 geo_assert(rhs.size() == superclass::size());
1479 for(index_t i=0; i<superclass::nb_elements(); ++i) {
1480 (*this)[i] = rhs[i];
1481 }
1482 }
1483
1488 T* data() {
1489 return (T*)AttributeStoreObserver::base_addr_;
1490 }
1491
1496 const T* data() const {
1497 return (const T*)AttributeStoreObserver::base_addr_;
1498 }
1499
1500
1501 private:
1505 Attribute(const Attribute<T>& rhs);
1509 Attribute<T>& operator=(const Attribute<T>& rhs);
1510 };
1511
1512 /*********************************************************************/
1513
1522 template <> class Attribute<bool> : public AttributeBase<Numeric::uint8> {
1523 public:
1525
1526 Attribute() : superclass() {
1527 }
1528
1529 Attribute(AttributesManager& manager, const std::string& name) :
1530 superclass(manager,name) {
1531 }
1532
1533 class BoolAttributeAccessor;
1534
1535
1540 class ConstBoolAttributeAccessor {
1541 public:
1546 const Attribute<bool>& attribute,
1547 index_t index
1548 ) :
1549 attribute_(&attribute),
1550 index_(index) {
1551 }
1552
1557 operator bool() const {
1558 return (attribute_->element(index_) != 0);
1559 }
1560
1561 private:
1562 const Attribute<bool>* attribute_;
1563 index_t index_;
1564
1565 friend class BoolAttributeAccessor;
1566 };
1567
1572 class BoolAttributeAccessor {
1573 public:
1578 Attribute<bool>& attribute,
1579 index_t index
1580 ) :
1581 attribute_(&attribute),
1582 index_(index) {
1583 }
1584
1589 operator bool() const {
1590 return (attribute_->element(index_) != 0);
1591 }
1592
1598 BoolAttributeAccessor(const BoolAttributeAccessor& rhs) {
1599 attribute_ = rhs.attribute_;
1600 index_ = rhs.index_;
1601 }
1602
1607 BoolAttributeAccessor& operator=(bool x) {
1608 attribute_->element(index_) = Numeric::uint8(x);
1609 return *this;
1610 }
1611
1617 BoolAttributeAccessor& operator=(
1618 const BoolAttributeAccessor& rhs
1619 ) {
1620 if(&rhs != this) {
1621 attribute_->element(index_) =
1622 rhs.attribute_->element(rhs.index_);
1623 }
1624 return *this;
1625 }
1626
1632 BoolAttributeAccessor& operator=(
1633 const ConstBoolAttributeAccessor& rhs
1634 ) {
1635 attribute_->element(index_) =
1636 rhs.attribute_->element(rhs.index_);
1637 return *this;
1638 }
1639
1640 private:
1641 Attribute<bool>* attribute_;
1642 index_t index_;
1643 };
1644
1645
1646 BoolAttributeAccessor operator[](index_t i) {
1647 return BoolAttributeAccessor(*this,i);
1648 }
1649
1650 ConstBoolAttributeAccessor operator[](index_t i) const {
1651 return ConstBoolAttributeAccessor(*this,i);
1652 }
1653
1659 void fill(bool val) {
1660 for(index_t i=0; i<superclass::nb_elements(); ++i) {
1661 element(i) = Numeric::uint8(val);
1662 }
1663 }
1664
1665 protected:
1666
1667 friend class BoolAttributeAccessor;
1668 friend class ConstBoolAttributeAccessor;
1669
1677 return ((Numeric::uint8*)superclass::base_addr_)[i];
1678 }
1679
1687 return ((const Numeric::uint8*)superclass::base_addr_)[i];
1688 }
1689
1690 private:
1694 Attribute(const Attribute<bool>& rhs);
1698 Attribute<bool>& operator=(const Attribute<bool>& rhs);
1699 } ;
1700
1701 /***********************************************************/
1702
1707 class GEOGRAM_API ScalarAttributeAdapterBase :
1708 public AttributeStoreObserver {
1709
1710 public:
1715 ET_NONE=0,
1716 ET_UINT8=1,
1717 ET_INT8=2,
1718 ET_UINT32=3,
1719 ET_INT32=4,
1720 ET_FLOAT32=5,
1721 ET_FLOAT64=6,
1722 ET_VEC2=7,
1723 ET_VEC3=8
1724 };
1725
1726
1731 class Accessor {
1732 public:
1733 Accessor(
1734 ScalarAttributeAdapterBase& attribute,
1735 index_t index
1736 ) : attribute_(attribute), index_(index) {
1737 }
1738
1739 operator double() const {
1740 return attribute_.get_element_as_double(index_);
1741 }
1742
1743 void operator=(double x) {
1744 attribute_.set_element_as_double(index_, x);
1745 }
1746
1747 private:
1748 ScalarAttributeAdapterBase& attribute_;
1749 index_t index_;
1750 };
1751
1757 public:
1759 const ScalarAttributeAdapterBase& attribute,
1760 index_t index
1761 ) : attribute_(attribute), index_(index) {
1762 }
1763
1764 operator double() const {
1765 return attribute_.get_element_as_double(index_);
1766 }
1767
1768 private:
1769 const ScalarAttributeAdapterBase& attribute_;
1770 index_t index_;
1771 };
1772
1777 manager_(nullptr),
1778 store_(nullptr),
1779 element_type_(ET_NONE),
1780 element_index_(index_t(-1)) {
1781 }
1782
1793 const AttributesManager& manager, const std::string& name
1794 ) :
1795 manager_(nullptr),
1796 store_(nullptr) {
1797 bind_if_is_defined(manager, name);
1798 }
1799
1805 bool is_bound() const {
1806 return (store_ != nullptr);
1807 }
1808
1813 void unbind() {
1814 geo_assert(is_bound());
1815 unregister_me(const_cast<AttributeStore*>(store_));
1816 manager_ = nullptr;
1817 store_ = nullptr;
1818 element_type_ = ET_NONE;
1819 element_index_ = index_t(-1);
1820 }
1821
1832 const AttributesManager& manager, const std::string& name
1833 );
1834
1843 if(is_bound()) {
1844 unbind();
1845 }
1846 }
1847
1857 static bool is_defined(
1858 const AttributesManager& manager, const std::string& name
1859 );
1860
1865 index_t size() const {
1866 return (store_ == nullptr) ? 0 : store_->size();
1867 }
1868
1877 return element_type_;
1878 }
1879
1887 return element_index_;
1888 }
1889
1895 return store_;
1896 }
1897
1898
1906 static bool can_be_bound_to(const AttributeStore* store) {
1907 return element_type(store) != ET_NONE;
1908 }
1909
1918
1919 protected:
1928 static std::string attribute_base_name(const std::string& name);
1929
1938 static index_t attribute_element_index(const std::string& name);
1939
1948
1957 double result = 0.0;
1958 switch(element_type()) {
1959 case ET_UINT8:
1960 result = double(get_element<Numeric::uint8>(i));
1961 break;
1962 case ET_INT8:
1963 result = double(get_element<Numeric::int8>(i));
1964 break;
1965 case ET_UINT32:
1966 result = double(get_element<Numeric::uint32>(i));
1967 break;
1968 case ET_INT32:
1969 result = double(get_element<Numeric::int32>(i));
1970 break;
1971 case ET_FLOAT32:
1972 result = double(get_element<Numeric::float32>(i));
1973 break;
1974 case ET_FLOAT64:
1975 result = double(get_element<Numeric::float64>(i));
1976 break;
1977 case ET_VEC2:
1978 result = double(get_element<Numeric::float64>(i,2));
1979 break;
1980 case ET_VEC3:
1981 result = double(get_element<Numeric::float64>(i,3));
1982 break;
1983 case ET_NONE:
1985 }
1986 return result;
1987 }
1988
1997 template <class T> T get_element(index_t i,index_t multiplier=1) const {
1998 geo_debug_assert(is_bound());
1999 geo_debug_assert(i < size());
2000 return static_cast<const T*>(store_->data())[
2001 (i * store_->dimension() * multiplier) +
2002 element_index_
2003 ];
2004 }
2005
2012 double set_element_as_double(index_t i, double value) {
2013 double result = 0.0;
2014 switch(element_type()) {
2015 case ET_UINT8:
2016 set_element<Numeric::uint8>(Numeric::uint8(value), i);
2017 break;
2018 case ET_INT8:
2019 set_element<Numeric::int8>(Numeric::int8(value),i);
2020 break;
2021 case ET_UINT32:
2022 set_element<Numeric::uint32>(Numeric::uint32(value),i);
2023 break;
2024 case ET_INT32:
2025 set_element<Numeric::int32>(Numeric::int32(value),i);
2026 break;
2027 case ET_FLOAT32:
2028 set_element<Numeric::float32>(Numeric::float32(value),i);
2029 break;
2030 case ET_FLOAT64:
2031 set_element<Numeric::float64>(Numeric::float64(value),i);
2032 break;
2033 case ET_VEC2:
2034 set_element<Numeric::float64>(Numeric::float64(value),i,2);
2035 break;
2036 case ET_VEC3:
2037 set_element<Numeric::float64>(Numeric::float64(value),i,3);
2038 break;
2039 case ET_NONE:
2041 }
2042 return result;
2043 }
2044
2054 template <class T> void set_element(
2055 T value, index_t i, index_t multiplier=1
2056 ) const {
2057 geo_debug_assert(is_bound());
2058 geo_debug_assert(i < size());
2059 const_cast<T*>(static_cast<const T*>(store_->data()))[
2060 (i * store_->dimension() * multiplier) +
2061 element_index_
2062 ] = value;
2063 }
2064
2065 protected:
2066 const AttributesManager* manager_;
2067 const AttributeStore* store_;
2068 ElementType element_type_;
2069 index_t element_index_;
2070 };
2071
2072 /***********************************************************/
2073
2079 public:
2081 }
2082
2084 const AttributesManager& manager, const std::string& name
2085 ) : ScalarAttributeAdapterBase(manager, name) {
2086 }
2087
2096 return get_element_as_double(i);
2097 }
2098 };
2099
2100 /***********************************************************/
2101
2107 public:
2109 }
2110
2112 const AttributesManager& manager, const std::string& name
2113 ) : ScalarAttributeAdapterBase(manager, name) {
2114 }
2115
2116 Accessor operator[](index_t i) {
2117 return Accessor(*this, i);
2118 }
2119
2120 ConstAccessor operator[](index_t i) const {
2121 return ConstAccessor(*this, i);
2122 }
2123
2124 protected:
2125 };
2126
2127
2128}
2129
2130#endif
#define geo_assert_not_reached
Sets a non reachable point in the program.
Definition assert.h:177
#define geo_assert(x)
Verifies that a condition is met.
Definition assert.h:149
#define geo_debug_assert(x)
Verifies that a condition is met.
Definition assert.h:196
Base class for Attributes, that manipulates an attribute stored in an AttributesManager.
void create_vector_attribute(AttributesManager &manager, const std::string &name, index_t dimension)
Creates and binds a new vector attribute.
AttributesManager * manager() const
Gets the AttributesManager this Attribute is bound to.
~AttributeBase()
Attribute destructor.
index_t size() const
Gets the size.
AttributeBase(AttributesManager &manager, const std::string &name)
Creates or retrieves a persistent attribute attached to a given AttributesManager.
const vector< T > & get_vector() const
Gets a const reference to the internal vector<T> used to store the attribute.
void zero()
Sets all the elements of this Attribute to zero.
bool is_bound() const
Tests whether an Attribute is bound.
void unbind()
Unbinds this Attribute.
void redim(index_t new_dim)
Sets the dimension.
void destroy()
Destroys this attribute in the AttributesManager.
vector< T > & get_vector()
Gets a reference to the internal vector<T> used to store the attribute.
bool bind_if_is_compatible(AttributesManager &manager, const std::string &name)
Binds this Attribute to an AttributesManager if it already exists in the AttributesManager and tyopes...
static bool is_defined(AttributesManager &manager, const std::string &name, index_t dim=0)
Tests whether an attribute with the specified name and with corresponding type exists in an Attribute...
bool can_get_vector()
Tests whether get_vector() can be called on this Attribute.
AttributeBase()
Creates an uninitialized (unbound) Attribute.
void bind(AttributesManager &manager, const std::string &name)
Binds this Attribute to an AttributesManager.
bool bind_if_is_defined(AttributesManager &manager, const std::string &name)
Binds this Attribute to an AttributesManager if it already exists in the AttributesManager.
Internal class for creating an AttributeStore from the type name of its elements.
Definition attributes.h:163
~AttributeStoreCreator() override
AttributeStoreCreator destructor.
virtual AttributeStore * create_attribute_store(index_t dimension)=0
Creates a new attribute store.
Base class for attributes. They are notified whenever the AttributeStore is modified.
Definition attributes.h:70
AttributeStoreObserver()
Creates a new uninitialied AttributeStore.
Definition attributes.h:76
void register_me(AttributeStore *store)
Registers this observer to an AttributeStore.
index_t size() const
Gets the size.
Definition attributes.h:100
void notify(Memory::pointer base_addr, index_t size, index_t dim)
Callback function, called by the AttributeStore whenever it is modified.
Definition attributes.h:88
index_t nb_elements() const
Gets the total number of elements.
Definition attributes.h:118
void disconnect()
Disconnects this AttributeStoreObserver from its AttributeStore.
Definition attributes.h:141
void unregister_me(AttributeStore *store)
Unregisters this observer from an AttributeStore.
index_t dimension() const
Gets the dimension.
Definition attributes.h:108
Notifies a set of AttributeStoreObservers each time the stored array changes size and/or base address...
Definition attributes.h:195
void * data()
Gets a pointer to the stored data.
Definition attributes.h:416
static AttributeStore * create_attribute_store_by_element_type_name(const std::string &element_type_name, index_t dimension)
Creates an attribute store of a given type.
Definition attributes.h:476
bool has_observers() const
Tests whether observers listen to this AttributeStore.
Definition attributes.h:276
void zero_item(index_t to)
Sets an item to zero.
Definition attributes.h:378
virtual ~AttributeStore()
AttributeStore destructor.
static std::string element_typeid_name_by_element_type_name(const std::string &element_type_name)
Gets an element mangled type name from its C++ name.
Definition attributes.h:507
static void register_attribute_creator(AttributeStoreCreator *creator, const std::string &element_type_name, const std::string &element_typeid_name)
Registers a new element type.
Definition attributes.h:524
index_t size() const
Gets the size.
Definition attributes.h:238
const void * data() const
Gets a pointer to the stored data.
Definition attributes.h:424
virtual void clear(bool keep_memory=false)=0
Resizes this AttributeStore to 0.
virtual void notify(Memory::pointer base_addr, index_t size, index_t dim)
If size or base address differ from the cached values, notify all the observers, and update the cache...
AttributeStore(size_t elemsize, index_t dim=1)
AttributeStore constructor.
index_t capacity() const
Gets the capacity.
Definition attributes.h:247
virtual void madd_item(index_t to, double s, index_t from)
Adds a scaled item to another item \detais item[to] += s * item[from].
static bool element_type_name_is_known(const std::string &element_type_name)
Tests whether a given element type is registered in the system.
Definition attributes.h:444
virtual void compress(const vector< index_t > &old2new)
Compresses the stored attributes, by applying an index mapping that fills-in the gaps.
virtual std::string element_typeid_name() const =0
Gets the typeid name of the element type stored in this AttributeStore.
void copy_item(index_t to, index_t from)
Copies an item.
Definition attributes.h:363
virtual AttributeStore * clone() const =0
Creates a new AttributeStore that is a carbon copy of this AttributeStore.
static bool element_typeid_name_is_known(const std::string &element_typeid_name)
Tests whether a given element type is registered in the system.
Definition attributes.h:461
void register_observer(AttributeStoreObserver *observer)
Registers an observer.
virtual void apply_permutation(const vector< index_t > &permutation)
Applies a permutation to the stored attributes.
virtual void resize(index_t new_size)=0
Resizes this AttributeStore.
virtual void scale_item(index_t to, double s)
Scales an item.
void unregister_observer(AttributeStoreObserver *observer)
Unregisters an observer.
virtual void redim(index_t dim)=0
Sets the dimension.
index_t dimension() const
Gets the dimension.
Definition attributes.h:285
static std::string element_type_name_by_element_typeid_name(const std::string &element_typeid_name)
Gets an element type name from its mangled name.
Definition attributes.h:492
virtual bool elements_type_matches(const std::string &type_name) const =0
Tests whether this AttributeStore stores elements of a given type.
void swap_items(index_t i, index_t j)
Swaps two items.
size_t element_size() const
Gets the element size.
Definition attributes.h:432
virtual void reserve(index_t new_capacity)=0
Reserves memory.
virtual void zero()
Zeroes all the memory associated with this AttributeStore.
BoolAttributeAccessor(Attribute< bool > &attribute, index_t index)
BoolAttributeAccessor constructor.
BoolAttributeAccessor & operator=(const BoolAttributeAccessor &rhs)
Copies a bool from another attribute.
BoolAttributeAccessor & operator=(const ConstBoolAttributeAccessor &rhs)
Copies a bool from another attribute.
BoolAttributeAccessor & operator=(bool x)
Assigns a bool to a BoolAttributeAccessor.
BoolAttributeAccessor(const BoolAttributeAccessor &rhs)
Copy-constructor.
ConstBoolAttributeAccessor(const Attribute< bool > &attribute, index_t index)
ConstBoolAttributeAccessor constructor.
Numeric::uint8 & element(index_t i)
Gets a modifiable element by index.
void fill(bool val)
Sets all the elements in this attribute to a specified value.
const Numeric::uint8 & element(index_t i) const
Gets an element by index.
Manages an attribute attached to a set of object.
Attribute(AttributesManager &manager, const std::string &name)
Creates or retrieves a persistent attribute attached to a given AttributesManager.
const T & operator[](index_t i) const
Gets an element by index.
Attribute()
Creates an uninitialized (unbound) Attribute.
T * data()
Gets the pointer to the data.
void fill(const T &val)
Sets all the elements in this attribute to a specified value.
const T * data() const
Gets the pointer to the data.
T & operator[](index_t i)
Gets a modifiable element by index.
void copy(const Attribute< T > &rhs)
Copies all the values from another attribute.
static void static_test_type()
Tests at compile time whether type can be used in an Attribute. If not the case generates a compile-t...
Managers a set of attributes attached to an object.
Definition attributes.h:832
void delete_attribute_store(AttributeStore *as)
Deletes an AttributeStore.
void compress(const vector< index_t > &old2new)
Compresses the stored attributes, by applying an index mapping that fills-in the gaps.
void zero_item(index_t to)
Sets an item to zero.
void copy(const AttributesManager &rhs)
Copies all the attributes from another AttributesManager.
AttributesManager()
Constructs a new empty AttributesManager.
index_t nb() const
Gets the number of attributes.
Definition attributes.h:850
bool rename_attribute(const std::string &old_name, const std::string &new_name)
Renames an attribute.
void scale_item(index_t to, double s)
Scales an item.
const AttributeStore * find_attribute_store(const std::string &name) const
Finds an AttributeStore by name.
void apply_permutation(const vector< index_t > &permutation)
Applies a permutation to the stored attributes.
AttributeStore * find_attribute_store(const std::string &name)
Finds an AttributeStore by name.
void bind_attribute_store(const std::string &name, AttributeStore *as)
Binds an AttributeStore with the specified name. Ownership of this AttributeStore is transferred to t...
void list_attribute_names(vector< std::string > &names) const
Gets the names of all the attributes in this AttributeStore.
~AttributesManager()
AttributesManager destructor.
void copy_item(index_t to, index_t from)
Copies all the attributes of an item into another one.
void madd_item(index_t to, double s, index_t from)
Adds a scaled item to another item \detais item[to] += s * item[from].
bool is_defined(const std::string &name) const
Tests whether an attribute is defined.
Definition attributes.h:947
void swap_items(index_t i, index_t j)
Swaps all the attributes of two items.
index_t size() const
Gets the size.
Definition attributes.h:867
void clear(bool keep_attributes, bool keep_memory=false)
Clears this AttributesManager.
index_t capacity() const
Gets the capacity.
Definition attributes.h:876
void delete_attribute_store(const std::string &name)
Deletes an AttributeStore by name.
bool copy_attribute(const std::string &name, const std::string &new_name)
Copies an attribute.
void zero()
Zeroes all the attributes.
void reserve(index_t new_capacity)
Pre-allocates memory for a number of items.
void resize(index_t new_size)
Resizes all the attributes managed by this AttributesManager.
Base class for reference-counted objects.
Definition counted.h:71
static void register_ascii_attribute_serializer(const std::string &type_name, AsciiAttributeSerializer read, AsciiAttributeSerializer write)
Declares a new attribute type that can be read from and written to ascii files.
Readonly access to an attribute as a double regardless its type.
double operator[](index_t i)
Gets a property value.
Readwrite access to an attribute as a double regardless its type.
Accessor class used by ScalarAttributeAdapter to implement indexing operator.
Accessor class used by ScalarAttributeAdapter to implement indexing operator (const version).
Access to an attribute as a double regardless its type.
index_t element_index() const
Gets the element index.
T get_element(index_t i, index_t multiplier=1) const
Gets an element.
ElementType
Internal representation of the attribute.
static std::string attribute_base_name(const std::string &name)
Gets the base attribute name from a compound name.
void bind_if_is_defined(const AttributesManager &manager, const std::string &name)
Binds this Attribute to an AttributesManager if it already exists in the AttributesManager.
bool is_bound() const
Tests whether an Attribute is bound.
static index_t nb_scalar_elements_per_item(const AttributeStore *store)
Gets the number of scalar components per item in an AttributeStore.
double get_element_as_double(index_t i) const
Gets an attribute value.
~ScalarAttributeAdapterBase()
ReadonlyScalarAttributeAdapterBase destructor.
static ElementType element_type(const AttributeStore *store)
Gets the element type stored in an AttributeStore.
const AttributeStore * attribute_store() const
Gets the AttributeStore.
ScalarAttributeAdapterBase()
ScalarAttributeAdapterBase constructor.
static bool can_be_bound_to(const AttributeStore *store)
Tests whether a ScalarAttributeAdapterBase can be bound to a given attribute store.
void unbind()
Unbinds this Attribute.
static bool is_defined(const AttributesManager &manager, const std::string &name)
Tests whether an attribute with the specified name and with a type that can be converted to double ex...
ScalarAttributeAdapterBase(const AttributesManager &manager, const std::string &name)
ScalarAttributeAdapterBase constructor.
double set_element_as_double(index_t i, double value)
Sets an attribute value.
void set_element(T value, index_t i, index_t multiplier=1) const
Sets an element.
static index_t attribute_element_index(const std::string &name)
Gets the base attribute name from a compound name.
index_t size() const
Gets the size.
ElementType element_type() const
Gets the internal representation of the elements.
A smart pointer with reference-counted copy semantics.
Implementation of AttributeStoreCreator for a specific type.
Definition attributes.h:779
AttributeStore * create_attribute_store(index_t dim) override
Creates a new attribute store.
Definition attributes.h:784
Stores an array of elements of a given type, and notifies a set of AttributeStoreObservers each time ...
Definition attributes.h:607
void scale_item(index_t to, double s) override
Scales an item.
Definition attributes.h:691
TypedAttributeStore(index_t dim=1)
Creates a new empty attribute store.
Definition attributes.h:616
void notify(Memory::pointer base_addr, index_t size, index_t dim) override
If size or base address differ from the cached values, notify all the observers, and update the cache...
Definition attributes.h:709
void clear(bool keep_memory=false) override
Resizes this AttributeStore to 0.
Definition attributes.h:641
AttributeStore * clone() const override
Creates a new AttributeStore that is a carbon copy of this AttributeStore.
Definition attributes.h:679
void resize(index_t new_size) override
Resizes this AttributeStore.
Definition attributes.h:620
std::string element_typeid_name() const override
Gets the typeid name of the element type stored in this AttributeStore.
Definition attributes.h:675
void reserve(index_t new_capacity) override
Reserves memory.
Definition attributes.h:629
bool elements_type_matches(const std::string &type_name) const override
Tests whether this AttributeStore stores elements of a given type.
Definition attributes.h:671
void madd_item(index_t to, double s, index_t from) override
Adds a scaled item to another item \detais item[to] += s * item[from].
Definition attributes.h:698
void redim(index_t dim) override
Sets the dimension.
Definition attributes.h:651
Helper class to register new attribute types.
Definition attributes.h:795
geo_register_attribute_type(const std::string &type_name)
geo_register_attribute_type constructor
Definition attributes.h:806
Vector with aligned memory allocation.
Definition memory.h:660
Common include file, providing basic definitions. Should be included before anything else by all head...
Functions to read and write structured files.
Types and functions for memory manipulation.
byte * pointer
Pointer to unsigned byte(s)
Definition memory.h:104
float float32
Definition numeric.h:147
uint8_t uint8
Definition numeric.h:135
int8_t int8
Definition numeric.h:123
double float64
Definition numeric.h:150
int32_t int32
Definition numeric.h:129
uint32_t uint32
Definition numeric.h:141
std::atomic_flag spinlock
A lightweight synchronization structure.
Global Vorpaline namespace.
bool write_ascii_attribute< bool >(FILE *file, Memory::pointer base_addr, index_t nb_elements)
Writes an ASCII attribute to a file.
Definition geofile.h:228
void geo_argused(const T &)
Suppresses compiler warnings about unused parameters.
Definition argused.h:60
geo_index_t index_t
The type for storing and manipulating indices.
Definition numeric.h:329
SmartPointer< AttributeStoreCreator > AttributeStoreCreator_var
An automatic reference-counted pointer to an AttributeStoreCreator.
Definition attributes.h:188
bool read_ascii_attribute< bool >(FILE *file, Memory::pointer base_addr, index_t nb_elements)
Reads an ASCII attribute from a file.
Definition geofile.h:204
Types and functions for numbers manipulation.
Function and classes for process manipulation.