From bd9234db175089dac931822369ee12a7ba1ff613 Mon Sep 17 00:00:00 2001 From: hschreiber Date: Thu, 28 Sep 2023 17:08:05 +0200 Subject: [PATCH] removal of child pointer + factorization of intrusive list column --- .../gudhi/Persistence_matrix/base_matrix.h | 31 +- .../base_matrix_row_access.h | 5 + .../base_matrix_with_column_compression.h | 25 +- .../gudhi/Persistence_matrix/base_pairing.h | 50 +- .../gudhi/Persistence_matrix/base_swap.h | 75 +- .../Persistence_matrix/boundary_matrix.h | 58 +- .../gudhi/Persistence_matrix/chain_matrix.h | 148 +-- .../gudhi/Persistence_matrix/chain_pairing.h | 6 + .../Persistence_matrix/chain_rep_cycles.h | 41 +- .../Persistence_matrix/chain_vine_swap.h | 195 ++-- .../Persistence_matrix/columns/cell_types.h | 50 +- .../columns/chain_column_extra_properties.h | 72 ++ .../columns/column_dimension_holder.h | 58 ++ .../columns/intrusive_list_column.h | 922 ++++++++++++++++++ .../Persistence_matrix/columns/row_access.h | 9 + .../matrix_dimension_holders.h | 5 + .../gudhi/Persistence_matrix/ru_matrix.h | 60 +- .../gudhi/Persistence_matrix/ru_pairing.h | 6 + .../gudhi/Persistence_matrix/ru_rep_cycles.h | 43 +- .../gudhi/Persistence_matrix/ru_vine_swap.h | 140 +-- src/Persistence_matrix/include/gudhi/matrix.h | 176 +--- .../include/gudhi/options.h | 2 +- ...ce_matrix_column_compression_unit_test.cpp | 692 ++++++------- .../test/Persistence_matrix_columns_test.cpp | 453 ++++----- .../Persistence_matrix_common_unit_test.cpp | 432 ++++---- ...rsistence_matrix_specialized_unit_test.cpp | 168 ++-- 26 files changed, 2438 insertions(+), 1484 deletions(-) create mode 100644 src/Persistence_matrix/include/gudhi/Persistence_matrix/columns/chain_column_extra_properties.h create mode 100644 src/Persistence_matrix/include/gudhi/Persistence_matrix/columns/column_dimension_holder.h create mode 100644 src/Persistence_matrix/include/gudhi/Persistence_matrix/columns/intrusive_list_column.h diff --git a/src/Persistence_matrix/include/gudhi/Persistence_matrix/base_matrix.h b/src/Persistence_matrix/include/gudhi/Persistence_matrix/base_matrix.h index e72d4d0ea1..c060608477 100644 --- a/src/Persistence_matrix/include/gudhi/Persistence_matrix/base_matrix.h +++ b/src/Persistence_matrix/include/gudhi/Persistence_matrix/base_matrix.h @@ -20,7 +20,8 @@ namespace persistence_matrix { template class Base_matrix - : public Master_matrix::Base_swap_option, protected Master_matrix::Matrix_row_access_option + : public Master_matrix::template Base_swap_option >, + protected Master_matrix::Matrix_row_access_option { public: using index = typename Master_matrix::index; @@ -73,8 +74,8 @@ class Base_matrix Base_matrix& operator=(const Base_matrix& other); friend void swap(Base_matrix& matrix1, Base_matrix& matrix2){ - swap(static_cast(matrix1), - static_cast(matrix2)); + swap(static_cast >&>(matrix1), + static_cast >&>(matrix2)); matrix1.matrix_.swap(matrix2.matrix_); std::swap(matrix1.nextInsertIndex_, matrix2.nextInsertIndex_); @@ -102,7 +103,7 @@ class Base_matrix void print(); //for debug private: - using swap_opt = typename Master_matrix::Base_swap_option; + using swap_opt = typename Master_matrix::template Base_swap_option >; using ra_opt = typename Master_matrix::Matrix_row_access_option; using matrix_type = typename Master_matrix::column_container_type; using cell_rep_type = typename std::conditional< @@ -111,6 +112,8 @@ class Base_matrix std::pair >::type; + friend swap_opt; //direct access to matrix_ to avoid row reorder. + matrix_type matrix_; index nextInsertIndex_; @@ -119,7 +122,7 @@ class Base_matrix template inline Base_matrix::Base_matrix() - : swap_opt(matrix_), + : swap_opt(), ra_opt(), nextInsertIndex_(0) {} @@ -127,7 +130,7 @@ inline Base_matrix::Base_matrix() template template inline Base_matrix::Base_matrix(const std::vector &columns) - : swap_opt(matrix_, columns.size()), + : swap_opt(columns.size()), ra_opt(columns.size()), //not ideal if max row index is much smaller than max column index, does that happen often? matrix_(!Master_matrix::Option_list::has_removable_columns && Master_matrix::Option_list::has_row_access ? 0 : columns.size()), nextInsertIndex_(columns.size()) @@ -154,7 +157,7 @@ inline Base_matrix::Base_matrix(const std::vector template inline Base_matrix::Base_matrix(unsigned int numberOfColumns) - : swap_opt(matrix_, numberOfColumns), + : swap_opt(numberOfColumns), ra_opt(numberOfColumns), matrix_(!Master_matrix::Option_list::has_removable_columns && Master_matrix::Option_list::has_row_access ? 0 : numberOfColumns), nextInsertIndex_(0) @@ -165,13 +168,10 @@ inline Base_matrix::Base_matrix(unsigned int numberOfColumns) template inline Base_matrix::Base_matrix(const Base_matrix &matrixToCopy) - : swap_opt(matrixToCopy), - ra_opt(matrixToCopy), + : swap_opt(static_cast(matrixToCopy)), + ra_opt(static_cast(matrixToCopy)), nextInsertIndex_(matrixToCopy.nextInsertIndex_) { - if constexpr (Master_matrix::Option_list::has_column_and_row_swaps) - swap_opt::matrix_ = &matrix_; - if constexpr (Master_matrix::Option_list::has_row_access){ matrix_.reserve(matrixToCopy.matrix_.size()); if constexpr (Master_matrix::Option_list::has_removable_columns){ @@ -191,14 +191,11 @@ inline Base_matrix::Base_matrix(const Base_matrix &matrixToCopy) template inline Base_matrix::Base_matrix(Base_matrix &&other) noexcept - : swap_opt(std::move(other)), - ra_opt(std::move(other)), + : swap_opt(std::move(static_cast(other))), + ra_opt(std::move(static_cast(other))), matrix_(std::move(other.matrix_)), nextInsertIndex_(std::exchange(other.nextInsertIndex_, 0)) { - if constexpr (Master_matrix::Option_list::has_column_and_row_swaps) - swap_opt::matrix_ = &matrix_; - if constexpr (Master_matrix::Option_list::has_row_access){ if constexpr (Master_matrix::Option_list::has_removable_columns){ for (auto& p : matrix_){ diff --git a/src/Persistence_matrix/include/gudhi/Persistence_matrix/base_matrix_row_access.h b/src/Persistence_matrix/include/gudhi/Persistence_matrix/base_matrix_row_access.h index 6122bde4e1..e6b55eed4d 100644 --- a/src/Persistence_matrix/include/gudhi/Persistence_matrix/base_matrix_row_access.h +++ b/src/Persistence_matrix/include/gudhi/Persistence_matrix/base_matrix_row_access.h @@ -16,6 +16,11 @@ namespace Gudhi { namespace persistence_matrix { +struct Dummy_matrix_row_access{ + Dummy_matrix_row_access(){}; + Dummy_matrix_row_access([[maybe_unused]] unsigned int numberOfColumns){}; +}; + template class Base_matrix_row_access { diff --git a/src/Persistence_matrix/include/gudhi/Persistence_matrix/base_matrix_with_column_compression.h b/src/Persistence_matrix/include/gudhi/Persistence_matrix/base_matrix_with_column_compression.h index 8d8e9e1670..562cf6e9e8 100644 --- a/src/Persistence_matrix/include/gudhi/Persistence_matrix/base_matrix_with_column_compression.h +++ b/src/Persistence_matrix/include/gudhi/Persistence_matrix/base_matrix_with_column_compression.h @@ -48,18 +48,14 @@ class Base_matrix_with_column_compression : protected Master_matrix::Matrix_row_ Column_type(const Container_type& nonZeroRowIndices) : Base(nonZeroRowIndices) {} - template - Column_type(const Container_type& nonZeroRowIndices, dimension_type dimension) - : Base(nonZeroRowIndices, dimension) - {} - template - Column_type(index columnIndex, Row_container_type &rowContainer) - : Base(columnIndex, rowContainer) - {} template Column_type(index columnIndex, const Container_type& nonZeroRowIndices, Row_container_type &rowContainer) : Base(columnIndex, nonZeroRowIndices, rowContainer) {} + template + Column_type(const Container_type& nonZeroRowIndices, dimension_type dimension) + : Base(nonZeroRowIndices, dimension) + {} template Column_type(index columnIndex, const Container_type& nonZeroRowIndices, dimension_type dimension, Row_container_type &rowContainer) : Base(columnIndex, nonZeroRowIndices, dimension, rowContainer) @@ -67,15 +63,12 @@ class Base_matrix_with_column_compression : protected Master_matrix::Matrix_row_ Column_type(const Column_type& column) : Base(static_cast(column)) {} - Column_type(const Column_type& column, index columnIndex) - : Base(static_cast(column), columnIndex) - {} template Column_type(const Column_type& column, index columnIndex, Row_container_type &rowContainer) : Base(static_cast(column), columnIndex, rowContainer) {} Column_type(Column_type&& column) noexcept - : Base(std::move(static_cast(column))) + : Base(std::move(static_cast(column))) {} index get_rep() const{ @@ -88,7 +81,7 @@ class Base_matrix_with_column_compression : protected Master_matrix::Matrix_row_ struct Hasher { size_t operator()(const Column_type& c) const { - return std::hash()(c); + return std::hash()(static_cast(c)); } }; @@ -216,7 +209,7 @@ inline Base_matrix_with_column_compression::Base_matrix_with_colu template inline Base_matrix_with_column_compression::Base_matrix_with_column_compression(const Base_matrix_with_column_compression &matrixToCopy) - : ra_opt(matrixToCopy), + : ra_opt(static_cast(matrixToCopy)), columnClasses_(matrixToCopy.columnClasses_), repToColumn_(matrixToCopy.repToColumn_.size(), nullptr), nextColumnIndex_(0) @@ -237,7 +230,7 @@ inline Base_matrix_with_column_compression::Base_matrix_with_colu template inline Base_matrix_with_column_compression::Base_matrix_with_column_compression(Base_matrix_with_column_compression &&other) noexcept - : ra_opt(std::move(other)), + : ra_opt(std::move(static_cast(other))), columnToRep_(std::move(other.columnToRep_)), columnClasses_(std::move(other.columnClasses_)), repToColumn_(std::move(other.repToColumn_)), @@ -512,7 +505,7 @@ inline void Base_matrix_with_column_compression::_insert_column(i return; } - repToColumn_[columnIndex]->set_rep(columnIndex); + col.set_rep(columnIndex); auto res = columnToRep_.insert(col); if (res.first->get_rep() != columnIndex){ _insert_double_column(columnIndex, res.first); diff --git a/src/Persistence_matrix/include/gudhi/Persistence_matrix/base_pairing.h b/src/Persistence_matrix/include/gudhi/Persistence_matrix/base_pairing.h index e60a86e3b8..c67b9e4196 100644 --- a/src/Persistence_matrix/include/gudhi/Persistence_matrix/base_pairing.h +++ b/src/Persistence_matrix/include/gudhi/Persistence_matrix/base_pairing.h @@ -17,6 +17,15 @@ namespace Gudhi { namespace persistence_matrix { +struct Dummy_base_pairing{ + Dummy_base_pairing& operator=([[maybe_unused]] Dummy_base_pairing other){return *this;} + friend void swap([[maybe_unused]] Dummy_base_pairing& d1, [[maybe_unused]] Dummy_base_pairing& d2){} + + Dummy_base_pairing(){} + Dummy_base_pairing([[maybe_unused]] const Dummy_base_pairing& matrixToCopy){} + Dummy_base_pairing([[maybe_unused]] Dummy_base_pairing&& other) noexcept{} +}; + template class Base_pairing { @@ -27,7 +36,7 @@ class Base_pairing using dimension_type = typename Master_matrix::dimension_type; using Bar = typename Master_matrix::Bar; - Base_pairing(matrix_type& matrix, dimension_type& maxDim); + Base_pairing(); Base_pairing(const Base_pairing& matrixToCopy); Base_pairing(Base_pairing&& other) noexcept; @@ -43,36 +52,34 @@ class Base_pairing protected: using column_type = typename Master_matrix::Column_type; using dictionnary_type = typename Master_matrix::bar_dictionnary_type; + using base_matrix = typename Master_matrix::Boundary_matrix_type; - matrix_type* matrix_; - dimension_type* maxDim_; barcode_type barcode_; dictionnary_type indexToBar_; bool isReduced_; void _reduce(); void _remove_maximal(index columnIndex); + + constexpr base_matrix* _matrix() { return static_cast(this); } + constexpr const base_matrix* _matrix() const { return static_cast(this); } }; template -inline Base_pairing::Base_pairing(matrix_type &matrix, dimension_type &maxDim) - : matrix_(&matrix), maxDim_(&maxDim), isReduced_(false) +inline Base_pairing::Base_pairing() + : isReduced_(false) {} template inline Base_pairing::Base_pairing(const Base_pairing &matrixToCopy) - : matrix_(matrixToCopy.matrix_), - maxDim_(matrixToCopy.maxDim_), - barcode_(matrixToCopy.barcode_), + : barcode_(matrixToCopy.barcode_), indexToBar_(matrixToCopy.indexToBar_), isReduced_(matrixToCopy.isReduced_) {} template inline Base_pairing::Base_pairing(Base_pairing &&other) noexcept - : matrix_(other.matrix_), - maxDim_(other.maxDim_), - barcode_(std::move(other.barcode_)), + : barcode_(std::move(other.barcode_)), indexToBar_(std::move(other.indexToBar_)), isReduced_(std::move(other.isReduced_)) {} @@ -89,20 +96,19 @@ template inline void Base_pairing::_reduce() { std::unordered_map pivotsToColumn; - auto& matrix = *matrix_; - for (int d = *maxDim_; d > 0; d--){ - for (unsigned int i = 0; i < matrix_->size(); i++){ - if (!(matrix.at(i).is_empty()) && matrix.at(i).get_dimension() == d) + for (int d = _matrix()->get_max_dimension(); d > 0; d--){ + for (unsigned int i = 0; i < _matrix()->get_number_of_columns(); i++){ + auto& curr = _matrix()->get_column(i); + if (!(curr.is_empty()) && curr.get_dimension() == d) { - column_type &curr = matrix.at(i); int pivot = curr.get_pivot(); while (pivot != -1 && pivotsToColumn.find(pivot) != pivotsToColumn.end()){ if constexpr (Master_matrix::Option_list::is_z2){ - curr += matrix.at(pivotsToColumn.at(pivot)); + curr += _matrix()->get_column(pivotsToColumn.at(pivot)); } else { - column_type &toadd = matrix.at(pivotsToColumn.at(pivot)); + column_type &toadd = _matrix()->get_column(pivotsToColumn.at(pivot)); typename Master_matrix::Field_type coef = curr.get_pivot_value(); coef = coef.get_inverse(); coef *= (Master_matrix::Field_type::get_characteristic() - static_cast(toadd.get_pivot_value())); @@ -114,17 +120,17 @@ inline void Base_pairing::_reduce() if (pivot != -1){ pivotsToColumn.emplace(pivot, i); - matrix.at(pivot).clear(); + _matrix()->get_column(pivot).clear(); barcode_.push_back(Bar(d - 1, pivot, i)); } else { - matrix.at(i).clear(); + curr.clear(); barcode_.push_back(Bar(d, i, -1)); } } } } - for (unsigned int i = 0; i < matrix_->size(); i++){ - if (matrix.at(i).get_dimension() == 0 && pivotsToColumn.find(i) == pivotsToColumn.end()){ + for (unsigned int i = 0; i < _matrix()->get_number_of_columns(); i++){ + if (_matrix()->get_column(i).get_dimension() == 0 && pivotsToColumn.find(i) == pivotsToColumn.end()){ barcode_.push_back(Bar(0, i, -1)); } } diff --git a/src/Persistence_matrix/include/gudhi/Persistence_matrix/base_swap.h b/src/Persistence_matrix/include/gudhi/Persistence_matrix/base_swap.h index ccb0523836..fa3e811009 100644 --- a/src/Persistence_matrix/include/gudhi/Persistence_matrix/base_swap.h +++ b/src/Persistence_matrix/include/gudhi/Persistence_matrix/base_swap.h @@ -17,15 +17,25 @@ namespace Gudhi { namespace persistence_matrix { -template +struct Dummy_base_swap{ + Dummy_base_swap& operator=([[maybe_unused]] Dummy_base_swap other){return *this;} + friend void swap([[maybe_unused]] Dummy_base_swap& d1, [[maybe_unused]] Dummy_base_swap& d2){} + + Dummy_base_swap(){} + Dummy_base_swap([[maybe_unused]] unsigned int numberOfColumns){} + Dummy_base_swap([[maybe_unused]] const Dummy_base_swap& matrixToCopy){} + Dummy_base_swap([[maybe_unused]] Dummy_base_swap&& other) noexcept{} +}; + +template class Base_swap { public: using matrix_type = typename Master_matrix::column_container_type; using index = typename Master_matrix::index; - Base_swap(matrix_type &matrix); - Base_swap(matrix_type &matrix, unsigned int numberOfColumns); + Base_swap(); + Base_swap(unsigned int numberOfColumns); Base_swap(const Base_swap& matrixToCopy); Base_swap(Base_swap&& other) noexcept; @@ -47,22 +57,23 @@ class Base_swap index_dictionnary_type indexToRow_; row_dictionnary_type rowToIndex_; bool rowSwapped_; - matrix_type *matrix_; void _orderRows(); + + constexpr Base_matrix* _matrix() { return static_cast(this); } + constexpr const Base_matrix* _matrix() const { return static_cast(this); } }; -template -inline Base_swap::Base_swap(matrix_type &matrix) - : rowSwapped_(false), matrix_(&matrix) +template +inline Base_swap::Base_swap() + : rowSwapped_(false) {} -template -inline Base_swap::Base_swap(matrix_type &matrix, unsigned int numberOfColumns) +template +inline Base_swap::Base_swap(unsigned int numberOfColumns) : indexToRow_(numberOfColumns), rowToIndex_(numberOfColumns), - rowSwapped_(false), - matrix_(&matrix) + rowSwapped_(false) { for (unsigned int i = 0; i < numberOfColumns; i++){ indexToRow_[i] = i; @@ -70,45 +81,43 @@ inline Base_swap::Base_swap(matrix_type &matrix, unsigned int num } } -template -inline Base_swap::Base_swap(const Base_swap& matrixToCopy) +template +inline Base_swap::Base_swap(const Base_swap& matrixToCopy) : indexToRow_(matrixToCopy.indexToRow_), rowToIndex_(matrixToCopy.rowToIndex_), - rowSwapped_(matrixToCopy.rowSwapped_), - matrix_(matrixToCopy.matrix_) + rowSwapped_(matrixToCopy.rowSwapped_) {} -template -inline Base_swap::Base_swap(Base_swap &&other) noexcept +template +inline Base_swap::Base_swap(Base_swap &&other) noexcept : indexToRow_(std::move(other.indexToRow_)), rowToIndex_(std::move(other.rowToIndex_)), - rowSwapped_(std::exchange(other.rowSwapped_, 0)), - matrix_(other.matrix_) + rowSwapped_(std::exchange(other.rowSwapped_, 0)) {} -template -inline void Base_swap::swap_columns(index columnIndex1, index columnIndex2) +template +inline void Base_swap::swap_columns(index columnIndex1, index columnIndex2) { - swap(matrix_->at(columnIndex1), matrix_->at(columnIndex2)); + swap(_matrix()->matrix_.at(columnIndex1), _matrix()->matrix_.at(columnIndex2)); } -template -inline void Base_swap::swap_rows(index rowIndex1, index rowIndex2) +template +inline void Base_swap::swap_rows(index rowIndex1, index rowIndex2) { rowSwapped_ = true; std::swap(rowToIndex_[indexToRow_[rowIndex1]], rowToIndex_[indexToRow_[rowIndex2]]); std::swap(indexToRow_[rowIndex1], indexToRow_[rowIndex2]); } -template -inline void Base_swap::swap_at_indices(index index1, index index2) +template +inline void Base_swap::swap_at_indices(index index1, index index2) { swap_columns(index1, index2); swap_rows(index1, index2); } -template -inline Base_swap &Base_swap::operator=(Base_swap other) +template +inline Base_swap &Base_swap::operator=(Base_swap other) { indexToRow_.swap(other.indexToRow_); rowToIndex_.swap(other.rowToIndex_); @@ -116,13 +125,13 @@ inline Base_swap &Base_swap::operator=(Base_swap -inline void Base_swap::_orderRows() +template +inline void Base_swap::_orderRows() { - for (unsigned int i = 0; i < matrix_->size(); i++){ - matrix_->at(i).reorder(rowToIndex_); + for (unsigned int i = 0; i < _matrix()->get_number_of_columns(); i++){ + _matrix()->matrix_.at(i).reorder(rowToIndex_); } - for (unsigned int i = 0; i < matrix_->size(); i++){ + for (unsigned int i = 0; i < _matrix()->get_number_of_columns(); i++){ indexToRow_[i] = i; rowToIndex_[i] = i; } diff --git a/src/Persistence_matrix/include/gudhi/Persistence_matrix/boundary_matrix.h b/src/Persistence_matrix/include/gudhi/Persistence_matrix/boundary_matrix.h index c47c1f24a5..493e54223a 100644 --- a/src/Persistence_matrix/include/gudhi/Persistence_matrix/boundary_matrix.h +++ b/src/Persistence_matrix/include/gudhi/Persistence_matrix/boundary_matrix.h @@ -21,7 +21,7 @@ namespace persistence_matrix { template class Boundary_matrix //TODO: factorize/inheritate/compose with base matrix? : public Master_matrix::Matrix_dimension_option, - public Master_matrix::Base_swap_option, + public Master_matrix::template Base_swap_option >, public Master_matrix::Base_pairing_option, protected Master_matrix::Matrix_row_access_option { @@ -75,8 +75,8 @@ class Boundary_matrix //TODO: factorize/inheritate/compose with base matrix? friend void swap(Boundary_matrix& matrix1, Boundary_matrix& matrix2){ swap(static_cast(matrix1), static_cast(matrix2)); - swap(static_cast(matrix1), - static_cast(matrix2)); + swap(static_cast >&>(matrix1), + static_cast >&>(matrix2)); swap(static_cast(matrix1), static_cast(matrix2)); matrix1.matrix_.swap(matrix2.matrix_); @@ -107,15 +107,17 @@ class Boundary_matrix //TODO: factorize/inheritate/compose with base matrix? private: using dim_opt = typename Master_matrix::Matrix_dimension_option; - using swap_opt = typename Master_matrix::Base_swap_option; + using swap_opt = typename Master_matrix::template Base_swap_option >; using pair_opt = typename Master_matrix::Base_pairing_option; using ra_opt = typename Master_matrix::Matrix_row_access_option; using matrix_type = typename Master_matrix::column_container_type; + friend swap_opt; + matrix_type matrix_; index nextInsertIndex_; - static const bool activeDimOption = Master_matrix::Option_list::has_dimension_access || Master_matrix::dimensionIsNeeded; + static const bool activeDimOption = Master_matrix::Option_list::has_matrix_maximal_dimension_access || Master_matrix::dimensionIsNeeded; static const bool activeSwapOption = Master_matrix::Option_list::has_column_and_row_swaps || Master_matrix::Option_list::has_vine_update; static const bool activePairingOption = Master_matrix::Option_list::has_column_pairings && !Master_matrix::Option_list::has_vine_update && !Master_matrix::Option_list::can_retrieve_representative_cycles; }; @@ -123,8 +125,8 @@ class Boundary_matrix //TODO: factorize/inheritate/compose with base matrix? template inline Boundary_matrix::Boundary_matrix() : dim_opt(-1), - swap_opt(matrix_), - pair_opt(matrix_, dim_opt::maxDim_), + swap_opt(), + pair_opt(), ra_opt(), nextInsertIndex_(0) {} @@ -133,8 +135,8 @@ template template inline Boundary_matrix::Boundary_matrix(const std::vector &orderedBoundaries) : dim_opt(-1), - swap_opt(matrix_, orderedBoundaries.size()), - pair_opt(matrix_, dim_opt::maxDim_), + swap_opt(orderedBoundaries.size()), + pair_opt(), ra_opt(orderedBoundaries.size()), nextInsertIndex_(orderedBoundaries.size()) { @@ -166,8 +168,8 @@ inline Boundary_matrix::Boundary_matrix(const std::vector inline Boundary_matrix::Boundary_matrix(unsigned int numberOfColumns) : dim_opt(-1), - swap_opt(matrix_, numberOfColumns), - pair_opt(matrix_, dim_opt::maxDim_), + swap_opt(numberOfColumns), + pair_opt(), ra_opt(numberOfColumns), matrix_(!Master_matrix::Option_list::has_removable_columns && Master_matrix::Option_list::has_row_access ? 0 : numberOfColumns), nextInsertIndex_(0) @@ -178,19 +180,12 @@ inline Boundary_matrix::Boundary_matrix(unsigned int numberOfColu template inline Boundary_matrix::Boundary_matrix(const Boundary_matrix &matrixToCopy) - : dim_opt(matrixToCopy), - swap_opt(matrixToCopy), - pair_opt(matrixToCopy), - ra_opt(matrixToCopy), + : dim_opt(static_cast(matrixToCopy)), + swap_opt(static_cast(matrixToCopy)), + pair_opt(static_cast(matrixToCopy)), + ra_opt(static_cast(matrixToCopy)), nextInsertIndex_(matrixToCopy.nextInsertIndex_) { - if constexpr (activeSwapOption) - swap_opt::matrix_ = &matrix_; - if constexpr (activePairingOption){ - pair_opt::matrix_ = &matrix_; - pair_opt::maxDim_ = &this->dim_opt::maxDim_; - } - if constexpr (Master_matrix::Option_list::has_row_access){ matrix_.reserve(matrixToCopy.matrix_.size()); if constexpr (Master_matrix::Option_list::has_removable_columns){ @@ -210,20 +205,13 @@ inline Boundary_matrix::Boundary_matrix(const Boundary_matrix &ma template inline Boundary_matrix::Boundary_matrix(Boundary_matrix &&other) noexcept - : dim_opt(std::move(other)), - swap_opt(std::move(other)), - pair_opt(std::move(other)), - ra_opt(std::move(other)), + : dim_opt(std::move(static_cast(other))), + swap_opt(std::move(static_cast(other))), + pair_opt(std::move(static_cast(other))), + ra_opt(std::move(static_cast(other))), matrix_(std::move(other.matrix_)), nextInsertIndex_(std::exchange(other.nextInsertIndex_, 0)) { - if constexpr (activeSwapOption) - swap_opt::matrix_ = &matrix_; - if constexpr (activePairingOption){ - pair_opt::matrix_ = &matrix_; - pair_opt::maxDim_ = &this->dim_opt::maxDim_; - } - if constexpr (Master_matrix::Option_list::has_row_access){ if constexpr (Master_matrix::Option_list::has_removable_columns){ for (auto& p : matrix_){ @@ -522,6 +510,10 @@ inline bool Boundary_matrix::is_zero_column(index columnIndex) template inline typename Boundary_matrix::index Boundary_matrix::get_pivot(index columnIndex) { + if constexpr (activeSwapOption){ + if (swap_opt::rowSwapped_) swap_opt::_orderRows(); + } + if constexpr (Master_matrix::Option_list::has_removable_columns){ return matrix_.at(columnIndex).get_pivot(); } else { diff --git a/src/Persistence_matrix/include/gudhi/Persistence_matrix/chain_matrix.h b/src/Persistence_matrix/include/gudhi/Persistence_matrix/chain_matrix.h index e76874c26e..76917cbf4b 100644 --- a/src/Persistence_matrix/include/gudhi/Persistence_matrix/chain_matrix.h +++ b/src/Persistence_matrix/include/gudhi/Persistence_matrix/chain_matrix.h @@ -160,7 +160,9 @@ class Chain_matrix void _update_largest_death_in_F(const std::vector& chainsInF); void _insert_chain(const tmp_column_type& column, dimension_type dimension); void _insert_chain(const tmp_column_type& column, dimension_type dimension, index pair); - void _add_to(const Column_type& column, tmp_column_type& set, unsigned int coef = 1u); + void _add_to(const Column_type& column, tmp_column_type& set, unsigned int coef); + template + void _add_to(Column_type& target, F&& addition); constexpr barcode_type& _barcode(); constexpr bar_dictionnary_type& _indexToBar(); @@ -170,8 +172,8 @@ template inline Chain_matrix::Chain_matrix() : dim_opt(-1), pair_opt(), - swap_opt(matrix_), - rep_opt(matrix_, pivotToColumnIndex_), + swap_opt(), + rep_opt(), ra_opt(), nextInsertIndex_(0) {} @@ -181,8 +183,8 @@ template inline Chain_matrix::Chain_matrix(const std::vector &orderedBoundaries) : dim_opt(-1), pair_opt(), - swap_opt(matrix_), - rep_opt(matrix_, pivotToColumnIndex_), + swap_opt(), + rep_opt(), ra_opt(orderedBoundaries.size()), nextInsertIndex_(0) { @@ -202,8 +204,8 @@ template inline Chain_matrix::Chain_matrix(unsigned int numberOfColumns) : dim_opt(-1), pair_opt(), - swap_opt(matrix_), - rep_opt(matrix_, pivotToColumnIndex_), + swap_opt(), + rep_opt(), ra_opt(numberOfColumns), nextInsertIndex_(0) { @@ -222,8 +224,8 @@ inline Chain_matrix::Chain_matrix( DeathComparatorFunction&& deathComparator) : dim_opt(-1), pair_opt(), - swap_opt(matrix_, birthComparator, deathComparator), - rep_opt(matrix_, pivotToColumnIndex_), + swap_opt(birthComparator, deathComparator), + rep_opt(), ra_opt(), nextInsertIndex_(0) {} @@ -236,8 +238,8 @@ inline Chain_matrix::Chain_matrix( DeathComparatorFunction&& deathComparator) : dim_opt(-1), pair_opt(), - swap_opt(matrix_, birthComparator, deathComparator), - rep_opt(matrix_, pivotToColumnIndex_), + swap_opt(birthComparator, deathComparator), + rep_opt(), ra_opt(orderedBoundaries.size()), nextInsertIndex_(0) { @@ -260,8 +262,8 @@ inline Chain_matrix::Chain_matrix( DeathComparatorFunction&& deathComparator) : dim_opt(-1), pair_opt(), - swap_opt(matrix_, birthComparator, deathComparator), - rep_opt(matrix_, pivotToColumnIndex_), + swap_opt(birthComparator, deathComparator), + rep_opt(), ra_opt(numberOfColumns), nextInsertIndex_(0) { @@ -275,43 +277,35 @@ inline Chain_matrix::Chain_matrix( template inline Chain_matrix::Chain_matrix(const Chain_matrix &matrixToCopy) - : dim_opt(matrixToCopy), - pair_opt(matrixToCopy), - swap_opt(matrixToCopy), - rep_opt(matrixToCopy), - ra_opt(matrixToCopy), + : dim_opt(static_cast(matrixToCopy)), + pair_opt(static_cast(matrixToCopy)), + swap_opt(static_cast(matrixToCopy)), + rep_opt(static_cast(matrixToCopy)), + ra_opt(static_cast(matrixToCopy)), pivotToColumnIndex_(matrixToCopy.pivotToColumnIndex_), nextInsertIndex_(matrixToCopy.nextInsertIndex_) { - if constexpr (Master_matrix::Option_list::can_retrieve_representative_cycles){ - rep_opt::matrix_ = &matrix_; - rep_opt::pivotToPosition_ = &pivotToColumnIndex_; - } - if constexpr (Master_matrix::Option_list::has_vine_update){ - swap_opt::matrix_ = &matrix_; - } - matrix_.reserve(matrixToCopy.matrix_.size()); if constexpr (Master_matrix::Option_list::has_row_access){ if constexpr (Master_matrix::Option_list::has_removable_columns){ for (const auto& p : matrixToCopy.matrix_){ const Column_type& col = p.second; - matrix_.try_emplace(p.first, Column_type(col, col.get_column_index(), ra_opt::rows_, pivotToColumnIndex_)); + matrix_.try_emplace(p.first, Column_type(col, col.get_column_index(), ra_opt::rows_)); } } else { for (const auto& col : matrixToCopy.matrix_){ - matrix_.emplace_back(col, col.get_column_index(), ra_opt::rows_, pivotToColumnIndex_); + matrix_.emplace_back(col, col.get_column_index(), ra_opt::rows_); } } } else { if constexpr (Master_matrix::Option_list::has_removable_columns){ for (const auto& p : matrixToCopy.matrix_){ const Column_type& col = p.second; - matrix_.try_emplace(p.first, Column_type(col, pivotToColumnIndex_)); + matrix_.try_emplace(p.first, Column_type(col)); } } else { for (const auto& col : matrixToCopy.matrix_){ - matrix_.emplace_back(col, pivotToColumnIndex_); + matrix_.emplace_back(col); } } } @@ -320,36 +314,26 @@ inline Chain_matrix::Chain_matrix(const Chain_matrix &matrixToCop template inline Chain_matrix::Chain_matrix( Chain_matrix &&other) noexcept - : dim_opt(std::move(other)), - pair_opt(std::move(other)), - swap_opt(std::move(other)), - rep_opt(std::move(other)), - ra_opt(std::move(other)), + : dim_opt(std::move(static_cast(other))), + pair_opt(std::move(static_cast(other))), + swap_opt(std::move(static_cast(other))), + rep_opt(std::move(static_cast(other))), + ra_opt(std::move(static_cast(other))), matrix_(std::move(other.matrix_)), pivotToColumnIndex_(std::move(other.pivotToColumnIndex_)), nextInsertIndex_(std::exchange(other.nextInsertIndex_, 0)) { - if constexpr (Master_matrix::Option_list::can_retrieve_representative_cycles){ - rep_opt::matrix_ = &matrix_; - rep_opt::pivotToPosition_ = &pivotToColumnIndex_; - } - if constexpr (Master_matrix::Option_list::has_vine_update){ - swap_opt::matrix_ = &matrix_; - } - if constexpr (Master_matrix::Option_list::has_removable_columns){ for (auto& p : matrix_){ if constexpr (Master_matrix::Option_list::has_row_access){ p.second.set_rows(&this->rows_); } - p.second.set_pivot_to_column_map(&pivotToColumnIndex_); } } else { for (auto& col : matrix_){ if constexpr (Master_matrix::Option_list::has_row_access){ col.set_rows(&this->rows_); } - col.set_pivot_to_column_map(&pivotToColumnIndex_); } } } @@ -383,7 +367,7 @@ inline std::vector Chain_matrix::remove_maximal_simplex([[maybe_unused]] //TODO: find simple test to verify that col at columnIndex is maximal. - if constexpr (Master_matrix::Option_list::has_dimension_access){ + if constexpr (Master_matrix::Option_list::has_matrix_maximal_dimension_access){ dim_opt::update_down(matrix_.at(toErase).get_dimension()); } @@ -500,9 +484,11 @@ template inline void Chain_matrix::add_to(index sourceColumnIndex, index targetColumnIndex) { if constexpr (Master_matrix::Option_list::has_removable_columns){ - matrix_.at(targetColumnIndex) += matrix_.at(sourceColumnIndex); + auto& col = matrix_.at(targetColumnIndex); + _add_to(col, [&](){col += matrix_.at(sourceColumnIndex);}); } else { - matrix_[targetColumnIndex] += matrix_[sourceColumnIndex]; + auto& col = matrix_[targetColumnIndex]; + _add_to(col, [&](){col += matrix_[sourceColumnIndex];}); } } @@ -510,9 +496,11 @@ template inline void Chain_matrix::add_to(Column_type& sourceColumn, index targetColumnIndex) { if constexpr (Master_matrix::Option_list::has_removable_columns){ - matrix_.at(targetColumnIndex) += sourceColumn; + auto& col = matrix_.at(targetColumnIndex); + _add_to(col, [&](){col += sourceColumn;}); } else { - matrix_[targetColumnIndex] += sourceColumn; + auto& col = matrix_[targetColumnIndex]; + _add_to(col, [&](){col += sourceColumn;}); } } @@ -520,9 +508,11 @@ template inline void Chain_matrix::add_to(Column_type& sourceColumn, const Field_element_type& coefficient, index targetColumnIndex) { if constexpr (Master_matrix::Option_list::has_removable_columns){ - matrix_.at(targetColumnIndex).multiply_and_add(coefficient, sourceColumn); + auto& col = matrix_.at(targetColumnIndex); + _add_to(col, [&](){col.multiply_and_add(coefficient, sourceColumn);}); } else { - matrix_[targetColumnIndex].multiply_and_add(coefficient, sourceColumn); + auto& col = matrix_[targetColumnIndex]; + _add_to(col, [&](){col.multiply_and_add(coefficient, sourceColumn);}); } } @@ -530,9 +520,11 @@ template inline void Chain_matrix::add_to(const Field_element_type& coefficient, Column_type& sourceColumn, index targetColumnIndex) { if constexpr (Master_matrix::Option_list::has_removable_columns){ - matrix_.at(targetColumnIndex).multiply_and_add(sourceColumn, coefficient); + auto& col = matrix_.at(targetColumnIndex); + _add_to(col, [&](){col.multiply_and_add(sourceColumn, coefficient);}); } else { - matrix_[targetColumnIndex].multiply_and_add(sourceColumn, coefficient); + auto& col = matrix_[targetColumnIndex]; + _add_to(col, [&](){col.multiply_and_add(sourceColumn, coefficient);}); } } @@ -593,22 +585,22 @@ inline Chain_matrix &Chain_matrix::operator=( if constexpr (Master_matrix::Option_list::has_removable_columns){ for (const auto& p : other.matrix_){ const Column_type& col = p.second; - matrix_.try_emplace(p.first, Column_type(col, col.get_column_index(), ra_opt::rows_, pivotToColumnIndex_)); + matrix_.try_emplace(p.first, Column_type(col, col.get_column_index(), ra_opt::rows_)); } } else { for (const auto& col : other.matrix_){ - matrix_.emplace_back(col, col.get_column_index(), ra_opt::rows_, pivotToColumnIndex_); + matrix_.emplace_back(col, col.get_column_index(), ra_opt::rows_); } } } else { if constexpr (Master_matrix::Option_list::has_removable_columns){ for (const auto& p : other.matrix_){ const Column_type& col = p.second; - matrix_.try_emplace(p.first, Column_type(col, col.get_column_index(), pivotToColumnIndex_)); + matrix_.try_emplace(p.first, Column_type(col, col.get_column_index())); } } else { for (const auto& col : other.matrix_){ - matrix_.emplace_back(col, col.get_column_index(), pivotToColumnIndex_); + matrix_.emplace_back(col, col.get_column_index()); } } } @@ -742,7 +734,7 @@ inline void Chain_matrix::_reduce_by_G( { Column_type& col = get_column(currentPivot); if constexpr (Master_matrix::Option_list::is_z2){ - _add_to(col, column); //Reduce with the column col_g + _add_to(col, column, 1u); //Reduce with the column col_g chainsInH.push_back(col.get_paired_chain_index());//keep the col_h with which col_g is paired } else { Field_element_type coef = col.get_pivot_value(); @@ -762,7 +754,7 @@ inline void Chain_matrix::_reduce_by_F( { Column_type& col = get_column(currentPivot); if constexpr (Master_matrix::Option_list::is_z2){ - _add_to(col, column); //Reduce with the column col_g + _add_to(col, column, 1u); //Reduce with the column col_g chainsInF.push_back(currentPivot); } else { Field_element_type coef = col.get_pivot_value(); @@ -783,7 +775,7 @@ inline void Chain_matrix::_build_from_H( if constexpr (Master_matrix::Option_list::is_z2){ column.insert(simplexIndex); for (index idx_h : chainsInH) { - _add_to(get_column(idx_h), column); + _add_to(get_column(idx_h), column, 1u); } } else { column.emplace(simplexIndex, 1); @@ -829,9 +821,9 @@ inline void Chain_matrix::_insert_chain( } if constexpr (Master_matrix::Option_list::has_row_access){ - matrix_.try_emplace(nextInsertIndex_, Column_type(nextInsertIndex_, column, dimension, ra_opt::rows_, pivotToColumnIndex_)); + matrix_.try_emplace(nextInsertIndex_, Column_type(nextInsertIndex_, column, dimension, ra_opt::rows_)); } else { - matrix_.try_emplace(nextInsertIndex_, Column_type(column, dimension, pivotToColumnIndex_)); + matrix_.try_emplace(nextInsertIndex_, Column_type(column, dimension)); } if constexpr (Master_matrix::Option_list::has_column_pairings){ @@ -840,9 +832,9 @@ inline void Chain_matrix::_insert_chain( } } else { if constexpr (Master_matrix::Option_list::has_row_access){ - matrix_.emplace_back(nextInsertIndex_, column, dimension, ra_opt::rows_, pivotToColumnIndex_); + matrix_.emplace_back(nextInsertIndex_, column, dimension, ra_opt::rows_); } else { - matrix_.emplace_back(column, dimension, pivotToColumnIndex_); + matrix_.emplace_back(column, dimension); } pivotToColumnIndex_[nextInsertIndex_] = nextInsertIndex_; @@ -870,9 +862,9 @@ inline void Chain_matrix::_insert_chain( pivotToColumnIndex_.try_emplace(pivot, nextInsertIndex_); if constexpr (Master_matrix::Option_list::has_row_access){ - matrix_.try_emplace(nextInsertIndex_, Column_type(nextInsertIndex_, column, dimension, ra_opt::rows_, pivotToColumnIndex_)); + matrix_.try_emplace(nextInsertIndex_, Column_type(nextInsertIndex_, column, dimension, ra_opt::rows_)); } else { - matrix_.try_emplace(nextInsertIndex_, Column_type(column, dimension, pivotToColumnIndex_)); + matrix_.try_emplace(nextInsertIndex_, Column_type(column, dimension)); } matrix_.at(nextInsertIndex_).assign_paired_chain(pair); @@ -890,9 +882,9 @@ inline void Chain_matrix::_insert_chain( } } else { if constexpr (Master_matrix::Option_list::has_row_access){ - matrix_.emplace_back(nextInsertIndex_, column, dimension, ra_opt::rows_, pivotToColumnIndex_); + matrix_.emplace_back(nextInsertIndex_, column, dimension, ra_opt::rows_); } else { - matrix_.emplace_back(column, dimension, pivotToColumnIndex_); + matrix_.emplace_back(column, dimension); } matrix_[nextInsertIndex_].assign_paired_chain(pair); @@ -947,6 +939,22 @@ inline void Chain_matrix::_add_to( } } +template +template +inline void Chain_matrix::_add_to(Column_type& target, F&& addition) +{ + auto pivot = target.get_pivot(); + addition(); + + if (pivot != target.get_pivot()){ + if constexpr (Master_matrix::Option_list::has_removable_columns){ + std::swap(pivotToColumnIndex_.at(pivot), pivotToColumnIndex_.at(target.get_pivot())); + } else { + std::swap(pivotToColumnIndex_[pivot], pivotToColumnIndex_[target.get_pivot()]); + } + } +} + template inline constexpr typename Chain_matrix::barcode_type & Chain_matrix::_barcode() diff --git a/src/Persistence_matrix/include/gudhi/Persistence_matrix/chain_pairing.h b/src/Persistence_matrix/include/gudhi/Persistence_matrix/chain_pairing.h index 671acadd0d..07fcb43273 100644 --- a/src/Persistence_matrix/include/gudhi/Persistence_matrix/chain_pairing.h +++ b/src/Persistence_matrix/include/gudhi/Persistence_matrix/chain_pairing.h @@ -16,6 +16,12 @@ namespace Gudhi { namespace persistence_matrix { +struct Dummy_chain_pairing{ + friend void swap([[maybe_unused]] Dummy_chain_pairing& d1, [[maybe_unused]] Dummy_chain_pairing& d2){} + + Dummy_chain_pairing(){} +}; + template class Chain_pairing { diff --git a/src/Persistence_matrix/include/gudhi/Persistence_matrix/chain_rep_cycles.h b/src/Persistence_matrix/include/gudhi/Persistence_matrix/chain_rep_cycles.h index 4a86b31d76..d355900acf 100644 --- a/src/Persistence_matrix/include/gudhi/Persistence_matrix/chain_rep_cycles.h +++ b/src/Persistence_matrix/include/gudhi/Persistence_matrix/chain_rep_cycles.h @@ -15,11 +15,17 @@ #include //std::sort #include -#include "../options.h" //Column_types - namespace Gudhi { namespace persistence_matrix { +struct Dummy_chain_representative_cycles{ + friend void swap([[maybe_unused]] Dummy_chain_representative_cycles& d1, [[maybe_unused]] Dummy_chain_representative_cycles& d2){} + + Dummy_chain_representative_cycles(){} + Dummy_chain_representative_cycles([[maybe_unused]] const Dummy_chain_representative_cycles& matrixToCopy){} + Dummy_chain_representative_cycles([[maybe_unused]] Dummy_chain_representative_cycles&& other) noexcept{} +}; + template class Chain_representative_cycles { @@ -30,7 +36,7 @@ class Chain_representative_cycles using matrix_type = typename Master_matrix::column_container_type; using dictionnary_type = typename Master_matrix::template dictionnary_type; - Chain_representative_cycles(matrix_type& matrix, dictionnary_type& pivotToPosition); + Chain_representative_cycles(); Chain_representative_cycles(const Chain_representative_cycles& matrixToCopy); Chain_representative_cycles(Chain_representative_cycles&& other) noexcept; @@ -46,33 +52,29 @@ class Chain_representative_cycles base1.birthToCycle_.swap(base2.birthToCycle_); } -protected: - matrix_type* matrix_; - dictionnary_type* pivotToPosition_; - private: + using chain_matrix = typename Master_matrix::Chain_matrix_type; + std::vector representativeCycles_; std::vector birthToCycle_; + + constexpr chain_matrix* _matrix() { return static_cast(this); } + constexpr const chain_matrix* _matrix() const { return static_cast(this); } }; template -inline Chain_representative_cycles::Chain_representative_cycles(matrix_type &matrix, dictionnary_type &pivotToPosition) - : matrix_(&matrix), pivotToPosition_(&pivotToPosition) +inline Chain_representative_cycles::Chain_representative_cycles() {} template inline Chain_representative_cycles::Chain_representative_cycles(const Chain_representative_cycles& matrixToCopy) - : matrix_(matrixToCopy.matrix_), - pivotToPosition_(matrixToCopy.pivotToPosition_), - representativeCycles_(matrixToCopy.representativeCycles_), + : representativeCycles_(matrixToCopy.representativeCycles_), birthToCycle_(matrixToCopy.birthToCycle_) {} template inline Chain_representative_cycles::Chain_representative_cycles(Chain_representative_cycles&& other) noexcept - : matrix_(other.matrix_), - pivotToPosition_(other.pivotToPosition_), - representativeCycles_(std::move(other.representativeCycles_)), + : representativeCycles_(std::move(other.representativeCycles_)), birthToCycle_(std::move(other.birthToCycle_)) {} @@ -80,17 +82,18 @@ template inline void Chain_representative_cycles::update_representative_cycles() { birthToCycle_.clear(); - birthToCycle_.resize(matrix_->size(), -1); + birthToCycle_.resize(_matrix()->get_number_of_columns(), -1); representativeCycles_.clear(); - for (index i = 0; i < matrix_->size(); i++){ - auto &col = matrix_->at(pivotToPosition_->at(i)); + for (index i = 0; i < _matrix()->get_number_of_columns(); i++){ + auto &col = _matrix()->get_column(_matrix()->get_column_with_pivot(i)); if (!col.is_paired() || i < col.get_paired_chain_index()){ cycle_type cycle; for (auto& c : col){ cycle.push_back(c.get_row_index()); } - if constexpr (Master_matrix::Option_list::column_type == Column_types::UNORDERED_SET) + if constexpr (std::is_same_v + || std::is_same_v) std::sort(cycle.begin(), cycle.end()); representativeCycles_.push_back(cycle); birthToCycle_[i] = representativeCycles_.size() - 1; diff --git a/src/Persistence_matrix/include/gudhi/Persistence_matrix/chain_vine_swap.h b/src/Persistence_matrix/include/gudhi/Persistence_matrix/chain_vine_swap.h index e84b92c57b..d23d49043e 100644 --- a/src/Persistence_matrix/include/gudhi/Persistence_matrix/chain_vine_swap.h +++ b/src/Persistence_matrix/include/gudhi/Persistence_matrix/chain_vine_swap.h @@ -27,6 +27,16 @@ static constexpr bool _no_G_death_comparator(std::size_t columnIndex1, std::size return false; } +struct Dummy_chain_vine_swap{ + friend void swap([[maybe_unused]] Dummy_chain_vine_swap& d1, [[maybe_unused]] Dummy_chain_vine_swap& d2){} + + Dummy_chain_vine_swap(){} + template + Dummy_chain_vine_swap([[maybe_unused]] BirthComparatorFunction&& birthComparator, [[maybe_unused]] DeathComparatorFunction&& deathComparator){} + Dummy_chain_vine_swap([[maybe_unused]] const Dummy_chain_vine_swap &matrixToCopy){} + Dummy_chain_vine_swap([[maybe_unused]] Dummy_chain_vine_swap&& other) noexcept{} +}; + template class Chain_barcode_swap : public Chain_pairing { @@ -35,8 +45,12 @@ class Chain_barcode_swap : public Chain_pairing using CP = Chain_pairing; Chain_barcode_swap(){}; - Chain_barcode_swap(const Chain_barcode_swap &toCopy) : Chain_pairing(toCopy), pivotToPosition_(toCopy.pivotToPosition_) {}; - Chain_barcode_swap(Chain_barcode_swap &&other) : Chain_pairing(std::move(other)), pivotToPosition_(std::move(other.pivotToPosition_)) {}; + Chain_barcode_swap(const Chain_barcode_swap &toCopy) + : CP(static_cast(toCopy)), + pivotToPosition_(toCopy.pivotToPosition_) {}; + Chain_barcode_swap(Chain_barcode_swap &&other) + : CP(std::move(static_cast(other))), + pivotToPosition_(std::move(other.pivotToPosition_)) {}; void swap_positions(index pivot1, index pivot2){ if constexpr (Master_matrix::Option_list::has_removable_columns){ @@ -166,7 +180,7 @@ template class Chain_vine_swap : public std::conditional< Master_matrix::Option_list::has_column_pairings, Chain_barcode_swap, - typename Master_matrix::Dummy_chain_pairing + Dummy_chain_pairing >::type { public: @@ -177,10 +191,9 @@ class Chain_vine_swap : public std::conditional< typedef bool (*BirthCompFuncPointer)(index,index); typedef bool (*DeathCompFuncPointer)(index,index); - Chain_vine_swap(matrix_type& matrix); + Chain_vine_swap(); template - Chain_vine_swap(matrix_type& matrix, - BirthComparatorFunction&& birthComparator, + Chain_vine_swap(BirthComparatorFunction&& birthComparator, DeathComparatorFunction&& deathComparator = _no_G_death_comparator); Chain_vine_swap(const Chain_vine_swap &matrixToCopy); Chain_vine_swap(Chain_vine_swap&& other) noexcept; @@ -198,32 +211,31 @@ class Chain_vine_swap : public std::conditional< std::swap(swap1.deathComp_, swap2.deathComp_); } -protected: - matrix_type* matrix_; - private: using CP = typename std::conditional< Master_matrix::Option_list::has_column_pairings, Chain_barcode_swap, - typename Master_matrix::Dummy_chain_pairing + Dummy_chain_pairing >::type; + using chain_matrix = typename Master_matrix::Chain_matrix_type; + + BirthCompFuncPointer birthComp_; // for F x F & H x H + DeathCompFuncPointer deathComp_; // for G x G - void _add_to(const typename Column_type::Column_type& column, std::set& set); bool _is_negative_in_pair(index columnIndex) const; - bool _is_paired(index columnIndex) const; index _positive_vine_swap(index columnIndex1, index columnIndex2); index _positive_negative_vine_swap(index columnIndex1, index columnIndex2); index _negative_positive_vine_swap(index columnIndex1, index columnIndex2); index _negative_vine_swap(index columnIndex1, index columnIndex2); - BirthCompFuncPointer birthComp_; // for F x F & H x H - DeathCompFuncPointer deathComp_; // for G x G + constexpr chain_matrix* _matrix() { return static_cast(this); } + constexpr const chain_matrix* _matrix() const { return static_cast(this); } }; template -inline Chain_vine_swap::Chain_vine_swap(matrix_type &matrix) - : CP(), matrix_(&matrix), birthComp_(nullptr), deathComp_(nullptr) +inline Chain_vine_swap::Chain_vine_swap() + : CP(), birthComp_(nullptr), deathComp_(nullptr) { static_assert(Master_matrix::Option_list::has_column_pairings, "If barcode is not stored, at least a birth comparator has to be specified."); @@ -232,25 +244,22 @@ inline Chain_vine_swap::Chain_vine_swap(matrix_type &matrix) template template inline Chain_vine_swap::Chain_vine_swap( - matrix_type &matrix, BirthComparatorFunction&& birthComparator, DeathComparatorFunction&& deathComparator) - : CP(), matrix_(&matrix), birthComp_(&birthComparator), deathComp_(&deathComparator) + : CP(), birthComp_(&birthComparator), deathComp_(&deathComparator) {} template inline Chain_vine_swap::Chain_vine_swap( const Chain_vine_swap &matrixToCopy) - : CP(matrixToCopy), - matrix_(matrixToCopy.matrix_), + : CP(static_cast(matrixToCopy)), birthComp_(matrixToCopy.birthComp_), deathComp_(matrixToCopy.deathComp_) {} template inline Chain_vine_swap::Chain_vine_swap(Chain_vine_swap &&other) noexcept - : CP(std::move(other)), - matrix_(other.matrix_), + : CP(std::move(static_cast(other))), birthComp_(std::move(other.birthComp_)), deathComp_(std::move(other.deathComp_)) {} @@ -259,17 +268,20 @@ template inline typename Chain_vine_swap::index Chain_vine_swap::vine_swap_with_z_eq_1_case(index columnIndex1, index columnIndex2) { if constexpr (Master_matrix::Option_list::has_column_pairings){ - assert(CP::are_adjacent(matrix_->at(columnIndex1).get_pivot(), matrix_->at(columnIndex2).get_pivot()) + assert(CP::are_adjacent(_matrix()->get_pivot(columnIndex1), _matrix()->get_pivot(columnIndex2)) && "Columns to be swaped need to be adjacent in the 'real' matrix."); } - if (_is_negative_in_pair(columnIndex1) && _is_negative_in_pair(columnIndex2)) + const bool col1IsNeg = _is_negative_in_pair(columnIndex1); + const bool col2IsNeg = _is_negative_in_pair(columnIndex2); + + if (col1IsNeg && col2IsNeg) return _negative_vine_swap(columnIndex1, columnIndex2); - if (_is_negative_in_pair(columnIndex1)) + if (col1IsNeg) return _negative_positive_vine_swap(columnIndex1, columnIndex2); - if (_is_negative_in_pair(columnIndex2)) + if (col2IsNeg) return _positive_negative_vine_swap(columnIndex1, columnIndex2); return _positive_vine_swap(columnIndex1, columnIndex2); @@ -279,14 +291,17 @@ template inline typename Chain_vine_swap::index Chain_vine_swap::vine_swap(index columnIndex1, index columnIndex2) { if constexpr (Master_matrix::Option_list::has_column_pairings){ - assert(CP::are_adjacent(matrix_->at(columnIndex1).get_pivot(), matrix_->at(columnIndex2).get_pivot()) && "Columns to be swaped need to be adjacent in the 'real' matrix."); + assert(CP::are_adjacent(_matrix()->get_pivot(columnIndex1), _matrix()->get_pivot(columnIndex2)) && "Columns to be swaped need to be adjacent in the 'real' matrix."); } - if (_is_negative_in_pair(columnIndex1) && _is_negative_in_pair(columnIndex2)){ - if (!matrix_->at(columnIndex2).is_non_zero(matrix_->at(columnIndex1).get_pivot())){ + const bool col1IsNeg = _is_negative_in_pair(columnIndex1); + const bool col2IsNeg = _is_negative_in_pair(columnIndex2); + + if (col1IsNeg && col2IsNeg){ + if (_matrix()->is_zero_cell(columnIndex2, _matrix()->get_pivot(columnIndex1))){ if constexpr (Master_matrix::Option_list::has_column_pairings){ - index pivot1 = matrix_->at(columnIndex1).get_pivot(); - index pivot2 = matrix_->at(columnIndex2).get_pivot(); + index pivot1 = _matrix()->get_pivot(columnIndex1); + index pivot2 = _matrix()->get_pivot(columnIndex2); CP::negative_transpose(pivot1, pivot2); CP::swap_positions(pivot1, pivot2); @@ -296,11 +311,11 @@ inline typename Chain_vine_swap::index Chain_vine_swapat(columnIndex2).is_non_zero(matrix_->at(columnIndex1).get_pivot())){ + if (col1IsNeg){ + if (_matrix()->is_zero_cell(columnIndex2, _matrix()->get_pivot(columnIndex1))){ if constexpr (Master_matrix::Option_list::has_column_pairings){ - index pivot1 = matrix_->at(columnIndex1).get_pivot(); - index pivot2 = matrix_->at(columnIndex2).get_pivot(); + index pivot1 = _matrix()->get_pivot(columnIndex1); + index pivot2 = _matrix()->get_pivot(columnIndex2); CP::negative_positive_transpose(pivot1, pivot2); CP::swap_positions(pivot1, pivot2); @@ -310,11 +325,11 @@ inline typename Chain_vine_swap::index Chain_vine_swapat(columnIndex2).is_non_zero(matrix_->at(columnIndex1).get_pivot())){ + if (col2IsNeg){ + if (_matrix()->is_zero_cell(columnIndex2, _matrix()->get_pivot(columnIndex1))){ if constexpr (Master_matrix::Option_list::has_column_pairings){ - index pivot1 = matrix_->at(columnIndex1).get_pivot(); - index pivot2 = matrix_->at(columnIndex2).get_pivot(); + index pivot1 = _matrix()->get_pivot(columnIndex1); + index pivot2 = _matrix()->get_pivot(columnIndex2); CP::positive_negative_transpose(pivot1, pivot2); CP::swap_positions(pivot1, pivot2); @@ -324,10 +339,10 @@ inline typename Chain_vine_swap::index Chain_vine_swapat(columnIndex2).is_non_zero(matrix_->at(columnIndex1).get_pivot())){ + if (_matrix()->is_zero_cell(columnIndex2, _matrix()->get_pivot(columnIndex1))){ if constexpr (Master_matrix::Option_list::has_column_pairings){ - index pivot1 = matrix_->at(columnIndex1).get_pivot(); - index pivot2 = matrix_->at(columnIndex2).get_pivot(); + index pivot1 = _matrix()->get_pivot(columnIndex1); + index pivot2 = _matrix()->get_pivot(columnIndex2); CP::positive_transpose(pivot1, pivot2); CP::swap_positions(pivot1, pivot2); @@ -347,74 +362,58 @@ inline Chain_vine_swap &Chain_vine_swap::operator= return *this; } -template -inline void Chain_vine_swap::_add_to( - const typename Column_type::Column_type& column, std::set& set) -{ - std::pair::iterator,bool> res_insert; - for (const typename Column_type::Cell &cell : column) { - res_insert = set.insert(cell.get_row_index()); - if (!res_insert.second) { - set.erase(res_insert.first); - } - } -} - template inline bool Chain_vine_swap::_is_negative_in_pair(index columnIndex) const { if constexpr (Master_matrix::Option_list::has_column_pairings){ - return CP::is_negative_in_pair(matrix_->at(columnIndex).get_pivot()); + return CP::is_negative_in_pair(_matrix()->get_pivot(columnIndex)); } else { - const auto& col = matrix_->at(columnIndex); + const auto& col = _matrix()->get_column(columnIndex); if (!col.is_paired()) return false; - return col.get_pivot() > matrix_->at(col.get_paired_chain_index()).get_pivot(); + return col.get_pivot() > _matrix()->get_pivot(col.get_paired_chain_index()); } } -template -inline bool Chain_vine_swap::_is_paired(index columnIndex) const -{ - // return CP::death(CP::_get_pivot_position(matrix_->at(columnIndex).get_pivot())) != -1; - return matrix_->at(columnIndex).is_paired(); -} - template inline typename Chain_vine_swap::index Chain_vine_swap::_positive_vine_swap(index columnIndex1, index columnIndex2) { + auto& col1 = _matrix()->get_column(columnIndex1); + auto& col2 = _matrix()->get_column(columnIndex2); + if constexpr (Master_matrix::Option_list::has_column_pairings){ - CP::swap_positions(matrix_->at(columnIndex1).get_pivot(), matrix_->at(columnIndex2).get_pivot()); + CP::swap_positions(col1.get_pivot(), col2.get_pivot()); } //TODO: factorize the cases. But for debug it is much more easier to understand what is happening splitted like this - if (!_is_paired(columnIndex1)){ // F x * + if (!col1.is_paired()){ // F x * bool hasSmallerBirth; if constexpr (Master_matrix::Option_list::has_column_pairings){ - hasSmallerBirth = (CP::birth(matrix_->at(columnIndex1).get_pivot()) < CP::birth(matrix_->at(columnIndex2).get_pivot())); + hasSmallerBirth = (CP::birth(col1.get_pivot()) < CP::birth(col2.get_pivot())); } else { hasSmallerBirth = birthComp_(columnIndex1, columnIndex2); } - if (!_is_paired(columnIndex2) && hasSmallerBirth){ - matrix_->at(columnIndex2) += matrix_->at(columnIndex1); + if (!col2.is_paired() && hasSmallerBirth){ + _matrix()->add_to(columnIndex1, columnIndex2); if constexpr (Master_matrix::Option_list::has_column_pairings){ - CP::positive_transpose(matrix_->at(columnIndex1).get_pivot(), matrix_->at(columnIndex2).get_pivot()); + CP::positive_transpose(col1.get_pivot(), col2.get_pivot()); } return columnIndex1; } - matrix_->at(columnIndex1) += matrix_->at(columnIndex2); + _matrix()->add_to(columnIndex2, columnIndex1); + return columnIndex2; } - if (!_is_paired(columnIndex2)){ // G x F - matrix_->at(columnIndex2) += matrix_->at(columnIndex1); + if (!col2.is_paired()){ // G x F + static_cast(this)->add_to(columnIndex1, columnIndex2); if constexpr (Master_matrix::Option_list::has_column_pairings){ - CP::positive_transpose(matrix_->at(columnIndex1).get_pivot(), matrix_->at(columnIndex2).get_pivot()); + CP::positive_transpose(col1.get_pivot(), col2.get_pivot()); } return columnIndex1; } bool hasSmallerDeath; if constexpr (Master_matrix::Option_list::has_column_pairings){ - hasSmallerDeath = (CP::death(matrix_->at(columnIndex1).get_pivot()) < CP::death(matrix_->at(columnIndex2).get_pivot())); + hasSmallerDeath = (CP::death(col1.get_pivot()) < CP::death(col2.get_pivot())); } else { hasSmallerDeath = deathComp_(columnIndex1, columnIndex2); } @@ -422,16 +421,16 @@ inline typename Chain_vine_swap::index Chain_vine_swapat(pairedIndex1).get_pivot() < matrix_->at(pairedIndex2).get_pivot()) ??? { - matrix_->at(matrix_->at(columnIndex2).get_paired_chain_index()) += matrix_->at(matrix_->at(columnIndex1).get_paired_chain_index()); - matrix_->at(columnIndex2) += matrix_->at(columnIndex1); + _matrix()->add_to(col1.get_paired_chain_index(), col2.get_paired_chain_index()); + _matrix()->add_to(columnIndex1, columnIndex2); if constexpr (Master_matrix::Option_list::has_column_pairings){ - CP::positive_transpose(matrix_->at(columnIndex1).get_pivot(), matrix_->at(columnIndex2).get_pivot()); + CP::positive_transpose(col1.get_pivot(), col2.get_pivot()); } return columnIndex1; } - matrix_->at(matrix_->at(columnIndex1).get_paired_chain_index()) += matrix_->at(matrix_->at(columnIndex2).get_paired_chain_index()); - matrix_->at(columnIndex1) += matrix_->at(columnIndex2); + _matrix()->add_to(col2.get_paired_chain_index(), col1.get_paired_chain_index()); + _matrix()->add_to(columnIndex2, columnIndex1); return columnIndex2; } @@ -439,11 +438,11 @@ inline typename Chain_vine_swap::index Chain_vine_swap inline typename Chain_vine_swap::index Chain_vine_swap::_positive_negative_vine_swap(index columnIndex1, index columnIndex2) { - matrix_->at(columnIndex2) += matrix_->at(columnIndex1); + _matrix()->add_to(columnIndex1, columnIndex2); if constexpr (Master_matrix::Option_list::has_column_pairings){ - index pivot1 = matrix_->at(columnIndex1).get_pivot(); - index pivot2 = matrix_->at(columnIndex2).get_pivot(); + index pivot1 = _matrix()->get_pivot(columnIndex1); + index pivot2 = _matrix()->get_pivot(columnIndex2); CP::positive_negative_transpose(pivot1, pivot2); CP::swap_positions(pivot1, pivot2); @@ -455,10 +454,10 @@ inline typename Chain_vine_swap::index Chain_vine_swap inline typename Chain_vine_swap::index Chain_vine_swap::_negative_positive_vine_swap(index columnIndex1, index columnIndex2) { - matrix_->at(columnIndex1) += matrix_->at(columnIndex2); + _matrix()->add_to(columnIndex2, columnIndex1); if constexpr (Master_matrix::Option_list::has_column_pairings){ - CP::swap_positions(matrix_->at(columnIndex1).get_pivot(), matrix_->at(columnIndex2).get_pivot()); + CP::swap_positions(_matrix()->get_pivot(columnIndex1), _matrix()->get_pivot(columnIndex2)); } return columnIndex2; @@ -467,39 +466,41 @@ inline typename Chain_vine_swap::index Chain_vine_swap inline typename Chain_vine_swap::index Chain_vine_swap::_negative_vine_swap(index columnIndex1, index columnIndex2) { - index pairedIndex1 = matrix_->at(columnIndex1).get_paired_chain_index(); - index pairedIndex2 = matrix_->at(columnIndex2).get_paired_chain_index(); + auto& col1 = _matrix()->get_column(columnIndex1); + auto& col2 = _matrix()->get_column(columnIndex2); + + index pairedIndex1 = col1.get_paired_chain_index(); + index pairedIndex2 = col2.get_paired_chain_index(); bool hasSmallerBirth; if constexpr (Master_matrix::Option_list::has_column_pairings){ - hasSmallerBirth = (CP::birth(matrix_->at(columnIndex1).get_pivot()) < CP::birth(matrix_->at(columnIndex2).get_pivot())); + hasSmallerBirth = (CP::birth(col1.get_pivot()) < CP::birth(col2.get_pivot())); } else { hasSmallerBirth = birthComp_(columnIndex1, columnIndex2); //for debug, to remove - if (hasSmallerBirth != - (matrix_->at(pairedIndex1).get_pivot() < matrix_->at(pairedIndex2).get_pivot())) + if (hasSmallerBirth != (_matrix()->get_pivot(pairedIndex1) < _matrix()->get_pivot(pairedIndex2))) std::cout << "!!!!!!!!!!!!!!!!!! not equal\n"; } if constexpr (Master_matrix::Option_list::has_column_pairings){ - CP::swap_positions(matrix_->at(columnIndex1).get_pivot(), matrix_->at(columnIndex2).get_pivot()); + CP::swap_positions(col1.get_pivot(), col2.get_pivot()); } if (hasSmallerBirth) //== matrix_->at(pairedIndex1).get_pivot() < matrix_->at(pairedIndex2).get_pivot() ? { - matrix_->at(pairedIndex2) += matrix_->at(pairedIndex1); - matrix_->at(columnIndex2) += matrix_->at(columnIndex1); + _matrix()->add_to(pairedIndex1, pairedIndex2); + _matrix()->add_to(columnIndex1, columnIndex2); if constexpr (Master_matrix::Option_list::has_column_pairings){ - CP::negative_transpose(matrix_->at(columnIndex1).get_pivot(), matrix_->at(columnIndex2).get_pivot()); + CP::negative_transpose(col1.get_pivot(), col2.get_pivot()); } return columnIndex1; } - matrix_->at(pairedIndex1) += matrix_->at(pairedIndex2); - matrix_->at(columnIndex1) += matrix_->at(columnIndex2); + _matrix()->add_to(pairedIndex2, pairedIndex1); + _matrix()->add_to(columnIndex2, columnIndex1); return columnIndex2; } diff --git a/src/Persistence_matrix/include/gudhi/Persistence_matrix/columns/cell_types.h b/src/Persistence_matrix/include/gudhi/Persistence_matrix/columns/cell_types.h index 00013d32dd..0736babd6b 100644 --- a/src/Persistence_matrix/include/gudhi/Persistence_matrix/columns/cell_types.h +++ b/src/Persistence_matrix/include/gudhi/Persistence_matrix/columns/cell_types.h @@ -17,6 +17,22 @@ namespace Gudhi { namespace persistence_matrix { +struct Dummy_cell_column_index_mixin{ + Dummy_cell_column_index_mixin(){} + template + Dummy_cell_column_index_mixin([[maybe_unused]] index columnIndex){} + Dummy_cell_column_index_mixin([[maybe_unused]] const Dummy_cell_column_index_mixin& cell){}; + Dummy_cell_column_index_mixin([[maybe_unused]] Dummy_cell_column_index_mixin&& cell){}; +}; + +struct Dummy_cell_field_element_mixin{ + Dummy_cell_field_element_mixin(){} + template + Dummy_cell_field_element_mixin([[maybe_unused]] Field_element_type t){} + Dummy_cell_field_element_mixin([[maybe_unused]] const Dummy_cell_field_element_mixin& cell){}; + Dummy_cell_field_element_mixin([[maybe_unused]] Dummy_cell_field_element_mixin&& cell){}; +}; + template class Cell_column_index { @@ -81,6 +97,10 @@ class Cell : public Master_matrix::Cell_column_index_option, public Master_matrix::row_hook_type, public Master_matrix::column_hook_type { +private: + using col_opt = typename Master_matrix::Cell_column_index_option; + using field_opt = typename Master_matrix::Cell_field_element_option; + public: using index = typename Master_matrix::index; using Field_element_type = typename Master_matrix::Field_type; @@ -89,33 +109,33 @@ class Cell : public Master_matrix::Cell_column_index_option, Cell(){}; Cell(index rowIndex) - : Master_matrix::Cell_column_index_option(), - Master_matrix::Cell_field_element_option(), + : col_opt(), + field_opt(), rowIndex_(rowIndex) {}; Cell(index columnIndex, index rowIndex) - : Master_matrix::Cell_column_index_option(columnIndex), - Master_matrix::Cell_field_element_option(), + : col_opt(columnIndex), + field_opt(), rowIndex_(rowIndex) {}; Cell(Field_element_type element, index rowIndex) - : Master_matrix::Cell_column_index_option(), - Master_matrix::Cell_field_element_option(element), + : col_opt(), + field_opt(element), rowIndex_(rowIndex) {}; Cell(Field_element_type element, index columnIndex, index rowIndex) - : Master_matrix::Cell_column_index_option(columnIndex), - Master_matrix::Cell_field_element_option(element), + : col_opt(columnIndex), + field_opt(element), rowIndex_(rowIndex) {}; Cell(const Cell& cell) - : Master_matrix::Cell_column_index_option(cell), - Master_matrix::Cell_field_element_option(cell), + : col_opt(static_cast(cell)), + field_opt(static_cast(cell)), rowIndex_(cell.rowIndex_) {}; Cell(Cell&& cell) noexcept - : Master_matrix::Cell_column_index_option(std::move(cell)), - Master_matrix::Cell_field_element_option(std::move(cell)), + : col_opt(std::move(static_cast(cell))), + field_opt(std::move(static_cast(cell))), rowIndex_(std::exchange(cell.rowIndex_, 0)) {}; @@ -127,8 +147,8 @@ class Cell : public Master_matrix::Cell_column_index_option, }; Cell& operator=(Cell other){ - Master_matrix::Cell_column_index_option::operator=(other); - Master_matrix::Cell_field_element_option::operator=(other); + col_opt::operator=(other); + field_opt::operator=(other); std::swap(rowIndex_, other.rowIndex_); return *this; }; @@ -147,7 +167,7 @@ class Cell : public Master_matrix::Cell_column_index_option, if constexpr (Master_matrix::Option_list::is_z2){ return {rowIndex_, 1}; } else { - return {rowIndex_, Master_matrix::Cell_field_element_option::element_}; + return {rowIndex_, field_opt::element_}; } } diff --git a/src/Persistence_matrix/include/gudhi/Persistence_matrix/columns/chain_column_extra_properties.h b/src/Persistence_matrix/include/gudhi/Persistence_matrix/columns/chain_column_extra_properties.h new file mode 100644 index 0000000000..1b55f2d07c --- /dev/null +++ b/src/Persistence_matrix/include/gudhi/Persistence_matrix/columns/chain_column_extra_properties.h @@ -0,0 +1,72 @@ +/* This file is part of the Gudhi Library - https://gudhi.inria.fr/ - which is released under MIT. + * See file LICENSE or go to https://gudhi.inria.fr/licensing/ for full license details. + * Author(s): Hannah Schreiber + * + * Copyright (C) 2022-23 Inria + * + * Modification(s): + * - YYYY/MM Author: Description of the modification + */ + +#ifndef PM_CHAIN_COLUMN_PROP_H +#define PM_CHAIN_COLUMN_PROP_H + +#include //std::swap + +namespace Gudhi { +namespace persistence_matrix { + +struct Dummy_chain_properties{ + Dummy_chain_properties(){} + Dummy_chain_properties([[maybe_unused]] int pivot){} + Dummy_chain_properties([[maybe_unused]] int pivot, [[maybe_unused]] int pair){} + Dummy_chain_properties([[maybe_unused]] const Dummy_chain_properties& col){} + Dummy_chain_properties([[maybe_unused]] Dummy_chain_properties&& col){} + + Dummy_chain_properties& operator=([[maybe_unused]] const Dummy_chain_properties& other){ return *this; } + + friend void swap([[maybe_unused]] Dummy_chain_properties& col1, [[maybe_unused]] Dummy_chain_properties& col2){} +}; + +template +class Chain_column_extra_properties{ +public: + using index = typename Master_matrix::index; + + Chain_column_extra_properties() : pivot_(-1), pairedColumn_(-1) {} + Chain_column_extra_properties(int pivot) : pivot_(pivot), pairedColumn_(-1) {} + Chain_column_extra_properties(int pivot, int pair) : pivot_(pivot), pairedColumn_(pair) {} + Chain_column_extra_properties(const Chain_column_extra_properties& col) + : pivot_(col.pivot_), pairedColumn_(col.pairedColumn_) {} + Chain_column_extra_properties(Chain_column_extra_properties&& col) + : pivot_(std::exchange(col.pivot_, -1)), pairedColumn_(std::exchange(col.pairedColumn_, -1)) {} + + index get_paired_chain_index() const { return pairedColumn_; } + bool is_paired() const { return pairedColumn_ != -1; } + void assign_paired_chain(index other_col){ pairedColumn_ = other_col; } + void unassign_paired_chain() { pairedColumn_ = -1; }; + + Chain_column_extra_properties& operator=(const Chain_column_extra_properties& other){ + pivot_ = other.pivot_; + pairedColumn_ = other.pairedColumn_; + return *this; + } + + friend void swap(Chain_column_extra_properties& col1, Chain_column_extra_properties& col2){ + std::swap(col1.pivot_, col2.pivot_); + std::swap(col1.pairedColumn_, col2.pairedColumn_); + } + +protected: + int get_pivot() const { return pivot_; } + void swap_pivots(Chain_column_extra_properties& other) { std::swap(pivot_, other.pivot_); } + +private: + int pivot_; //simplex index associated to the chain + int pairedColumn_; //represents the (F, G x H) partition of the columns +}; + +} //namespace persistence_matrix +} //namespace Gudhi + +#endif // PM_CHAIN_COLUMN_PROP_H diff --git a/src/Persistence_matrix/include/gudhi/Persistence_matrix/columns/column_dimension_holder.h b/src/Persistence_matrix/include/gudhi/Persistence_matrix/columns/column_dimension_holder.h new file mode 100644 index 0000000000..b931de8f5d --- /dev/null +++ b/src/Persistence_matrix/include/gudhi/Persistence_matrix/columns/column_dimension_holder.h @@ -0,0 +1,58 @@ +/* This file is part of the Gudhi Library - https://gudhi.inria.fr/ - which is released under MIT. + * See file LICENSE or go to https://gudhi.inria.fr/licensing/ for full license details. + * Author(s): Hannah Schreiber + * + * Copyright (C) 2022-23 Inria + * + * Modification(s): + * - YYYY/MM Author: Description of the modification + */ + +#ifndef PM_COLUMN_DIM_HOLDER_H +#define PM_COLUMN_DIM_HOLDER_H + +#include //std::swap + +namespace Gudhi { +namespace persistence_matrix { + +struct Dummy_dimension_holder{ + Dummy_dimension_holder(){} + template + Dummy_dimension_holder([[maybe_unused]] dimension_type dim){} + Dummy_dimension_holder([[maybe_unused]] const Dummy_dimension_holder& col){} + Dummy_dimension_holder([[maybe_unused]] Dummy_dimension_holder&& col){} + + Dummy_dimension_holder& operator=([[maybe_unused]] const Dummy_dimension_holder& other){ return *this; } + + friend void swap([[maybe_unused]] Dummy_dimension_holder& col1, [[maybe_unused]] Dummy_dimension_holder& col2){} +}; + +template +struct Column_dimension_holder{ + using dimension_type = typename Master_matrix::dimension_type; + + Column_dimension_holder() : dim_(Master_matrix::Option_list::is_of_boundary_type ? 0 : -1) {} + Column_dimension_holder(dimension_type dim) : dim_(dim) {} + Column_dimension_holder(const Column_dimension_holder& col) : dim_(col.dim_) {} + Column_dimension_holder(Column_dimension_holder&& col) : dim_(std::exchange(col.dim_, -1)) {} + + dimension_type get_dimension() const { return dim_; } + + Column_dimension_holder& operator=(const Column_dimension_holder& other){ + dim_ = other.dim_; + return *this; + } + + friend void swap(Column_dimension_holder& col1, Column_dimension_holder& col2){ + std::swap(col1.dim_, col2.dim_); + } + +private: + dimension_type dim_; +}; + +} //namespace persistence_matrix +} //namespace Gudhi + +#endif // PM_COLUMN_DIM_HOLDER_H diff --git a/src/Persistence_matrix/include/gudhi/Persistence_matrix/columns/intrusive_list_column.h b/src/Persistence_matrix/include/gudhi/Persistence_matrix/columns/intrusive_list_column.h new file mode 100644 index 0000000000..6275366f0b --- /dev/null +++ b/src/Persistence_matrix/include/gudhi/Persistence_matrix/columns/intrusive_list_column.h @@ -0,0 +1,922 @@ +/* This file is part of the Gudhi Library - https://gudhi.inria.fr/ - which is released under MIT. + * See file LICENSE or go to https://gudhi.inria.fr/licensing/ for full license details. + * Author(s): Hannah Schreiber + * + * Copyright (C) 2022-23 Inria + * + * Modification(s): + * - YYYY/MM Author: Description of the modification + */ + +#ifndef PM_INTRUSIVE_LIST_COLUMN_H +#define PM_INTRUSIVE_LIST_COLUMN_H + +// #include +#include +#include + +#include + +#include + +// #include "cell.h" + +namespace Gudhi { +namespace persistence_matrix { + +template +class Intrusive_list_column : public Master_matrix::Row_access_option, + public Master_matrix::Column_dimension_option, + public Master_matrix::Chain_column_option +{ +public: + using Field_element_type = typename std::conditional< + Master_matrix::Option_list::is_z2, + bool, + typename Master_matrix::Field_type + >::type; + using index = typename Master_matrix::index; + using dimension_type = typename Master_matrix::dimension_type; + + using Cell = typename Master_matrix::Cell_type; + using Column_type = boost::intrusive::list < + Cell, + boost::intrusive::constant_time_size, + boost::intrusive::base_hook< typename Master_matrix::base_hook_matrix_list_column > + >; + using iterator = typename Column_type::iterator; + using const_iterator = typename Column_type::const_iterator; + using reverse_iterator = typename Column_type::reverse_iterator; + using const_reverse_iterator = typename Column_type::const_reverse_iterator; + + Intrusive_list_column(); + template + Intrusive_list_column(const Container_type& nonZeroRowIndices); //has to be a boundary for boundary, has no sense for chain if dimension is needed + template + Intrusive_list_column(index columnIndex, const Container_type& nonZeroRowIndices, Row_container_type &rowContainer); //has to be a boundary for boundary, has no sense for chain if dimension is needed + template + Intrusive_list_column(const Container_type& nonZeroChainRowIndices, dimension_type dimension); //dimension gets ignored for base + template + Intrusive_list_column(index columnIndex, const Container_type& nonZeroChainRowIndices, dimension_type dimension, Row_container_type &rowContainer); //dimension gets ignored for base + Intrusive_list_column(const Intrusive_list_column& column); + template + Intrusive_list_column(const Intrusive_list_column& column, index columnIndex, Row_container_type &rowContainer); + Intrusive_list_column(Intrusive_list_column&& column) noexcept; + ~Intrusive_list_column(); + + std::vector get_content(int columnLength = -1) const; + bool is_non_zero(index rowIndex) const; + bool is_empty() const; + + //**************** + //only for base and boundary + template + void reorder(const Map_type& valueMap); //used for lazy row swaps + void clear(); + void clear(index rowIndex); + //**************** + + //**************** + //only for chain and boundary + int get_pivot() const; + Field_element_type get_pivot_value() const; + //**************** + + iterator begin() noexcept; + const_iterator begin() const noexcept; + iterator end() noexcept; + const_iterator end() const noexcept; + reverse_iterator rbegin() noexcept; + const_reverse_iterator rbegin() const noexcept; + reverse_iterator rend() noexcept; + const_reverse_iterator rend() const noexcept; + + template + Intrusive_list_column& operator+=(const Cell_range& column); //for base & boundary except vector + Intrusive_list_column& operator+=(Intrusive_list_column &column); //for chain and vector + friend Intrusive_list_column operator+(Intrusive_list_column column1, Intrusive_list_column& column2){ + column1 += column2; + return column1; + } + + Intrusive_list_column& operator*=(unsigned int v); + friend Intrusive_list_column operator*(Intrusive_list_column column, unsigned int const& v){ + column *= v; + return column; + } + friend Intrusive_list_column operator*(unsigned int const& v, Intrusive_list_column column){ + column *= v; + return column; + } + + //this = v * this + column + template + Intrusive_list_column& multiply_and_add(const Field_element_type& val, const Cell_range& column); //for base & boundary except vector + Intrusive_list_column& multiply_and_add(const Field_element_type& val, Intrusive_list_column& column); //for chain and vector + //this = this + column * v + template + Intrusive_list_column& multiply_and_add(const Cell_range& column, const Field_element_type& val); //for base & boundary except vector + Intrusive_list_column& multiply_and_add(Intrusive_list_column& column, const Field_element_type& val); //for chain and vector + + friend bool operator==(const Intrusive_list_column& c1, const Intrusive_list_column& c2){ + if (&c1 == &c2) return true; + + if constexpr (Master_matrix::Option_list::is_z2){ + return c1.column_ == c2.column_; + } else { + auto it1 = c1.column_.begin(); + auto it2 = c2.column_.begin(); + if (c1.column_.size() != c2.column_.size()) return false; + while (it1 != c1.column_.end() && it2 != c2.column_.end()) { + if (it1->get_row_index() != it2->get_row_index() || it1->get_element() != it2->get_element()) + return false; + ++it1; ++it2; + } + return true; + } + } + friend bool operator<(const Intrusive_list_column& c1, const Intrusive_list_column& c2){ + if (&c1 == &c2) return false; + + if constexpr (Master_matrix::Option_list::is_z2){ + return c1.column_ < c2.column_; + } else { + auto it1 = c1.column_.begin(); + auto it2 = c2.column_.begin(); + while (it1 != c1.column_.end() && it2 != c2.column_.end()) { + if (it1->get_row_index() != it2->get_row_index()) + return it1->get_row_index() < it2->get_row_index(); + if (it1->get_element() != it2->get_element()) + return it1->get_element() < it2->get_element(); + ++it1; ++it2; + } + return it2 != c2.column_.end(); + } + } + + //Disabled with row access. + Intrusive_list_column& operator=(const Intrusive_list_column& other); + + friend void swap(Intrusive_list_column& col1, Intrusive_list_column& col2){ + swap(static_cast(col1), + static_cast(col2)); + swap(static_cast(col1), + static_cast(col2)); + swap(static_cast(col1), + static_cast(col2)); + col1.column_.swap(col2.column_); + } + +protected: + Column_type column_; + inline static Simple_object_pool cellPool_; + + void _delete_cell(iterator& it); + void _insert_cell(const Field_element_type& value, index rowIndex, const iterator& position); + void _insert_cell(index rowIndex, const iterator& position); + template + bool _add(const Cell_range& column); + template + bool _multiply_and_add(const Field_element_type& val, const Cell_range& column); + template + bool _multiply_and_add(const Cell_range& column, const Field_element_type& val); + +private: + using ra_opt = typename Master_matrix::Row_access_option; + using dim_opt = typename Master_matrix::Column_dimension_option; + using chain_opt = typename Master_matrix::Chain_column_option; + + //Cloner object function for boost intrusive container + struct new_cloner + { + Cell *operator()(const Cell& clone_this) + { + return cellPool_.construct(clone_this); + } + }; + + //The disposer object function for boost intrusive container + struct delete_disposer + { + delete_disposer(){}; + delete_disposer(Intrusive_list_column* col) : col_(col) + {}; + + void operator()(Cell *delete_this) + { + if constexpr (Master_matrix::Option_list::has_row_access) + col_->unlink(delete_this); + cellPool_.destroy(delete_this); + } + + Intrusive_list_column* col_; + }; +}; + +template +inline Intrusive_list_column::Intrusive_list_column() : ra_opt(), dim_opt(), chain_opt() +{} + +template +template +inline Intrusive_list_column::Intrusive_list_column(const Container_type &nonZeroRowIndices) + : ra_opt(), + dim_opt(nonZeroRowIndices.size() == 0 ? 0 : nonZeroRowIndices.size() - 1), + chain_opt() +{ + static_assert(!Master_matrix::isNonBasic || Master_matrix::Option_list::is_of_boundary_type, + "Constructor not available for chain columns, please specify the dimension of the chain."); + + if constexpr (Master_matrix::Option_list::is_z2){ + for (index id : nonZeroRowIndices){ + _insert_cell(id, column_.end()); + } + } else { + for (const auto& p : nonZeroRowIndices){ + _insert_cell(p.second, p.first, column_.end()); + } + } +} + +template +template +inline Intrusive_list_column::Intrusive_list_column( + index columnIndex, const Container_type &nonZeroRowIndices, Row_container_type &rowContainer) + : ra_opt(columnIndex, rowContainer), + dim_opt(nonZeroRowIndices.size() == 0 ? 0 : nonZeroRowIndices.size() - 1), + chain_opt([&]{ + if constexpr (Master_matrix::Option_list::is_z2){ + return nonZeroRowIndices.begin() == nonZeroRowIndices.end() ? -1 : *std::prev(nonZeroRowIndices.end()); + } else { + return nonZeroRowIndices.begin() == nonZeroRowIndices.end() ? -1 : std::prev(nonZeroRowIndices.end())->first; + } + }()) +{ + static_assert(!Master_matrix::isNonBasic || Master_matrix::Option_list::is_of_boundary_type, + "Constructor not available for chain columns, please specify the dimension of the chain."); + + if constexpr (Master_matrix::Option_list::is_z2){ + for (index id : nonZeroRowIndices){ + _insert_cell(id, column_.end()); + } + } else { + for (const auto& p : nonZeroRowIndices){ + _insert_cell(p.second, p.first, column_.end()); + } + } +} + +template +template +inline Intrusive_list_column::Intrusive_list_column( + const Container_type &nonZeroRowIndices, dimension_type dimension) + : ra_opt(), + dim_opt(dimension), + chain_opt([&]{ + if constexpr (Master_matrix::Option_list::is_z2){ + return nonZeroRowIndices.begin() == nonZeroRowIndices.end() ? -1 : *std::prev(nonZeroRowIndices.end()); + } else { + return nonZeroRowIndices.begin() == nonZeroRowIndices.end() ? -1 : std::prev(nonZeroRowIndices.end())->first; + } + }()) +{ + if constexpr (Master_matrix::Option_list::is_z2){ + for (index id : nonZeroRowIndices){ + _insert_cell(id, column_.end()); + } + } else { + for (const auto& p : nonZeroRowIndices){ + _insert_cell(p.second, p.first, column_.end()); + } + } +} + +template +template +inline Intrusive_list_column::Intrusive_list_column( + index columnIndex, const Container_type &nonZeroRowIndices, dimension_type dimension, Row_container_type &rowContainer) + : ra_opt(columnIndex, rowContainer), + dim_opt(dimension), + chain_opt([&]{ + if constexpr (Master_matrix::Option_list::is_z2){ + return nonZeroRowIndices.begin() == nonZeroRowIndices.end() ? -1 : *std::prev(nonZeroRowIndices.end()); + } else { + return nonZeroRowIndices.begin() == nonZeroRowIndices.end() ? -1 : std::prev(nonZeroRowIndices.end())->first; + } + }()) +{ + if constexpr (Master_matrix::Option_list::is_z2){ + for (index id : nonZeroRowIndices){ + _insert_cell(id, column_.end()); + } + } else { + for (const auto& p : nonZeroRowIndices){ + _insert_cell(p.second, p.first, column_.end()); + } + } +} + +template +inline Intrusive_list_column::Intrusive_list_column(const Intrusive_list_column &column) + : ra_opt(), + dim_opt(static_cast(column)), + chain_opt(static_cast(column)) +{ + static_assert(!Master_matrix::Option_list::has_row_access, + "Simple copy constructor not available when row access option enabled. Please specify the new column index and the row container."); + + column_.clone_from(column.column_, new_cloner(), delete_disposer(this)); +} + +template +template +inline Intrusive_list_column::Intrusive_list_column( + const Intrusive_list_column &column, index columnIndex, Row_container_type &rowContainer) + : ra_opt(columnIndex, rowContainer), + dim_opt(static_cast(column)), + chain_opt(static_cast(column)) +{ + for (const Cell& cell : column.column_){ + if constexpr (Master_matrix::Option_list::is_z2){ + _insert_cell(cell.get_row_index(), column_.end()); + } else { + _insert_cell(cell.get_element(), cell.get_row_index(), column_.end()); + } + } +} + +template +inline Intrusive_list_column::Intrusive_list_column(Intrusive_list_column &&column) noexcept + : ra_opt(std::move(static_cast(column))), + dim_opt(std::move(static_cast(column))), + chain_opt(std::move(static_cast(column))), + column_(std::move(column.column_)) +{} + +template +inline Intrusive_list_column::~Intrusive_list_column() +{ + column_.clear_and_dispose(delete_disposer(this)); +} + +template +inline std::vector::Field_element_type> +Intrusive_list_column::get_content(int columnLength) const +{ + if (columnLength < 0) columnLength = column_.back().get_row_index() + 1; + + std::vector container(columnLength); + for (auto it = column_.begin(); it != column_.end() && it->get_row_index() < static_cast(columnLength); ++it){ + if constexpr (Master_matrix::Option_list::is_z2){ + container[it->get_row_index()] = 1; + } else { + container[it->get_row_index()] = it->get_element(); + } + } + return container; +} + +template +inline bool Intrusive_list_column::is_non_zero(index rowIndex) const +{ + for (const Cell& cell : column_) + if (cell.get_row_index() == rowIndex) return true; + + return false; +} + +template +inline bool Intrusive_list_column::is_empty() const +{ + return column_.empty(); +} + +template +template +inline void Intrusive_list_column::reorder(const Map_type &valueMap) +{ + static_assert(!Master_matrix::isNonBasic || Master_matrix::Option_list::is_of_boundary_type, + "Method not available for chain columns."); + + if constexpr (Master_matrix::Option_list::has_intrusive_rows){ + for (auto it = column_.begin(); it != column_.end(); ++it) { + Cell* cell = &(*it); + if constexpr (Master_matrix::Option_list::has_row_access) ra_opt::unlink(cell); + cell->set_row_index(valueMap.at(cell->get_row_index())); + if constexpr (Master_matrix::Option_list::has_row_access) ra_opt::insert_cell(cell->get_row_index(), cell); + } + } else { + for (auto it = column_.begin(); it != column_.end(); ++it) { + Cell* cell = &(*it); + if constexpr (Master_matrix::Option_list::has_row_access) ra_opt::unlink(cell); + cell->set_row_index(valueMap.at(cell->get_row_index())); + } + //all cells have to be deleted first, to avoid problem with insertion when row is a set + if constexpr (Master_matrix::Option_list::has_row_access){ + for (auto it = column_.begin(); it != column_.end(); ++it) { + Cell* cell = &(*it); + ra_opt::insert_cell(cell->get_row_index(), cell); + } + } + } + + column_.sort(); +} + +template +inline void Intrusive_list_column::clear() +{ + static_assert(!Master_matrix::isNonBasic || Master_matrix::Option_list::is_of_boundary_type, + "Method not available for chain columns as a base element should not be empty."); + + column_.clear_and_dispose(delete_disposer(this)); +} + +template +inline void Intrusive_list_column::clear(index rowIndex) +{ + static_assert(!Master_matrix::isNonBasic || Master_matrix::Option_list::is_of_boundary_type, + "Method not available for chain columns."); + + auto it = column_.begin(); + while (it != column_.end() && it->get_row_index() != rowIndex) it++; + if (it != column_.end()) _delete_cell(it); +} + +template +inline int Intrusive_list_column::get_pivot() const +{ + static_assert(Master_matrix::isNonBasic, "Method not available for base columns."); //could technically be, but is the notion usefull then? + + if constexpr (Master_matrix::Option_list::is_of_boundary_type){ + if (column_.empty()) return -1; + return column_.back().get_row_index(); + } else { + return chain_opt::get_pivot(); + } +} + +template +inline typename Intrusive_list_column::Field_element_type Intrusive_list_column::get_pivot_value() const +{ + static_assert(Master_matrix::isNonBasic, "Method not available for base columns."); //could technically be, but is the notion usefull then? + + if constexpr (Master_matrix::Option_list::is_z2){ + return 1; + } else { + if constexpr (Master_matrix::Option_list::is_of_boundary_type){ + if (column_.empty()) return 0; + return column_.back().get_element(); + } else { + if (chain_opt::get_pivot() == -1) return Field_element_type(); + for (const Cell& cell : column_){ + if (cell.get_row_index() == static_cast(chain_opt::get_pivot())) return cell.get_element(); + } + return Field_element_type(); //should never happen if chain column is used properly + } + } +} + +template +inline typename Intrusive_list_column::iterator +Intrusive_list_column::begin() noexcept +{ + return column_.begin(); +} + +template +inline typename Intrusive_list_column::const_iterator +Intrusive_list_column::begin() const noexcept +{ + return column_.begin(); +} + +template +inline typename Intrusive_list_column::iterator +Intrusive_list_column::end() noexcept +{ + return column_.end(); +} + +template +inline typename Intrusive_list_column::const_iterator +Intrusive_list_column::end() const noexcept +{ + return column_.end(); +} + +template +inline typename Intrusive_list_column::reverse_iterator +Intrusive_list_column::rbegin() noexcept +{ + return column_.rbegin(); +} + +template +inline typename Intrusive_list_column::const_reverse_iterator +Intrusive_list_column::rbegin() const noexcept +{ + return column_.rbegin(); +} + +template +inline typename Intrusive_list_column::reverse_iterator +Intrusive_list_column::rend() noexcept +{ + return column_.rend(); +} + +template +inline typename Intrusive_list_column::const_reverse_iterator +Intrusive_list_column::rend() const noexcept +{ + return column_.rend(); +} + +template +template +inline Intrusive_list_column & +Intrusive_list_column::operator+=(const Cell_range &column) +{ + static_assert((!Master_matrix::isNonBasic || std::is_same_v), + "For boundary columns, the range has to be a column of same type to help ensure the validity of the base element."); //could be removed, if we give the responsability to the user. + static_assert((!Master_matrix::isNonBasic || Master_matrix::Option_list::is_of_boundary_type), + "For chain columns, the given column cannot be constant."); + + _add(column); + + return *this; +} + +template +inline Intrusive_list_column & +Intrusive_list_column::operator+=(Intrusive_list_column &column) +{ + if constexpr (Master_matrix::isNonBasic && !Master_matrix::Option_list::is_of_boundary_type){ + //assumes that the addition never zeros out this column. + if (_add(column)) chain_opt::swap_pivots(column); + } else { + _add(column); + } + + return *this; +} + +template +inline Intrusive_list_column & +Intrusive_list_column::operator*=(unsigned int v) +{ + if constexpr (Master_matrix::Option_list::is_z2){ + if (v % 2 == 0){ + if constexpr (Master_matrix::isNonBasic && !Master_matrix::Option_list::is_of_boundary_type){ + throw std::invalid_argument("A chain column should not be multiplied by 0."); + } else { + clear(); + } + } + } else { + // v %= Field_element_type::get_characteristic(); //don't work because of multifields... + Field_element_type val(v); + + if (val == 0u) { + if constexpr (Master_matrix::isNonBasic && !Master_matrix::Option_list::is_of_boundary_type){ + throw std::invalid_argument("A chain column should not be multiplied by 0."); + } else { + clear(); + } + return *this; + } + + if (val == 1u) return *this; + + for (Cell& cell : column_){ + cell.get_element() *= val; + if constexpr (Master_matrix::Option_list::has_row_access) + ra_opt::update_cell(cell); + } + } + + return *this; +} + +template +template +inline Intrusive_list_column & +Intrusive_list_column::multiply_and_add(const Field_element_type& val, const Cell_range& column) +{ + static_assert((!Master_matrix::isNonBasic || std::is_same_v), + "For boundary columns, the range has to be a column of same type to help ensure the validity of the base element."); //could be removed, if we give the responsability to the user. + static_assert((!Master_matrix::isNonBasic || Master_matrix::Option_list::is_of_boundary_type), + "For chain columns, the given column cannot be constant."); + + if constexpr (Master_matrix::Option_list::is_z2){ + if (val){ + _add(column); + } else { + clear(); + _add(column); + } + } else { + _multiply_and_add(val, column); + } + + return *this; +} + +template +inline Intrusive_list_column & +Intrusive_list_column::multiply_and_add(const Field_element_type& val, Intrusive_list_column& column) +{ + if constexpr (Master_matrix::isNonBasic && !Master_matrix::Option_list::is_of_boundary_type){ + //assumes that the addition never zeros out this column. + if constexpr (Master_matrix::Option_list::is_z2){ + if (val){ + if (_add(column)) chain_opt::swap_pivots(column); + } else { + throw std::invalid_argument("A chain column should not be multiplied by 0."); + } + } else { + if (_multiply_and_add(val, column)) chain_opt::swap_pivots(column); + } + } else { + if constexpr (Master_matrix::Option_list::is_z2){ + if (val){ + _add(column); + } else { + clear(); + _add(column); + } + } else { + _multiply_and_add(val, column); + } + } + + return *this; +} + +template +template +inline Intrusive_list_column & +Intrusive_list_column::multiply_and_add(const Cell_range& column, const Field_element_type& val) +{ + static_assert((!Master_matrix::isNonBasic || std::is_same_v), + "For boundary columns, the range has to be a column of same type to help ensure the validity of the base element."); //could be removed, if we give the responsability to the user. + static_assert((!Master_matrix::isNonBasic || Master_matrix::Option_list::is_of_boundary_type), + "For chain columns, the given column cannot be constant."); + + if constexpr (Master_matrix::Option_list::is_z2){ + if (val){ + _add(column); + } + } else { + _multiply_and_add(column, val); + } + + return *this; +} + +template +inline Intrusive_list_column & +Intrusive_list_column::multiply_and_add(Intrusive_list_column& column, const Field_element_type& val) +{ + if constexpr (Master_matrix::isNonBasic && !Master_matrix::Option_list::is_of_boundary_type){ + //assumes that the addition never zeros out this column. + if constexpr (Master_matrix::Option_list::is_z2){ + if (val){ + if (_add(column)) chain_opt::swap_pivots(column); + } + } else { + if (_multiply_and_add(column, val)) chain_opt::swap_pivots(column); + } + } else { + if constexpr (Master_matrix::Option_list::is_z2){ + if (val){ + _add(column); + } + } else { + _multiply_and_add(column, val); + } + } + + return *this; +} + +template +inline Intrusive_list_column & +Intrusive_list_column::operator=(const Intrusive_list_column& other) +{ + static_assert(!Master_matrix::Option_list::has_row_access, "= assignement not enabled with row access option."); + + dim_opt::operator=(other); + chain_opt::operator=(other); + column_.clone_from(other.column_, new_cloner(), delete_disposer(this)); + return *this; +} + +template +inline void Intrusive_list_column::_delete_cell(iterator &it) +{ + it = column_.erase_and_dispose(it, delete_disposer(this)); +} + +template +inline void Intrusive_list_column::_insert_cell( + const Field_element_type &value, index rowIndex, const iterator &position) +{ + if constexpr (Master_matrix::Option_list::has_row_access){ + Cell *new_cell = cellPool_.construct(value, ra_opt::columnIndex_, rowIndex); + column_.insert(position, *new_cell); + ra_opt::insert_cell(rowIndex, new_cell); + } else { + Cell *new_cell = cellPool_.construct(value, rowIndex); + column_.insert(position, *new_cell); + } +} + +template +inline void Intrusive_list_column::_insert_cell( + index rowIndex, const iterator &position) +{ + if constexpr (Master_matrix::Option_list::has_row_access){ + Cell *new_cell = cellPool_.construct(ra_opt::columnIndex_, rowIndex); + column_.insert(position, *new_cell); + ra_opt::insert_cell(rowIndex, new_cell); + } else { + Cell *new_cell = cellPool_.construct(rowIndex); + column_.insert(position, *new_cell); + } +} + +template +template +inline bool Intrusive_list_column::_add(const Cell_range &column) +{ + auto itTarget = column_.begin(); + auto itSource = column.begin(); + bool pivotIsZeroed = false; + + while (itTarget != column_.end() && itSource != column.end()) + { + if (itTarget->get_row_index() < itSource->get_row_index()) { + ++itTarget; + } else if (itTarget->get_row_index() > itSource->get_row_index()) { + if constexpr (Master_matrix::Option_list::is_z2){ + _insert_cell(itSource->get_row_index(), itTarget); + } else { + _insert_cell(itSource->get_element(), itSource->get_row_index(), itTarget); + } + ++itSource; + } else { + if constexpr (Master_matrix::Option_list::is_z2){ + if constexpr (Master_matrix::isNonBasic && !Master_matrix::Option_list::is_of_boundary_type){ + if (static_cast(itTarget->get_row_index()) == chain_opt::get_pivot()) pivotIsZeroed = true; + } + _delete_cell(itTarget); + } else { + itTarget->get_element() += itSource->get_element(); + if (itTarget->get_element() == Field_element_type::get_additive_identity()){ + if constexpr (Master_matrix::isNonBasic && !Master_matrix::Option_list::is_of_boundary_type){ + if (static_cast(itTarget->get_row_index()) == chain_opt::get_pivot()) pivotIsZeroed = true; + } + _delete_cell(itTarget); + } else { + if constexpr (Master_matrix::Option_list::has_row_access) + ra_opt::update_cell(*itTarget); + ++itTarget; + } + } + ++itSource; + } + } + + while (itSource != column.end()) { + if constexpr (Master_matrix::Option_list::is_z2){ + _insert_cell(itSource->get_row_index(), column_.end()); + } else { + _insert_cell(itSource->get_element(), itSource->get_row_index(), column_.end()); + } + ++itSource; + } + + return pivotIsZeroed; +} + +template +template +inline bool Intrusive_list_column::_multiply_and_add(const Field_element_type& val, const Cell_range& column) +{ + bool pivotIsZeroed = false; + + if (val == 0u) { + if constexpr (Master_matrix::isNonBasic && !Master_matrix::Option_list::is_of_boundary_type){ + throw std::invalid_argument("A chain column should not be multiplied by 0."); + //this would not only mess up the base, but also the pivots stored. + } else { + clear(); + } + } + + auto itTarget = column_.begin(); + auto itSource = column.begin(); + while (itTarget != column_.end() && itSource != column.end()) + { + if (itTarget->get_row_index() < itSource->get_row_index()) { + itTarget->get_element() *= val; + if constexpr (Master_matrix::Option_list::has_row_access) + ra_opt::update_cell(*itTarget); + ++itTarget; + } else if (itTarget->get_row_index() > itSource->get_row_index()) { + _insert_cell(itSource->get_element(), itSource->get_row_index(), itTarget); + ++itSource; + } else { + itTarget->get_element() *= val; + itTarget->get_element() += itSource->get_element(); + if (itTarget->get_element() == Field_element_type::get_additive_identity()){ + if constexpr (Master_matrix::isNonBasic && !Master_matrix::Option_list::is_of_boundary_type){ + if (static_cast(itTarget->get_row_index()) == chain_opt::get_pivot()) pivotIsZeroed = true; + } + _delete_cell(itTarget); + } else { + if constexpr (Master_matrix::Option_list::has_row_access) + ra_opt::update_cell(*itTarget); + ++itTarget; + } + ++itSource; + } + } + + while (itTarget != column_.end()){ + itTarget->get_element() *= val; + if constexpr (Master_matrix::Option_list::has_row_access) + ra_opt::update_cell(*itTarget); + itTarget++; + } + + while (itSource != column.end()) { + _insert_cell(itSource->get_element(), itSource->get_row_index(), column_.end()); + ++itSource; + } + + return pivotIsZeroed; +} + +template +template +inline bool Intrusive_list_column::_multiply_and_add(const Cell_range& column, const Field_element_type& val) +{ + if (val == 0u) { + return false; + } + + bool pivotIsZeroed = false; + + auto itTarget = column_.begin(); + auto itSource = column.begin(); + while (itTarget != column_.end() && itSource != column.end()) + { + if (itTarget->get_row_index() < itSource->get_row_index()) { + ++itTarget; + } else if (itTarget->get_row_index() > itSource->get_row_index()) { + _insert_cell(itSource->get_element() * val, itSource->get_row_index(), itTarget); + ++itSource; + } else { + itTarget->get_element() += (itSource->get_element() * val); + if (itTarget->get_element() == Field_element_type::get_additive_identity()){ + if constexpr (Master_matrix::isNonBasic && !Master_matrix::Option_list::is_of_boundary_type){ + if (static_cast(itTarget->get_row_index()) == chain_opt::get_pivot()) pivotIsZeroed = true; + } + _delete_cell(itTarget); + } else { + if constexpr (Master_matrix::Option_list::has_row_access) + ra_opt::update_cell(*itTarget); + ++itTarget; + } + ++itSource; + } + } + + while (itSource != column.end()) { + _insert_cell(itSource->get_element() * val, itSource->get_row_index(), column_.end()); + ++itSource; + } + + return pivotIsZeroed; +} + +} //namespace persistence_matrix +} //namespace Gudhi + +template +struct std::hash > +{ + size_t operator()(const Gudhi::persistence_matrix::Intrusive_list_column& column) const + { + std::size_t seed = 0; + for (auto& cell : column){ + seed ^= std::hash()(cell.get_row_index() * static_cast(cell.get_element())) + 0x9e3779b9 + (seed << 6) + (seed >> 2); + } + return seed; + } +}; + +#endif // PM_INTRUSIVE_LIST_COLUMN_H diff --git a/src/Persistence_matrix/include/gudhi/Persistence_matrix/columns/row_access.h b/src/Persistence_matrix/include/gudhi/Persistence_matrix/columns/row_access.h index 1c77fae26f..0f801167a2 100644 --- a/src/Persistence_matrix/include/gudhi/Persistence_matrix/columns/row_access.h +++ b/src/Persistence_matrix/include/gudhi/Persistence_matrix/columns/row_access.h @@ -18,6 +18,15 @@ namespace Gudhi { namespace persistence_matrix { +struct Dummy_row_access{ + friend void swap([[maybe_unused]] Dummy_row_access& d1, [[maybe_unused]] Dummy_row_access& d2){} + + Dummy_row_access(){} + Dummy_row_access([[maybe_unused]] Dummy_row_access&& other) noexcept{} + + static constexpr bool isActive_ = false; //TODO: to remove when columns are redone without +}; + template class Row_access { diff --git a/src/Persistence_matrix/include/gudhi/Persistence_matrix/matrix_dimension_holders.h b/src/Persistence_matrix/include/gudhi/Persistence_matrix/matrix_dimension_holders.h index 0a9b984ac1..b29c030e6d 100644 --- a/src/Persistence_matrix/include/gudhi/Persistence_matrix/matrix_dimension_holders.h +++ b/src/Persistence_matrix/include/gudhi/Persistence_matrix/matrix_dimension_holders.h @@ -17,6 +17,11 @@ namespace Gudhi { namespace persistence_matrix { +struct Dummy_matrix_dimension_holder{ + template + Dummy_matrix_dimension_holder([[maybe_unused]] dimension_type maximalDimension){} +}; + template class Matrix_max_dimension_holder { diff --git a/src/Persistence_matrix/include/gudhi/Persistence_matrix/ru_matrix.h b/src/Persistence_matrix/include/gudhi/Persistence_matrix/ru_matrix.h index 71cd793d6c..9b826a96f1 100644 --- a/src/Persistence_matrix/include/gudhi/Persistence_matrix/ru_matrix.h +++ b/src/Persistence_matrix/include/gudhi/Persistence_matrix/ru_matrix.h @@ -18,9 +18,6 @@ namespace Gudhi { namespace persistence_matrix { -template -class RU_matrix; - template class RU_matrix : public Master_matrix::RU_pairing_option, @@ -101,6 +98,9 @@ class RU_matrix using r_matrix_type = typename Master_matrix::Boundary_matrix_type; using u_matrix_type = typename Master_matrix::Base_matrix_type; + friend rep_opt; //direct access to the two matrices + friend swap_opt; //direct access to the two matrices + r_matrix_type reducedMatrixR_; u_matrix_type mirrorMatrixU_; //make U not accessible by default and add option to enable access? Inaccessible, it needs less options and we could avoid some ifs. dictionnary_type pivotToColumnIndex_; @@ -116,18 +116,18 @@ class RU_matrix template inline RU_matrix::RU_matrix() - : Master_matrix::RU_pairing_option(), - Master_matrix::RU_vine_swap_option(reducedMatrixR_, mirrorMatrixU_, pivotToColumnIndex_), - Master_matrix::RU_representative_cycles_option(reducedMatrixR_, mirrorMatrixU_), + : pair_opt(), + swap_opt(), + rep_opt(), nextInsertIndex_(0) {} template template inline RU_matrix::RU_matrix(const std::vector &orderedBoundaries) - : Master_matrix::RU_pairing_option(), - Master_matrix::RU_vine_swap_option(reducedMatrixR_, mirrorMatrixU_, pivotToColumnIndex_), - Master_matrix::RU_representative_cycles_option(reducedMatrixR_, mirrorMatrixU_), + : pair_opt(), + swap_opt(), + rep_opt(), reducedMatrixR_(orderedBoundaries), mirrorMatrixU_(orderedBoundaries.size()), nextInsertIndex_(orderedBoundaries.size()) @@ -144,9 +144,9 @@ inline RU_matrix::RU_matrix(const std::vector &ord template inline RU_matrix::RU_matrix(unsigned int numberOfColumns) - : Master_matrix::RU_pairing_option(), - Master_matrix::RU_vine_swap_option(reducedMatrixR_, mirrorMatrixU_, pivotToColumnIndex_), - Master_matrix::RU_representative_cycles_option(reducedMatrixR_, mirrorMatrixU_), + : pair_opt(), + swap_opt(), + rep_opt(), reducedMatrixR_(numberOfColumns), mirrorMatrixU_(numberOfColumns), nextInsertIndex_(0) @@ -160,45 +160,25 @@ inline RU_matrix::RU_matrix(unsigned int numberOfColumns) template inline RU_matrix::RU_matrix(const RU_matrix &matrixToCopy) - : Master_matrix::RU_pairing_option(matrixToCopy), - Master_matrix::RU_vine_swap_option(matrixToCopy), - Master_matrix::RU_representative_cycles_option(matrixToCopy), + : pair_opt(static_cast(matrixToCopy)), + swap_opt(static_cast(matrixToCopy)), + rep_opt(static_cast(matrixToCopy)), reducedMatrixR_(matrixToCopy.reducedMatrixR_), mirrorMatrixU_(matrixToCopy.mirrorMatrixU_), pivotToColumnIndex_(matrixToCopy.pivotToColumnIndex_), nextInsertIndex_(matrixToCopy.nextInsertIndex_) -{ - if constexpr (Master_matrix::Option_list::can_retrieve_representative_cycles){ - rep_opt::reducedMatrixR_ = &reducedMatrixR_; - rep_opt::mirrorMatrixU_ = &mirrorMatrixU_; - } - if constexpr (Master_matrix::Option_list::has_vine_update){ - swap_opt::reducedMatrixR_ = &reducedMatrixR_; - swap_opt::mirrorMatrixU_ = &mirrorMatrixU_; - swap_opt::pivotToColumnIndex_ = &pivotToColumnIndex_; - } -} +{} template inline RU_matrix::RU_matrix(RU_matrix &&other) noexcept - : Master_matrix::RU_pairing_option(std::move(other)), - Master_matrix::RU_vine_swap_option(std::move(other)), - Master_matrix::RU_representative_cycles_option(std::move(other)), + : pair_opt(std::move(static_cast(other))), + swap_opt(std::move(static_cast(other))), + rep_opt(std::move(static_cast(other))), reducedMatrixR_(std::move(other.reducedMatrixR_)), mirrorMatrixU_(std::move(other.mirrorMatrixU_)), pivotToColumnIndex_(std::move(other.pivotToColumnIndex_)), nextInsertIndex_(std::exchange(other.nextInsertIndex_, 0)) -{ - if constexpr (Master_matrix::Option_list::can_retrieve_representative_cycles){ - rep_opt::reducedMatrixR_ = &reducedMatrixR_; - rep_opt::mirrorMatrixU_ = &mirrorMatrixU_; - } - if constexpr (Master_matrix::Option_list::has_vine_update){ - swap_opt::reducedMatrixR_ = &reducedMatrixR_; - swap_opt::mirrorMatrixU_ = &mirrorMatrixU_; - swap_opt::pivotToColumnIndex_ = &pivotToColumnIndex_; - } -} +{} template template diff --git a/src/Persistence_matrix/include/gudhi/Persistence_matrix/ru_pairing.h b/src/Persistence_matrix/include/gudhi/Persistence_matrix/ru_pairing.h index efbdba18d7..05e30a9a9b 100644 --- a/src/Persistence_matrix/include/gudhi/Persistence_matrix/ru_pairing.h +++ b/src/Persistence_matrix/include/gudhi/Persistence_matrix/ru_pairing.h @@ -16,6 +16,12 @@ namespace Gudhi { namespace persistence_matrix { +struct Dummy_ru_pairing{ + friend void swap([[maybe_unused]] Dummy_ru_pairing& d1, [[maybe_unused]] Dummy_ru_pairing& d2){} + + Dummy_ru_pairing(){} +}; + template class RU_pairing { diff --git a/src/Persistence_matrix/include/gudhi/Persistence_matrix/ru_rep_cycles.h b/src/Persistence_matrix/include/gudhi/Persistence_matrix/ru_rep_cycles.h index 9bce50e86c..c8a633dacb 100644 --- a/src/Persistence_matrix/include/gudhi/Persistence_matrix/ru_rep_cycles.h +++ b/src/Persistence_matrix/include/gudhi/Persistence_matrix/ru_rep_cycles.h @@ -15,11 +15,17 @@ #include //std::sort #include -#include "../options.h" //Column_types - namespace Gudhi { namespace persistence_matrix { +struct Dummy_ru_representative_cycles{ + friend void swap([[maybe_unused]] Dummy_ru_representative_cycles& d1, [[maybe_unused]] Dummy_ru_representative_cycles& d2){} + + Dummy_ru_representative_cycles(){} + Dummy_ru_representative_cycles([[maybe_unused]] const Dummy_ru_representative_cycles& matrixToCopy){} + Dummy_ru_representative_cycles([[maybe_unused]] Dummy_ru_representative_cycles&& other) noexcept{} +}; + template class RU_representative_cycles { @@ -30,7 +36,7 @@ class RU_representative_cycles using Base_matrix = typename Master_matrix::Base_matrix_type; using cycle_type = std::vector; //TODO: add coefficients - RU_representative_cycles(Boundary_matrix &matrixR, Base_matrix &matrixU); + RU_representative_cycles(); RU_representative_cycles(const RU_representative_cycles& matrixToCopy); RU_representative_cycles(RU_representative_cycles&& other) noexcept; @@ -45,33 +51,29 @@ class RU_representative_cycles base1.birthToCycle_.swap(base2.birthToCycle_); } -protected: - Boundary_matrix *reducedMatrixR_; - Base_matrix *mirrorMatrixU_; - private: + using ru_matrix = typename Master_matrix::RU_matrix_type; + std::vector representativeCycles_; std::vector birthToCycle_; + + constexpr ru_matrix* _matrix() { return static_cast(this); } + constexpr const ru_matrix* _matrix() const { return static_cast(this); } }; template -inline RU_representative_cycles::RU_representative_cycles(Boundary_matrix &matrixR, Base_matrix &matrixU) - : reducedMatrixR_(&matrixR), mirrorMatrixU_(&matrixU) +inline RU_representative_cycles::RU_representative_cycles() {} template inline RU_representative_cycles::RU_representative_cycles(const RU_representative_cycles& matrixToCopy) - : reducedMatrixR_(matrixToCopy.reducedMatrixR_), - mirrorMatrixU_(matrixToCopy.mirrorMatrixU_), - representativeCycles_(matrixToCopy.representativeCycles_), + : representativeCycles_(matrixToCopy.representativeCycles_), birthToCycle_(matrixToCopy.birthToCycle_) {} template inline RU_representative_cycles::RU_representative_cycles(RU_representative_cycles &&other) noexcept - : reducedMatrixR_(other.reducedMatrixR_), - mirrorMatrixU_(other.mirrorMatrixU_), - representativeCycles_(std::move(other.representativeCycles_)), + : representativeCycles_(std::move(other.representativeCycles_)), birthToCycle_(std::move(other.birthToCycle_)) {} @@ -79,14 +81,15 @@ template inline void RU_representative_cycles::update_representative_cycles() { birthToCycle_.clear(); - birthToCycle_.resize(reducedMatrixR_->get_number_of_columns(), -1); - for (unsigned int i = 0; i < reducedMatrixR_->get_number_of_columns(); i++){ - if (reducedMatrixR_->is_zero_column(i)){ + birthToCycle_.resize(_matrix()->reducedMatrixR_.get_number_of_columns(), -1); + for (unsigned int i = 0; i < _matrix()->reducedMatrixR_.get_number_of_columns(); i++){ + if (_matrix()->reducedMatrixR_.is_zero_column(i)){ representativeCycles_.push_back(cycle_type()); - for (const auto& cell : mirrorMatrixU_->get_column(i)){ + for (const auto& cell : _matrix()->mirrorMatrixU_.get_column(i)){ representativeCycles_.back().push_back(cell.get_row_index()); } - if constexpr (Master_matrix::Option_list::column_type == Column_types::HEAP || Master_matrix::Option_list::column_type == Column_types::UNORDERED_SET) + if constexpr (std::is_same_v + || std::is_same_v) std::sort(representativeCycles_.back().begin(), representativeCycles_.back().end()); birthToCycle_[i] = representativeCycles_.size() - 1; } diff --git a/src/Persistence_matrix/include/gudhi/Persistence_matrix/ru_vine_swap.h b/src/Persistence_matrix/include/gudhi/Persistence_matrix/ru_vine_swap.h index e540493ffe..728b8e1839 100644 --- a/src/Persistence_matrix/include/gudhi/Persistence_matrix/ru_vine_swap.h +++ b/src/Persistence_matrix/include/gudhi/Persistence_matrix/ru_vine_swap.h @@ -12,6 +12,7 @@ #define PM_RU_VINE_SWAP_H #include //std::move +#include //std::conditional #include #include "ru_pairing.h" @@ -19,11 +20,19 @@ namespace Gudhi { namespace persistence_matrix { +struct Dummy_ru_vine_swap{ + friend void swap([[maybe_unused]] Dummy_ru_vine_swap& d1, [[maybe_unused]] Dummy_ru_vine_swap& d2){} + + Dummy_ru_vine_swap(){} + Dummy_ru_vine_swap([[maybe_unused]] const Dummy_ru_vine_swap& matrixToCopy){} + Dummy_ru_vine_swap([[maybe_unused]] Dummy_ru_vine_swap&& other) noexcept{} +}; + template class RU_vine_swap : public std::conditional< Master_matrix::Option_list::has_column_pairings, RU_pairing, - typename Master_matrix::Dummy_ru_pairing + Dummy_ru_pairing >::type { public: @@ -32,7 +41,7 @@ class RU_vine_swap : public std::conditional< using dictionnary_type = typename Master_matrix::template dictionnary_type; using index = typename Master_matrix::index; - RU_vine_swap(Boundary_matrix &matrixR, Base_matrix &matrixU, dictionnary_type &pivotToColumn); + RU_vine_swap(); RU_vine_swap(const RU_vine_swap &matrixToCopy); RU_vine_swap(RU_vine_swap&& other) noexcept; @@ -46,17 +55,13 @@ class RU_vine_swap : public std::conditional< } } -protected: - Boundary_matrix *reducedMatrixR_; - Base_matrix *mirrorMatrixU_; - dictionnary_type *pivotToColumnIndex_; - private: using RUP = typename std::conditional< Master_matrix::Option_list::has_column_pairings, RU_pairing, - typename Master_matrix::Dummy_ru_pairing + Dummy_ru_pairing >::type; + using ru_matrix = typename Master_matrix::RU_matrix_type; bool _is_paired(index index); int _get_pair(index index); @@ -76,40 +81,37 @@ class RU_vine_swap : public std::conditional< int& _birth(index simplexIndex); int _death(index simplexIndex) const; int _birth(index simplexIndex) const; + + constexpr ru_matrix* _matrix() { return static_cast(this); } + constexpr const ru_matrix* _matrix() const { return static_cast(this); } }; template -inline RU_vine_swap::RU_vine_swap(Boundary_matrix &matrixR, Base_matrix &matrixU, dictionnary_type &pivotToColumn) - : RUP(), reducedMatrixR_(&matrixR), mirrorMatrixU_(&matrixU), pivotToColumnIndex_(&pivotToColumn) +inline RU_vine_swap::RU_vine_swap() + : RUP() {} template inline RU_vine_swap::RU_vine_swap( const RU_vine_swap &matrixToCopy) - : RUP(matrixToCopy), - reducedMatrixR_(matrixToCopy.reducedMatrixR_), - mirrorMatrixU_(matrixToCopy.mirrorMatrixU_), - pivotToColumnIndex_(matrixToCopy.pivotToColumnIndex_) + : RUP(static_cast(matrixToCopy)) {} template inline RU_vine_swap::RU_vine_swap(RU_vine_swap &&other) noexcept - : RUP(std::move(other)), - reducedMatrixR_(other.reducedMatrixR_), - mirrorMatrixU_(other.mirrorMatrixU_), - pivotToColumnIndex_(other.pivotToColumnIndex_) + : RUP(std::move(static_cast(other))) {} template inline bool RU_vine_swap::vine_swap_with_z_eq_1_case(index index) { - assert(index < reducedMatrixR_->get_number_of_columns() - 1 && "Index to swap out of bound."); + assert(index < _matrix()->reducedMatrixR_.get_number_of_columns() - 1 && "Index to swap out of bound."); - bool iIsPositive = reducedMatrixR_->is_zero_column(index); - bool iiIsPositive = reducedMatrixR_->is_zero_column(index + 1); + bool iIsPositive = _matrix()->reducedMatrixR_.is_zero_column(index); + bool iiIsPositive = _matrix()->reducedMatrixR_.is_zero_column(index + 1); if (iIsPositive && iiIsPositive) { - mirrorMatrixU_->zero_cell(index + 1, index); + _matrix()->mirrorMatrixU_.zero_cell(index + 1, index); return _positive_vine_swap(index); } else if (!iIsPositive && !iiIsPositive) return _negative_vine_swap(index); @@ -122,37 +124,37 @@ inline bool RU_vine_swap::vine_swap_with_z_eq_1_case(index index) template inline bool RU_vine_swap::vine_swap(index index) { - assert(index < reducedMatrixR_->get_number_of_columns() - 1 && "Index to swap out of bound."); + assert(index < _matrix()->reducedMatrixR_.get_number_of_columns() - 1 && "Index to swap out of bound."); - bool iIsPositive = reducedMatrixR_->is_zero_column(index); - bool iiIsPositive = reducedMatrixR_->is_zero_column(index + 1); + bool iIsPositive = _matrix()->reducedMatrixR_.is_zero_column(index); + bool iiIsPositive = _matrix()->reducedMatrixR_.is_zero_column(index + 1); if (iIsPositive && iiIsPositive) { - if (reducedMatrixR_->get_column_dimension(index) != reducedMatrixR_->get_column_dimension(index + 1)){ + if (_matrix()->reducedMatrixR_.get_column_dimension(index) != _matrix()->reducedMatrixR_.get_column_dimension(index + 1)){ _swap_at_index(index); _positive_transpose(index); return true; } - if (!mirrorMatrixU_->is_zero_cell(index + 1, index)){ - mirrorMatrixU_->zero_cell(index + 1, index); + if (!_matrix()->mirrorMatrixU_.is_zero_cell(index + 1, index)){ + _matrix()->mirrorMatrixU_.zero_cell(index + 1, index); } return _positive_vine_swap(index); } else if (!iIsPositive && !iiIsPositive) { - if (reducedMatrixR_->get_column_dimension(index) != reducedMatrixR_->get_column_dimension(index + 1) || mirrorMatrixU_->is_zero_cell(index + 1, index)){ + if (_matrix()->reducedMatrixR_.get_column_dimension(index) != _matrix()->reducedMatrixR_.get_column_dimension(index + 1) || _matrix()->mirrorMatrixU_.is_zero_cell(index + 1, index)){ _swap_at_index(index); _negative_transpose(index); return true; } return _negative_vine_swap(index); } else if (iIsPositive && !iiIsPositive) { - if (reducedMatrixR_->get_column_dimension(index) != reducedMatrixR_->get_column_dimension(index + 1) || mirrorMatrixU_->is_zero_cell(index + 1, index)){ + if (_matrix()->reducedMatrixR_.get_column_dimension(index) != _matrix()->reducedMatrixR_.get_column_dimension(index + 1) || _matrix()->mirrorMatrixU_.is_zero_cell(index + 1, index)){ _swap_at_index(index); _positive_negative_transpose(index); return true; } return _positive_negative_vine_swap(index); } else { - if (reducedMatrixR_->get_column_dimension(index) != reducedMatrixR_->get_column_dimension(index + 1) || mirrorMatrixU_->is_zero_cell(index + 1, index)){ + if (_matrix()->reducedMatrixR_.get_column_dimension(index) != _matrix()->reducedMatrixR_.get_column_dimension(index + 1) || _matrix()->mirrorMatrixU_.is_zero_cell(index + 1, index)){ _swap_at_index(index); _negative_positive_transpose(index); return true; @@ -174,12 +176,12 @@ inline bool RU_vine_swap::_is_paired(index index){ if constexpr (Master_matrix::Option_list::has_column_pairings){ return _death(index) != -1; } else { - if (!reducedMatrixR_->is_zero_column(index)) return true; + if (!_matrix()->reducedMatrixR_.is_zero_column(index)) return true; if constexpr (Master_matrix::Option_list::has_removable_columns){ - if (pivotToColumnIndex_->find(index) == pivotToColumnIndex_->end()) return false; + if (_matrix()->pivotToColumnIndex_.find(index) == _matrix()->pivotToColumnIndex_.end()) return false; } else { - if (pivotToColumnIndex_->operator[](index) == -1) return false; + if (_matrix()->pivotToColumnIndex_.operator[](index) == -1) return false; } return true; @@ -188,29 +190,29 @@ inline bool RU_vine_swap::_is_paired(index index){ template inline int RU_vine_swap::_get_pair(index index) { - if (!reducedMatrixR_->is_zero_column(index)) - return reducedMatrixR_->get_column(index).get_pivot(); + if (!_matrix()->reducedMatrixR_.is_zero_column(index)) + return _matrix()->reducedMatrixR_.get_column(index).get_pivot(); if constexpr (Master_matrix::Option_list::has_removable_columns) { - auto it = pivotToColumnIndex_->find(index); - if (it == pivotToColumnIndex_->end()) return -1; + auto it = _matrix()->pivotToColumnIndex_.find(index); + if (it == _matrix()->pivotToColumnIndex_.end()) return -1; return *it; } else { - return pivotToColumnIndex_->operator[](index); + return _matrix()->pivotToColumnIndex_.operator[](index); } } template inline void RU_vine_swap::_swap_at_index(index index){ - reducedMatrixR_->swap_at_indices(index, index + 1); - mirrorMatrixU_->swap_at_indices(index + 1, index); + _matrix()->reducedMatrixR_.swap_at_indices(index, index + 1); + _matrix()->mirrorMatrixU_.swap_at_indices(index + 1, index); } template inline void RU_vine_swap::_add_to(index sourceIndex, index targetIndex) { - reducedMatrixR_->add_to(sourceIndex, targetIndex); - mirrorMatrixU_->add_to(sourceIndex, targetIndex); + _matrix()->reducedMatrixR_.add_to(sourceIndex, targetIndex); + _matrix()->mirrorMatrixU_.add_to(sourceIndex, targetIndex); } template @@ -218,16 +220,16 @@ inline void RU_vine_swap::_positive_transpose(index index) { if constexpr (Master_matrix::Option_list::has_removable_columns){ if (_is_paired(index) && _is_paired(index + 1)){ - std::swap(pivotToColumnIndex_->at(index), pivotToColumnIndex_->at(index + 1)); + std::swap(_matrix()->pivotToColumnIndex_.at(index), _matrix()->pivotToColumnIndex_.at(index + 1)); } else if (_is_paired(index)){ - pivotToColumnIndex_->emplace(index + 1, pivotToColumnIndex_->at(index)); - pivotToColumnIndex_->erase(index); + _matrix()->pivotToColumnIndex_.emplace(index + 1, _matrix()->pivotToColumnIndex_.at(index)); + _matrix()->pivotToColumnIndex_.erase(index); } else if (_is_paired(index + 1)){ - pivotToColumnIndex_->emplace(index, pivotToColumnIndex_->at(index + 1)); - pivotToColumnIndex_->erase(index + 1); + _matrix()->pivotToColumnIndex_.emplace(index, _matrix()->pivotToColumnIndex_.at(index + 1)); + _matrix()->pivotToColumnIndex_.erase(index + 1); } } else { - std::swap(pivotToColumnIndex_->operator[](index), pivotToColumnIndex_->operator[](index + 1)); + std::swap(_matrix()->pivotToColumnIndex_.operator[](index), _matrix()->pivotToColumnIndex_.operator[](index + 1)); } if constexpr (Master_matrix::Option_list::has_column_pairings){ @@ -245,21 +247,21 @@ inline void RU_vine_swap::_negative_transpose(index index) _death(index + 1) = index; std::swap(RUP::indexToBar_.at(index), RUP::indexToBar_.at(index + 1)); } - std::swap(pivotToColumnIndex_->at(_birth(index)), pivotToColumnIndex_->at(_birth(index + 1))); + std::swap(_matrix()->pivotToColumnIndex_.at(_birth(index)), _matrix()->pivotToColumnIndex_.at(_birth(index + 1))); } template inline void RU_vine_swap::_positive_negative_transpose(index index) { - pivotToColumnIndex_->at(_birth(index + 1)) = index; + _matrix()->pivotToColumnIndex_.at(_birth(index + 1)) = index; if constexpr (Master_matrix::Option_list::has_removable_columns){ if (_is_paired(index)){ - pivotToColumnIndex_->emplace(index + 1, pivotToColumnIndex_->at(index)); - pivotToColumnIndex_->erase(index); + _matrix()->pivotToColumnIndex_.emplace(index + 1, _matrix()->pivotToColumnIndex_.at(index)); + _matrix()->pivotToColumnIndex_.erase(index); } } else { - pivotToColumnIndex_->operator[](index + 1) = pivotToColumnIndex_->operator[](index); - pivotToColumnIndex_->operator[](index) = -1; + _matrix()->pivotToColumnIndex_.operator[](index + 1) = _matrix()->pivotToColumnIndex_.operator[](index); + _matrix()->pivotToColumnIndex_.operator[](index) = -1; } if constexpr (Master_matrix::Option_list::has_column_pairings){ @@ -272,15 +274,15 @@ inline void RU_vine_swap::_positive_negative_transpose(index inde template inline void RU_vine_swap::_negative_positive_transpose(index index) { - pivotToColumnIndex_->at(_birth(index)) = index + 1; + _matrix()->pivotToColumnIndex_.at(_birth(index)) = index + 1; if constexpr (Master_matrix::Option_list::has_removable_columns){ if (_is_paired(index + 1)){ - pivotToColumnIndex_->emplace(index, pivotToColumnIndex_->at(index + 1)); - pivotToColumnIndex_->erase(index + 1); + _matrix()->pivotToColumnIndex_.emplace(index, _matrix()->pivotToColumnIndex_.at(index + 1)); + _matrix()->pivotToColumnIndex_.erase(index + 1); } } else { - pivotToColumnIndex_->operator[](index) = pivotToColumnIndex_->operator[](index + 1); - pivotToColumnIndex_->operator[](index + 1) = -1; + _matrix()->pivotToColumnIndex_.operator[](index) = _matrix()->pivotToColumnIndex_.operator[](index + 1); + _matrix()->pivotToColumnIndex_.operator[](index + 1) = -1; } if constexpr (Master_matrix::Option_list::has_column_pairings){ @@ -296,7 +298,7 @@ inline bool RU_vine_swap::_positive_vine_swap(index index) int iDeath = _death(index); int iiDeath = _death(index + 1); - if (iDeath != -1 && iiDeath != -1 && !(reducedMatrixR_->is_zero_cell(iiDeath, index))) + if (iDeath != -1 && iiDeath != -1 && !(_matrix()->reducedMatrixR_.is_zero_cell(iiDeath, index))) { if (iDeath < iiDeath) { _swap_at_index(index); @@ -312,7 +314,7 @@ inline bool RU_vine_swap::_positive_vine_swap(index index) _swap_at_index(index); - if (iDeath != -1 || iiDeath == -1 || reducedMatrixR_->is_zero_cell(iiDeath, index + 1)) { + if (iDeath != -1 || iiDeath == -1 || _matrix()->reducedMatrixR_.is_zero_cell(iiDeath, index + 1)) { _positive_transpose(index); return true; } @@ -340,7 +342,7 @@ inline bool RU_vine_swap::_negative_vine_swap(index index) template inline bool RU_vine_swap::_positive_negative_vine_swap(index index) { - mirrorMatrixU_->zero_cell(index + 1, index); + _matrix()->mirrorMatrixU_.zero_cell(index + 1, index); _swap_at_index(index); _positive_negative_transpose(index); @@ -392,15 +394,15 @@ inline int RU_vine_swap::_death(index simplexIndex) const return RUP::barcode_.at(RUP::indexToBar_.at(simplexIndex)).death; } } else { - if (!reducedMatrixR_->is_zero_column(simplexIndex)) - return reducedMatrixR_->get_column(simplexIndex).get_pivot(); + if (!_matrix()->reducedMatrixR_.is_zero_column(simplexIndex)) + return _matrix()->reducedMatrixR_.get_column(simplexIndex).get_pivot(); if constexpr (Master_matrix::Option_list::has_removable_columns) { - auto it = pivotToColumnIndex_->find(simplexIndex); - if (it == pivotToColumnIndex_->end()) return -1; + auto it = _matrix()->pivotToColumnIndex_.find(simplexIndex); + if (it == _matrix()->pivotToColumnIndex_.end()) return -1; return *it; } else { - return pivotToColumnIndex_->operator[](simplexIndex); + return _matrix()->pivotToColumnIndex_.operator[](simplexIndex); } } } @@ -415,7 +417,7 @@ inline int RU_vine_swap::_birth(index simplexIndex) const return RUP::barcode_.at(RUP::indexToBar_.at(simplexIndex)).birth; } } else { - return reducedMatrixR_->get_column(simplexIndex).get_pivot(); + return _matrix()->reducedMatrixR_.get_column(simplexIndex).get_pivot(); } } diff --git a/src/Persistence_matrix/include/gudhi/matrix.h b/src/Persistence_matrix/include/gudhi/matrix.h index fe6306c3de..ff2cc72a23 100644 --- a/src/Persistence_matrix/include/gudhi/matrix.h +++ b/src/Persistence_matrix/include/gudhi/matrix.h @@ -75,6 +75,9 @@ // #include "chain_matrix/chain_matrix_0011.h" #include "Persistence_matrix/chain_matrix.h" +#include "Persistence_matrix/columns/column_dimension_holder.h" +#include "Persistence_matrix/columns/chain_column_extra_properties.h" +#include "Persistence_matrix/columns/intrusive_list_column.h" #include "gudhi/column_types/list_column.h" #include "gudhi/column_types/set_column.h" #include "gudhi/column_types/unordered_set_column.h" @@ -84,9 +87,9 @@ #include "gudhi/column_types/z2_set_column.h" #include "gudhi/column_types/z2_unordered_set_column.h" #include "gudhi/column_types/z2_vector_column.h" -#include "gudhi/column_types/intrusive_list_column.h" +// #include "gudhi/column_types/intrusive_list_column.h" #include "gudhi/column_types/intrusive_set_column.h" -#include "gudhi/column_types/z2_intrusive_list_column.h" +// #include "gudhi/column_types/z2_intrusive_list_column.h" #include "gudhi/column_types/z2_intrusive_set_column.h" #include "gudhi/column_types/boundary_columns/boundary_list_column.h" #include "gudhi/column_types/boundary_columns/boundary_set_column.h" @@ -97,9 +100,9 @@ #include "gudhi/column_types/boundary_columns/z2_boundary_set_column.h" #include "gudhi/column_types/boundary_columns/z2_boundary_unordered_set_column.h" #include "gudhi/column_types/boundary_columns/z2_boundary_vector_column.h" -#include "gudhi/column_types/boundary_columns/boundary_intrusive_list_column.h" +// #include "gudhi/column_types/boundary_columns/boundary_intrusive_list_column.h" #include "gudhi/column_types/boundary_columns/boundary_intrusive_set_column.h" -#include "gudhi/column_types/boundary_columns/z2_boundary_intrusive_list_column.h" +// #include "gudhi/column_types/boundary_columns/z2_boundary_intrusive_list_column.h" #include "gudhi/column_types/boundary_columns/z2_boundary_intrusive_set_column.h" #include "gudhi/column_types/chain_columns/chain_list_column.h" #include "gudhi/column_types/chain_columns/chain_set_column.h" @@ -110,9 +113,9 @@ #include "gudhi/column_types/chain_columns/z2_chain_set_column.h" #include "gudhi/column_types/chain_columns/z2_chain_unordered_set_column.h" #include "gudhi/column_types/chain_columns/z2_chain_vector_column.h" -#include "gudhi/column_types/chain_columns/chain_intrusive_list_column.h" +// #include "gudhi/column_types/chain_columns/chain_intrusive_list_column.h" #include "gudhi/column_types/chain_columns/chain_intrusive_set_column.h" -#include "gudhi/column_types/chain_columns/z2_chain_intrusive_list_column.h" +// #include "gudhi/column_types/chain_columns/z2_chain_intrusive_list_column.h" #include "gudhi/column_types/chain_columns/z2_chain_intrusive_set_column.h" namespace Gudhi { @@ -127,6 +130,7 @@ class Matrix using index = typename Options::index_type; using dimension_type = typename Options::dimension_type; + //TODO: move outside struct Bar{ Bar() : dim(-1), birth(-1), death(-1) {} @@ -173,15 +177,6 @@ class Matrix >::type >::type; - struct Dummy_cell_column_index_mixin{ - Dummy_cell_column_index_mixin(){} - Dummy_cell_column_index_mixin([[maybe_unused]] index columnIndex){} - }; - struct Dummy_cell_field_element_mixin{ - Dummy_cell_field_element_mixin(){} - Dummy_cell_field_element_mixin([[maybe_unused]] Field_type t){} - }; - using Cell_column_index_option = typename std::conditional< Options::has_row_access, Cell_column_index, @@ -228,20 +223,6 @@ class Matrix std::vector >::type; - struct Dummy_row_access{ - friend void swap([[maybe_unused]] Dummy_row_access& d1, [[maybe_unused]] Dummy_row_access& d2){} - - Dummy_row_access(){} - Dummy_row_access([[maybe_unused]] Dummy_row_access&& other) noexcept{} - - static constexpr bool isActive_ = false; - }; - - struct Dummy_matrix_row_access{ - Dummy_matrix_row_access(){}; - Dummy_matrix_row_access([[maybe_unused]] unsigned int numberOfColumns){}; - }; - using Row_access_option = typename std::conditional< Options::has_row_access, Row_access >, @@ -263,6 +244,18 @@ class Matrix static const bool isNonBasic = Options::has_column_pairings || Options::has_vine_update || Options::can_retrieve_representative_cycles; + using Column_dimension_option = typename std::conditional< + isNonBasic, + Column_dimension_holder >, + Dummy_dimension_holder + >::type; + + using Chain_column_option = typename std::conditional< + isNonBasic && !Options::is_of_boundary_type, + Chain_column_extra_properties >, + Dummy_chain_properties + >::type; + using Heap_column_type = typename std::conditional< isNonBasic, typename std::conditional< @@ -361,27 +354,7 @@ class Matrix >::type >::type; - using Intrusive_list_column_type = typename std::conditional< - isNonBasic, - typename std::conditional< - Options::is_of_boundary_type, - typename std::conditional< - Options::is_z2, - Z2_intrusive_list_boundary_column, - Intrusive_list_boundary_column - >::type, - typename std::conditional< - Options::is_z2, - Z2_intrusive_list_chain_column, Cell_type, Row_access_option>, - Intrusive_list_chain_column, Field_type, Cell_type, Row_access_option> - >::type - >::type, - typename std::conditional< - Options::is_z2, - Z2_intrusive_list_column, - Intrusive_list_column - >::type - >::type; + using Intrusive_list_column_type = Intrusive_list_column >; using Intrusive_set_column_type = typename std::conditional< isNonBasic, @@ -454,16 +427,10 @@ class Matrix std::initializer_list > >::type; - struct Dummy_matrix_dimension_holder{ - Dummy_matrix_dimension_holder([[maybe_unused]] dimension_type maximalDimension){}; - - // friend void swap([[maybe_unused]] Dummy_matrix_dimension_holder& d1, [[maybe_unused]] Dummy_matrix_dimension_holder& d2){} - }; - static const bool dimensionIsNeeded = Options::has_column_pairings && !Options::has_vine_update && !Options::can_retrieve_representative_cycles; using Matrix_dimension_option = typename std::conditional< - Options::has_dimension_access || dimensionIsNeeded, + Options::has_matrix_maximal_dimension_access || dimensionIsNeeded, typename std::conditional< Options::has_removable_columns, Matrix_all_dimension_holder, @@ -477,128 +444,48 @@ class Matrix Base_matrix_with_column_compression >, Base_matrix > >::type; - using Boundary_matrix_type = Boundary_matrix >; - using RU_matrix_type = RU_matrix >; - using Chain_matrix_type = Chain_matrix >; - struct Dummy_base_swap{ - Dummy_base_swap& operator=([[maybe_unused]] Dummy_base_swap other){return *this;} - friend void swap([[maybe_unused]] Dummy_base_swap& d1, [[maybe_unused]] Dummy_base_swap& d2){} - - Dummy_base_swap([[maybe_unused]] column_container_type &matrix){} - Dummy_base_swap([[maybe_unused]] column_container_type &matrix, [[maybe_unused]] unsigned int numberOfColumns){} - Dummy_base_swap([[maybe_unused]] const Dummy_base_swap& matrixToCopy){} - Dummy_base_swap([[maybe_unused]] Dummy_base_swap&& other) noexcept{} - }; - + template using Base_swap_option = typename std::conditional< Options::has_vine_update || Options::has_column_and_row_swaps, - Base_swap >, + Base_swap,Base>, Dummy_base_swap >::type; - - struct Dummy_base_pairing{ - Dummy_base_pairing& operator=([[maybe_unused]] Dummy_base_pairing other){return *this;} - friend void swap([[maybe_unused]] Dummy_base_pairing& d1, [[maybe_unused]] Dummy_base_pairing& d2){} - - Dummy_base_pairing([[maybe_unused]] column_container_type& matrix, [[maybe_unused]] dimension_type& maxDim){} - Dummy_base_pairing([[maybe_unused]] const Dummy_base_pairing& matrixToCopy){} - Dummy_base_pairing([[maybe_unused]] Dummy_base_pairing&& other) noexcept{} - }; - using Base_pairing_option = typename std::conditional< Options::has_column_pairings && !Options::has_vine_update && !Options::can_retrieve_representative_cycles, Base_pairing >, Dummy_base_pairing >::type; - struct Dummy_ru_pairing{ - Dummy_ru_pairing& operator=([[maybe_unused]] Dummy_ru_pairing other){return *this;} - friend void swap([[maybe_unused]] Dummy_ru_pairing& d1, [[maybe_unused]] Dummy_ru_pairing& d2){} - - Dummy_ru_pairing(){} - Dummy_ru_pairing([[maybe_unused]] const Dummy_base_pairing& matrixToCopy){} - Dummy_ru_pairing([[maybe_unused]] Dummy_base_pairing&& other) noexcept{} - }; - using RU_pairing_option = typename std::conditional< Options::has_column_pairings && !Options::has_vine_update, RU_pairing >, Dummy_ru_pairing >::type; - - struct Dummy_ru_vine_swap{ - Dummy_ru_vine_swap& operator=([[maybe_unused]] Dummy_ru_vine_swap other){return *this;} - friend void swap([[maybe_unused]] Dummy_ru_vine_swap& d1, [[maybe_unused]] Dummy_ru_vine_swap& d2){} - - Dummy_ru_vine_swap([[maybe_unused]] Boundary_matrix_type &matrixR, [[maybe_unused]] Base_matrix_type &matrixU, [[maybe_unused]] dictionnary_type &pivotToColumn){} - Dummy_ru_vine_swap([[maybe_unused]] const Dummy_ru_vine_swap& matrixToCopy){} - Dummy_ru_vine_swap([[maybe_unused]] Dummy_ru_vine_swap&& other) noexcept{} - }; - using RU_vine_swap_option = typename std::conditional< Options::has_vine_update, RU_vine_swap >, Dummy_ru_vine_swap >::type; - - struct Dummy_chain_pairing{ - Dummy_chain_pairing& operator=([[maybe_unused]] Dummy_chain_pairing other){return *this;} - friend void swap([[maybe_unused]] Dummy_chain_pairing& d1, [[maybe_unused]] Dummy_chain_pairing& d2){} - - Dummy_chain_pairing(){} - Dummy_chain_pairing([[maybe_unused]] const Dummy_chain_pairing& matrixToCopy){} - Dummy_chain_pairing([[maybe_unused]] Dummy_chain_pairing&& other) noexcept{} - }; + using RU_representative_cycles_option = typename std::conditional< + Options::can_retrieve_representative_cycles, + RU_representative_cycles>, + Dummy_ru_representative_cycles + >::type; using Chain_pairing_option = typename std::conditional< Options::has_column_pairings && !Options::has_vine_update, Chain_pairing >, Dummy_chain_pairing >::type; - - struct Dummy_chain_vine_swap{ - Dummy_chain_vine_swap& operator=([[maybe_unused]] Dummy_chain_vine_swap other){return *this;} - friend void swap([[maybe_unused]] Dummy_chain_vine_swap& d1, [[maybe_unused]] Dummy_chain_vine_swap& d2){} - - Dummy_chain_vine_swap([[maybe_unused]] column_container_type& matrix){} - Dummy_chain_vine_swap([[maybe_unused]] const Dummy_chain_vine_swap& matrixToCopy){} - Dummy_chain_vine_swap([[maybe_unused]] Dummy_chain_vine_swap&& other) noexcept{} - }; - using Chain_vine_swap_option = typename std::conditional< Options::has_vine_update, Chain_vine_swap >, Dummy_chain_vine_swap >::type; - - struct Dummy_ru_representative_cycles{ - Dummy_ru_representative_cycles& operator=([[maybe_unused]] Dummy_ru_representative_cycles other){return *this;} - friend void swap([[maybe_unused]] Dummy_ru_representative_cycles& d1, [[maybe_unused]] Dummy_ru_representative_cycles& d2){} - - Dummy_ru_representative_cycles([[maybe_unused]] Boundary_matrix_type &matrixR, [[maybe_unused]] Base_matrix_type &matrixU){} - Dummy_ru_representative_cycles([[maybe_unused]] const Dummy_ru_representative_cycles& matrixToCopy){} - Dummy_ru_representative_cycles([[maybe_unused]] Dummy_ru_representative_cycles&& other) noexcept{} - }; - - using RU_representative_cycles_option = typename std::conditional< - Options::can_retrieve_representative_cycles, - RU_representative_cycles>, - Dummy_ru_representative_cycles - >::type; - - struct Dummy_chain_representative_cycles{ - Dummy_chain_representative_cycles& operator=([[maybe_unused]] Dummy_chain_representative_cycles other){return *this;} - friend void swap([[maybe_unused]] Dummy_chain_representative_cycles& d1, [[maybe_unused]] Dummy_chain_representative_cycles& d2){} - - Dummy_chain_representative_cycles([[maybe_unused]] column_container_type& matrix, [[maybe_unused]] dictionnary_type& pivotToPosition){} - Dummy_chain_representative_cycles([[maybe_unused]] const Dummy_chain_representative_cycles& matrixToCopy){} - Dummy_chain_representative_cycles([[maybe_unused]] Dummy_chain_representative_cycles&& other) noexcept{} - }; - using Chain_representative_cycles_option = typename std::conditional< Options::can_retrieve_representative_cycles, Chain_representative_cycles>, @@ -617,7 +504,6 @@ class Matrix const Row_type, Row_type >::type; - using insertion_return_type = typename std::conditional< Options::is_of_boundary_type || !isNonBasic, void, diff --git a/src/Persistence_matrix/include/gudhi/options.h b/src/Persistence_matrix/include/gudhi/options.h index 7fbecb8289..5c55ee46f3 100644 --- a/src/Persistence_matrix/include/gudhi/options.h +++ b/src/Persistence_matrix/include/gudhi/options.h @@ -44,7 +44,7 @@ struct Default_options{ static const bool is_parallelizable = parallelizable; //not implemented yet static const bool is_double_linked = true; //not implemented yet, it depends of the column type for now (all double linked except for UNORDERED_SET). usefull? - static const bool has_dimension_access = true; //ignored and put to false if matrix is not specialised, as the notion of dimension makes no sense for free columns. Also ignored but set to true if `has_column_pairings` is true and `has_vine_update` and `can_retrieve_representative_cycles` are false, because needed. + static const bool has_matrix_maximal_dimension_access = true; //ignored and put to false if matrix is not specialised, as the notion of dimension makes no sense for free columns. Also ignored but set to true if `has_column_pairings` is true and `has_vine_update` and `can_retrieve_representative_cycles` are false, because needed. static const bool has_row_access = false; static const bool has_intrusive_rows = column_type != Column_types::SET && column_type != Column_types::UNORDERED_SET; //ignored if has_row_access = false static const bool has_removable_rows = false; //ignored if has_row_access = false diff --git a/src/Persistence_matrix/test/Persistence_matrix_column_compression_unit_test.cpp b/src/Persistence_matrix/test/Persistence_matrix_column_compression_unit_test.cpp index 1e943db5ed..b1e6cc8b3f 100644 --- a/src/Persistence_matrix/test/Persistence_matrix_column_compression_unit_test.cpp +++ b/src/Persistence_matrix/test/Persistence_matrix_column_compression_unit_test.cpp @@ -141,108 +141,108 @@ void build_boundary_matrix(std::vector >, +typedef boost::mpl::list >, Matrix >, Matrix >, - Matrix >, + Matrix >, */ Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + Matrix >/* , Matrix >, Matrix >, - Matrix > + Matrix > */ > list_of_matrix_types_with_removable_columns; -typedef boost::mpl::list >, +typedef boost::mpl::list >, Matrix >, Matrix >, - Matrix >, + Matrix >, */ Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + Matrix >/* , Matrix >, Matrix >, - Matrix > + Matrix > */ > list_of_matrix_types; -typedef boost::mpl::list >, +typedef boost::mpl::list >, Matrix >, Matrix >, - Matrix >, + Matrix >, */ Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + Matrix >/* , Matrix >, Matrix >, - Matrix > + Matrix > */ > list_of_matrix_types_with_column_compression; template @@ -355,59 +355,59 @@ BOOST_AUTO_TEST_CASE_TEMPLATE(Matrix_basics_with_column_compression, Matrix, lis test_basic_methods(); } -typedef boost::mpl::list >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix > - > list_of_z2_matrix_types_without_column_compression; - -typedef boost::mpl::list >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix > - > list_of_z2_matrix_types_with_column_compression; +// typedef boost::mpl::list >, +// Matrix >, */ +// Matrix >, +// // Matrix >, +// // Matrix >, +// // Matrix >, +// // Matrix >, +// // Matrix >, +// Matrix >, +// // Matrix >, +// // Matrix >, +// // Matrix >, +// // Matrix >, +// // Matrix >, +// // Matrix >, +// Matrix >, +// // Matrix >, +// // Matrix >, +// // Matrix >, +// Matrix >, +// // Matrix >, +// // Matrix >, +// // Matrix >, +// // Matrix >, +// // Matrix >, +// Matrix >, +// // Matrix >, +// // Matrix >, +// // Matrix >, +// // Matrix >, +// // Matrix >, +// // Matrix >, +// Matrix >/* , +// Matrix > */ +// > list_of_z2_matrix_types_without_column_compression; + +// typedef boost::mpl::list >, +// Matrix >, */ +// Matrix >, +// // Matrix >, +// // Matrix >, +// // Matrix >, +// // Matrix >, +// // Matrix >, +// Matrix >, +// // Matrix >, +// // Matrix >, +// // Matrix >, +// // Matrix >, +// // Matrix >, +// Matrix >/* , +// Matrix > */ +// > list_of_z2_matrix_types_with_column_compression; template void test_equal_columns(const std::vector& col1, const Column_type& col2) @@ -442,105 +442,105 @@ void test_equal_columns(const std::vector >& col1, co } } -template -void test_z2_insertion_and_addition(){ - using boundary_matrix = std::vector >; - - boundary_matrix ordered_boundaries; - build_boundary_matrix(ordered_boundaries); - auto boundary2 = ordered_boundaries.back(); - ordered_boundaries.pop_back(); - auto boundary1 = ordered_boundaries.back(); - ordered_boundaries.pop_back(); - Matrix m(ordered_boundaries); - - unsigned int i = 0; - for (auto& b : ordered_boundaries){ - const auto& col = m.get_column(i++); - test_equal_columns(b, col); - } - - m.insert_boundary(boundary1); - ordered_boundaries.push_back(boundary1); - m.insert_boundary(boundary2); - ordered_boundaries.push_back(boundary2); - - i = 0; - for (auto& b : ordered_boundaries){ - const auto& col = m.get_column(i++); - test_equal_columns(b, col); - } - - m.add_to(6, 7); - - std::vector column = m.get_column(7).get_content(); - BOOST_CHECK_EQUAL(column.size(), 6); - auto it = column.begin(); - BOOST_CHECK_EQUAL(*it++, 1); - BOOST_CHECK_EQUAL(*it++, 0); - BOOST_CHECK_EQUAL(*it++, 0); - BOOST_CHECK_EQUAL(*it++, 1); - BOOST_CHECK_EQUAL(*it++, 0); - BOOST_CHECK_EQUAL(*it, 1); -} - -BOOST_AUTO_TEST_CASE_TEMPLATE(Matrix_z2_insertion_and_addition, Matrix, list_of_z2_matrix_types_without_column_compression) { - test_z2_insertion_and_addition(); -} -BOOST_AUTO_TEST_CASE_TEMPLATE(Matrix_z2_insertion_and_addition_with_cc, Matrix, list_of_z2_matrix_types_with_column_compression) { - test_z2_insertion_and_addition(); -} - -typedef boost::mpl::list >, - Matrix >, +// template +// void test_z2_insertion_and_addition(){ +// using boundary_matrix = std::vector >; + +// boundary_matrix ordered_boundaries; +// build_boundary_matrix(ordered_boundaries); +// auto boundary2 = ordered_boundaries.back(); +// ordered_boundaries.pop_back(); +// auto boundary1 = ordered_boundaries.back(); +// ordered_boundaries.pop_back(); +// Matrix m(ordered_boundaries); + +// unsigned int i = 0; +// for (auto& b : ordered_boundaries){ +// const auto& col = m.get_column(i++); +// test_equal_columns(b, col); +// } + +// m.insert_boundary(boundary1); +// ordered_boundaries.push_back(boundary1); +// m.insert_boundary(boundary2); +// ordered_boundaries.push_back(boundary2); + +// i = 0; +// for (auto& b : ordered_boundaries){ +// const auto& col = m.get_column(i++); +// test_equal_columns(b, col); +// } + +// m.add_to(6, 7); + +// std::vector column = m.get_column(7).get_content(); +// BOOST_CHECK_EQUAL(column.size(), 6); +// auto it = column.begin(); +// BOOST_CHECK_EQUAL(*it++, 1); +// BOOST_CHECK_EQUAL(*it++, 0); +// BOOST_CHECK_EQUAL(*it++, 0); +// BOOST_CHECK_EQUAL(*it++, 1); +// BOOST_CHECK_EQUAL(*it++, 0); +// BOOST_CHECK_EQUAL(*it, 1); +// } + +// BOOST_AUTO_TEST_CASE_TEMPLATE(Matrix_z2_insertion_and_addition, Matrix, list_of_z2_matrix_types_without_column_compression) { +// test_z2_insertion_and_addition(); +// } +// BOOST_AUTO_TEST_CASE_TEMPLATE(Matrix_z2_insertion_and_addition_with_cc, Matrix, list_of_z2_matrix_types_with_column_compression) { +// test_z2_insertion_and_addition(); +// } + +typedef boost::mpl::list >, + Matrix >, */ Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, Matrix >, - Matrix >, - Matrix >, - Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix > + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + Matrix >/* , + Matrix > */ > list_of_z5_matrix_types_without_column_compression; -typedef boost::mpl::list >, - Matrix >, +typedef boost::mpl::list >, + Matrix >, */ Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix > + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + Matrix >/* , + Matrix > */ > list_of_z5_matrix_types_with_column_compression; template @@ -596,36 +596,36 @@ BOOST_AUTO_TEST_CASE_TEMPLATE(Matrix_z5_insertion_and_addition_with_cc, Matrix, test_z5_insertion_and_addition(); } -typedef boost::mpl::list >, - Matrix >, +typedef boost::mpl::list >, + Matrix >, */ Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, Matrix >, - Matrix >, - Matrix >, - Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, Matrix >, - Matrix >, - Matrix >, - Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix > + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + Matrix >/* , + Matrix > */ > list_of_z5_matrix_types_with_row_access; BOOST_AUTO_TEST_CASE_TEMPLATE(Base_row_access_columns_options, Matrix, list_of_z5_matrix_types_with_row_access) { @@ -682,109 +682,109 @@ BOOST_AUTO_TEST_CASE_TEMPLATE(Base_row_access_columns_options, Matrix, list_of_z } } -typedef boost::mpl::list >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix > - > list_of_z2_matrix_types_with_row_access; - -BOOST_AUTO_TEST_CASE_TEMPLATE(Z2_base_row_access_columns_options, Matrix, list_of_z2_matrix_types_with_row_access) { - using boundary_matrix = std::vector >; - - boundary_matrix ordered_boundaries; - build_boundary_matrix(ordered_boundaries); - - Matrix mb(ordered_boundaries); - - std::vector > rows; - if constexpr (Matrix::Option_list::has_column_compression){ - //if the union find structure changes, the column_index values of de cells could also change. Change the test with all possibilities? -// rows.push_back({9,10}); -// rows.push_back({8,9}); -// rows.push_back({8}); -// rows.push_back({7}); -// rows.push_back({7,9,10}); -// rows.push_back({7}); - rows.push_back({3,6}); - rows.push_back({3,5}); - rows.push_back({5}); - rows.push_back({7}); - rows.push_back({3,6,7}); - rows.push_back({7}); - } else { - rows.push_back({3,6,9,10,11}); - rows.push_back({3,5,8,9,11}); - rows.push_back({5,8}); - rows.push_back({7}); - rows.push_back({3,6,7,9,10,11}); - rows.push_back({7}); - } - - //rows are unordered - std::vector > ordered_rows(rows.size()); - for (unsigned int i = 0; i < rows.size(); ++i){ - for (auto& cell : mb.get_row(i)){ - ordered_rows[i].insert(cell.get_column_index()); - } - } - for (unsigned int i = 0; i < rows.size(); ++i){ - auto& row = ordered_rows[i]; - BOOST_CHECK_EQUAL(row.size(), rows[i].size()); - auto itRow = row.begin(); - auto itChain = rows[i].begin(); - while (itChain != rows[i].end()){ - BOOST_CHECK_EQUAL(*itRow, *itChain); - ++itRow; ++itChain; - } - } -} - -typedef boost::mpl::list >, +// typedef boost::mpl::list >, +// Matrix >, */ +// Matrix >, +// // Matrix >, +// // Matrix >, +// // Matrix >, +// // Matrix >, +// // Matrix >, +// Matrix >, +// // Matrix >, +// // Matrix >, +// // Matrix >, +// Matrix >, +// // Matrix >, +// // Matrix >, +// // Matrix >, +// // Matrix >, +// // Matrix >, +// Matrix >, +// // Matrix >, +// // Matrix >, +// // Matrix >, +// Matrix >, +// // Matrix >, +// // Matrix >, +// // Matrix >, +// // Matrix >, +// // Matrix >, +// Matrix >/* , +// Matrix > */ +// > list_of_z2_matrix_types_with_row_access; + +// BOOST_AUTO_TEST_CASE_TEMPLATE(Z2_base_row_access_columns_options, Matrix, list_of_z2_matrix_types_with_row_access) { +// using boundary_matrix = std::vector >; + +// boundary_matrix ordered_boundaries; +// build_boundary_matrix(ordered_boundaries); + +// Matrix mb(ordered_boundaries); + +// std::vector > rows; +// if constexpr (Matrix::Option_list::has_column_compression){ +// //if the union find structure changes, the column_index values of de cells could also change. Change the test with all possibilities? +// // rows.push_back({9,10}); +// // rows.push_back({8,9}); +// // rows.push_back({8}); +// // rows.push_back({7}); +// // rows.push_back({7,9,10}); +// // rows.push_back({7}); +// rows.push_back({3,6}); +// rows.push_back({3,5}); +// rows.push_back({5}); +// rows.push_back({7}); +// rows.push_back({3,6,7}); +// rows.push_back({7}); +// } else { +// rows.push_back({3,6,9,10,11}); +// rows.push_back({3,5,8,9,11}); +// rows.push_back({5,8}); +// rows.push_back({7}); +// rows.push_back({3,6,7,9,10,11}); +// rows.push_back({7}); +// } + +// //rows are unordered +// std::vector > ordered_rows(rows.size()); +// for (unsigned int i = 0; i < rows.size(); ++i){ +// for (auto& cell : mb.get_row(i)){ +// ordered_rows[i].insert(cell.get_column_index()); +// } +// } +// for (unsigned int i = 0; i < rows.size(); ++i){ +// auto& row = ordered_rows[i]; +// BOOST_CHECK_EQUAL(row.size(), rows[i].size()); +// auto itRow = row.begin(); +// auto itChain = rows[i].begin(); +// while (itChain != rows[i].end()){ +// BOOST_CHECK_EQUAL(*itRow, *itChain); +// ++itRow; ++itChain; +// } +// } +// } + +typedef boost::mpl::list >, Matrix >, Matrix >, - Matrix >, + Matrix >, */ Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + Matrix >/* , Matrix >, Matrix >, - Matrix > + Matrix > */ > list_of_matrix_types_with_removable_columns_and_row_access; BOOST_AUTO_TEST_CASE_TEMPLATE(Removable_columns_options, Matrix, list_of_matrix_types_with_removable_columns_and_row_access) { diff --git a/src/Persistence_matrix/test/Persistence_matrix_columns_test.cpp b/src/Persistence_matrix/test/Persistence_matrix_columns_test.cpp index 232f821b27..88ee471fcb 100644 --- a/src/Persistence_matrix/test/Persistence_matrix_columns_test.cpp +++ b/src/Persistence_matrix/test/Persistence_matrix_columns_test.cpp @@ -27,6 +27,10 @@ // #include "gudhi/column_types/cell.h" #include "gudhi/Persistence_matrix/columns/cell_types.h" #include "gudhi/Persistence_matrix/columns/row_access.h" +#include "gudhi/Persistence_matrix/columns/column_dimension_holder.h" +#include "gudhi/Persistence_matrix/columns/chain_column_extra_properties.h" + +#include "gudhi/Persistence_matrix/columns/intrusive_list_column.h" #include "gudhi/column_types/list_column.h" #include "gudhi/column_types/set_column.h" @@ -37,9 +41,9 @@ #include "gudhi/column_types/z2_set_column.h" #include "gudhi/column_types/z2_unordered_set_column.h" #include "gudhi/column_types/z2_vector_column.h" -#include "gudhi/column_types/intrusive_list_column.h" +// #include "gudhi/column_types/intrusive_list_column.h" #include "gudhi/column_types/intrusive_set_column.h" -#include "gudhi/column_types/z2_intrusive_list_column.h" +// #include "gudhi/column_types/z2_intrusive_list_column.h" #include "gudhi/column_types/z2_intrusive_set_column.h" #include "gudhi/column_types/boundary_columns/boundary_list_column.h" @@ -51,9 +55,9 @@ #include "gudhi/column_types/boundary_columns/z2_boundary_set_column.h" #include "gudhi/column_types/boundary_columns/z2_boundary_unordered_set_column.h" #include "gudhi/column_types/boundary_columns/z2_boundary_vector_column.h" -#include "gudhi/column_types/boundary_columns/boundary_intrusive_list_column.h" +// #include "gudhi/column_types/boundary_columns/boundary_intrusive_list_column.h" #include "gudhi/column_types/boundary_columns/boundary_intrusive_set_column.h" -#include "gudhi/column_types/boundary_columns/z2_boundary_intrusive_list_column.h" +// #include "gudhi/column_types/boundary_columns/z2_boundary_intrusive_list_column.h" #include "gudhi/column_types/boundary_columns/z2_boundary_intrusive_set_column.h" #include "gudhi/column_types/chain_columns/chain_list_column.h" @@ -65,13 +69,15 @@ #include "gudhi/column_types/chain_columns/z2_chain_set_column.h" #include "gudhi/column_types/chain_columns/z2_chain_unordered_set_column.h" #include "gudhi/column_types/chain_columns/z2_chain_vector_column.h" -#include "gudhi/column_types/chain_columns/chain_intrusive_list_column.h" +// #include "gudhi/column_types/chain_columns/chain_intrusive_list_column.h" #include "gudhi/column_types/chain_columns/chain_intrusive_set_column.h" -#include "gudhi/column_types/chain_columns/z2_chain_intrusive_list_column.h" +// #include "gudhi/column_types/chain_columns/z2_chain_intrusive_list_column.h" #include "gudhi/column_types/chain_columns/z2_chain_intrusive_set_column.h" using Gudhi::persistence_matrix::Zp_field_element; +using Gudhi::persistence_matrix::Intrusive_list_column; + using Gudhi::persistence_matrix::List_column; using Gudhi::persistence_matrix::Set_column; using Gudhi::persistence_matrix::Unordered_set_column; @@ -81,9 +87,8 @@ using Gudhi::persistence_matrix::Z2_list_column; using Gudhi::persistence_matrix::Z2_set_column; using Gudhi::persistence_matrix::Z2_unordered_set_column; using Gudhi::persistence_matrix::Z2_vector_column; -using Gudhi::persistence_matrix::Intrusive_list_column; using Gudhi::persistence_matrix::Intrusive_set_column; -using Gudhi::persistence_matrix::Z2_intrusive_list_column; +// using Gudhi::persistence_matrix::Z2_intrusive_list_column; using Gudhi::persistence_matrix::Z2_intrusive_set_column; using Gudhi::persistence_matrix::List_boundary_column; @@ -95,9 +100,9 @@ using Gudhi::persistence_matrix::Z2_list_boundary_column; using Gudhi::persistence_matrix::Z2_set_boundary_column; using Gudhi::persistence_matrix::Z2_unordered_set_boundary_column; using Gudhi::persistence_matrix::Z2_vector_boundary_column; -using Gudhi::persistence_matrix::Intrusive_list_boundary_column; +// using Gudhi::persistence_matrix::Intrusive_list_boundary_column; using Gudhi::persistence_matrix::Intrusive_set_boundary_column; -using Gudhi::persistence_matrix::Z2_intrusive_list_boundary_column; +// using Gudhi::persistence_matrix::Z2_intrusive_list_boundary_column; using Gudhi::persistence_matrix::Z2_intrusive_set_boundary_column; using Gudhi::persistence_matrix::List_chain_column; @@ -109,9 +114,9 @@ using Gudhi::persistence_matrix::Z2_list_chain_column; using Gudhi::persistence_matrix::Z2_set_chain_column; using Gudhi::persistence_matrix::Z2_unordered_set_chain_column; using Gudhi::persistence_matrix::Z2_vector_chain_column; -using Gudhi::persistence_matrix::Intrusive_list_chain_column; +// using Gudhi::persistence_matrix::Intrusive_list_chain_column; using Gudhi::persistence_matrix::Intrusive_set_chain_column; -using Gudhi::persistence_matrix::Z2_intrusive_list_chain_column; +// using Gudhi::persistence_matrix::Z2_intrusive_list_chain_column; using Gudhi::persistence_matrix::Z2_intrusive_set_chain_column; using Gudhi::persistence_matrix::Cell; @@ -119,9 +124,14 @@ using Gudhi::persistence_matrix::Cell_column_index; using Gudhi::persistence_matrix::Cell_field_element; using Gudhi::persistence_matrix::Row_access; +using Gudhi::persistence_matrix::Dummy_row_access; using Gudhi::persistence_matrix::dimension_type; using Gudhi::persistence_matrix::matrix_row_tag; using Gudhi::persistence_matrix::matrix_column_tag; +using Gudhi::persistence_matrix::Column_dimension_holder; +using Gudhi::persistence_matrix::Dummy_dimension_holder; +using Gudhi::persistence_matrix::Chain_column_extra_properties; +using Gudhi::persistence_matrix::Dummy_chain_properties; using Z5 = Zp_field_element<5>; using Z2 = Zp_field_element<2>; @@ -137,7 +147,7 @@ enum Column_types { INTRUSIVE_SET }; -template +template struct z2_options{ using field_coeff_type = Z2; static const bool is_z2 = true; @@ -145,9 +155,12 @@ struct z2_options{ static const bool has_row_access = false; static const bool has_intrusive_rows = false; static const bool has_removable_rows = false; + + static const bool is_basic = basic; //exists just for the tests + static const bool is_of_boundary_type = boundary; }; -template +template struct z2_ra_options{ using field_coeff_type = Z2; static const bool is_z2 = true; @@ -155,9 +168,12 @@ struct z2_ra_options{ static const bool has_row_access = true; static const bool has_intrusive_rows = false; static const bool has_removable_rows = false; + + static const bool is_basic = basic; //exists just for the tests + static const bool is_of_boundary_type = boundary; }; -template +template struct z2_ra_i_options{ using field_coeff_type = Z2; static const bool is_z2 = true; @@ -165,9 +181,12 @@ struct z2_ra_i_options{ static const bool has_row_access = true; static const bool has_intrusive_rows = true; static const bool has_removable_rows = false; + + static const bool is_basic = basic; //exists just for the tests + static const bool is_of_boundary_type = boundary; }; -template +template struct z5_options{ using field_coeff_type = Z5; static const bool is_z2 = false; @@ -175,9 +194,12 @@ struct z5_options{ static const bool has_row_access = false; static const bool has_intrusive_rows = false; static const bool has_removable_rows = false; + + static const bool is_basic = basic; //exists just for the tests + static const bool is_of_boundary_type = boundary; }; -template +template struct z5_ra_options{ using field_coeff_type = Z5; static const bool is_z2 = false; @@ -185,9 +207,12 @@ struct z5_ra_options{ static const bool has_row_access = true; static const bool has_intrusive_rows = false; static const bool has_removable_rows = false; + + static const bool is_basic = basic; //exists just for the tests + static const bool is_of_boundary_type = boundary; }; -template +template struct z5_ra_i_options{ using field_coeff_type = Z5; static const bool is_z2 = false; @@ -195,6 +220,9 @@ struct z5_ra_i_options{ static const bool has_row_access = true; static const bool has_intrusive_rows = true; static const bool has_removable_rows = false; + + static const bool is_basic = basic; //exists just for the tests + static const bool is_of_boundary_type = boundary; }; template @@ -204,6 +232,8 @@ struct Mini_matrix{ using index = unsigned int; using dimension_type = int; + static constexpr bool isNonBasic = !Options::is_basic; + // struct matrix_row_tag; // struct matrix_column_tag; @@ -281,104 +311,107 @@ struct Mini_matrix{ using row_container_type = std::vector; - struct Dummy_row_access{ - friend void swap([[maybe_unused]] Dummy_row_access& d1, [[maybe_unused]] Dummy_row_access& d2){} - - Dummy_row_access(){} - Dummy_row_access([[maybe_unused]] Dummy_row_access&& other) noexcept{} - - static constexpr bool isActive_ = false; - }; - using Row_access_option = typename std::conditional< Options::has_row_access, Row_access >, Dummy_row_access >::type; + + using Column_dimension_option = typename std::conditional< + isNonBasic, + Column_dimension_holder >, + Dummy_dimension_holder + >::type; + + using Chain_column_option = typename std::conditional< + isNonBasic && !Options::is_of_boundary_type, + Chain_column_extra_properties >, + Dummy_chain_properties + >::type; }; -template -using matrix_z2 = Mini_matrix >; -template -using matrix_z2_ra = Mini_matrix >; -template -using matrix_z2_ra_i = Mini_matrix >; -template -using matrix_z5 = Mini_matrix >; -template -using matrix_z5_ra = Mini_matrix >; -template -using matrix_z5_ra_i = Mini_matrix >; +template +using matrix_z2 = Mini_matrix >; +template +using matrix_z2_ra = Mini_matrix >; +template +using matrix_z2_ra_i = Mini_matrix >; +template +using matrix_z5 = Mini_matrix >; +template +using matrix_z5_ra = Mini_matrix >; +template +using matrix_z5_ra_i = Mini_matrix >; ////////////// BASICS ////////////// -typedef boost::mpl::list::Cell_type, matrix_z5::Row_access_option>, +typedef boost::mpl::list::Cell_type, matrix_z5::Row_access_option>, Set_column::Cell_type, matrix_z5::Row_access_option>, Unordered_set_column::Cell_type, matrix_z5::Row_access_option>, - Vector_column::Cell_type, matrix_z5::Row_access_option>, - Intrusive_list_column::Cell_type, matrix_z5::Row_access_option>, - Intrusive_set_column::Cell_type, matrix_z5::Row_access_option>, - List_boundary_column::Cell_type, matrix_z5::Row_access_option>, - Set_boundary_column::Cell_type, matrix_z5::Row_access_option>, - Unordered_set_boundary_column::Cell_type, matrix_z5::Row_access_option>, - Vector_boundary_column::Cell_type, matrix_z5::Row_access_option>, - Intrusive_list_boundary_column::Cell_type, matrix_z5::Row_access_option>, - Intrusive_set_boundary_column::Cell_type, matrix_z5::Row_access_option> + Vector_column::Cell_type, matrix_z5::Row_access_option>, */ + Intrusive_list_column >, + // Intrusive_set_column::Cell_type, matrix_z5::Row_access_option>, + // List_boundary_column::Cell_type, matrix_z5::Row_access_option>, + // Set_boundary_column::Cell_type, matrix_z5::Row_access_option>, + // Unordered_set_boundary_column::Cell_type, matrix_z5::Row_access_option>, + // Vector_boundary_column::Cell_type, matrix_z5::Row_access_option>, + Intrusive_list_column >/* , + Intrusive_set_boundary_column::Cell_type, matrix_z5::Row_access_option> */ > list_of_5_columns; -typedef boost::mpl::list::Cell_type, matrix_z2::Row_access_option>, Z2_set_column::Cell_type, matrix_z2::Row_access_option>, Z2_unordered_set_column::Cell_type, matrix_z2::Row_access_option>, - Z2_vector_column::Cell_type, matrix_z2::Row_access_option>, - Z2_intrusive_list_column::Cell_type, matrix_z2::Row_access_option>, - Z2_intrusive_set_column::Cell_type, matrix_z2::Row_access_option>, - Z2_heap_boundary_column, - Z2_list_boundary_column::Cell_type, matrix_z2::Row_access_option>, - Z2_set_boundary_column::Cell_type, matrix_z2::Row_access_option>, - Z2_unordered_set_boundary_column::Cell_type, matrix_z2::Row_access_option>, - Z2_vector_boundary_column::Cell_type, matrix_z2::Row_access_option>, - Z2_intrusive_list_boundary_column::Cell_type, matrix_z2::Row_access_option>, - Z2_intrusive_set_boundary_column::Cell_type, matrix_z2::Row_access_option> + Z2_vector_column::Cell_type, matrix_z2::Row_access_option>, */ + Intrusive_list_column >, + // Z2_intrusive_set_column::Cell_type, matrix_z2::Row_access_option>, + // Z2_heap_boundary_column, + // Z2_list_boundary_column::Cell_type, matrix_z2::Row_access_option>, + // Z2_set_boundary_column::Cell_type, matrix_z2::Row_access_option>, + // Z2_unordered_set_boundary_column::Cell_type, matrix_z2::Row_access_option>, + // Z2_vector_boundary_column::Cell_type, matrix_z2::Row_access_option>, + Intrusive_list_column >/* , + Z2_intrusive_set_boundary_column::Cell_type, matrix_z2::Row_access_option> */ > list_of_2_columns; -typedef boost::mpl::list::Cell_type, matrix_z5::Row_access_option>, +typedef boost::mpl::list::Cell_type, matrix_z5::Row_access_option>, Set_boundary_column::Cell_type, matrix_z5::Row_access_option>, Unordered_set_boundary_column::Cell_type, matrix_z5::Row_access_option>, - Vector_boundary_column::Cell_type, matrix_z5::Row_access_option>, - Intrusive_list_boundary_column::Cell_type, matrix_z5::Row_access_option>, - Intrusive_set_boundary_column::Cell_type, matrix_z5::Row_access_option> + Vector_boundary_column::Cell_type, matrix_z5::Row_access_option>, */ + Intrusive_list_column >/* , + Intrusive_set_boundary_column::Cell_type, matrix_z5::Row_access_option> */ > list_of_5_boundary_columns; -typedef boost::mpl::list::Cell_type, matrix_z2::Row_access_option>, Z2_set_boundary_column::Cell_type, matrix_z2::Row_access_option>, Z2_unordered_set_boundary_column::Cell_type, matrix_z2::Row_access_option>, - Z2_vector_boundary_column::Cell_type, matrix_z2::Row_access_option>, - Z2_intrusive_list_boundary_column::Cell_type, matrix_z2::Row_access_option>, - Z2_intrusive_set_boundary_column::Cell_type, matrix_z2::Row_access_option> + Z2_vector_boundary_column::Cell_type, matrix_z2::Row_access_option>, */ + Intrusive_list_column >/* , + Z2_intrusive_set_boundary_column::Cell_type, matrix_z2::Row_access_option> */ > list_of_2_boundary_columns; -typedef boost::mpl::list::Cell_type, matrix_z5::Row_access_option>, +typedef boost::mpl::list::Cell_type, matrix_z5::Row_access_option>, Set_chain_column::Cell_type, matrix_z5::Row_access_option>, Unordered_set_chain_column::Cell_type, matrix_z5::Row_access_option>, - Vector_chain_column::Cell_type, matrix_z5::Row_access_option>, - Intrusive_list_chain_column::Cell_type, matrix_z5::Row_access_option>, - Intrusive_set_chain_column::Cell_type, matrix_z5::Row_access_option> + Vector_chain_column::Cell_type, matrix_z5::Row_access_option>, */ + Intrusive_list_column >/* , + Intrusive_set_chain_column::Cell_type, matrix_z5::Row_access_option> */ > list_of_5_chain_columns; -typedef boost::mpl::list, +typedef boost::mpl::list, Z2_list_chain_column::Cell_type, matrix_z2::Row_access_option>, Z2_set_chain_column::Cell_type, matrix_z2::Row_access_option>, Z2_unordered_set_chain_column::Cell_type, matrix_z2::Row_access_option>, - Z2_vector_chain_column::Cell_type, matrix_z2::Row_access_option>, - Z2_intrusive_list_chain_column::Cell_type, matrix_z2::Row_access_option>, - Z2_intrusive_set_chain_column::Cell_type, matrix_z2::Row_access_option> + Z2_vector_chain_column::Cell_type, matrix_z2::Row_access_option>, */ + Intrusive_list_column >/* , + Z2_intrusive_set_chain_column::Cell_type, matrix_z2::Row_access_option> */ > list_of_2_chain_columns; ////////////// ROWS ////////////// -typedef boost::mpl::list::Cell_type, matrix_z5_ra_i::Row_access_option>, +typedef boost::mpl::list::Cell_type, matrix_z5_ra_i::Row_access_option>, Vector_column::Cell_type, matrix_z5_ra_i::Row_access_option>, List_boundary_column::Cell_type, matrix_z5_ra_i::Row_access_option>, Vector_boundary_column::Cell_type, matrix_z5_ra_i::Row_access_option>, @@ -389,18 +422,18 @@ typedef boost::mpl::list::Cel List_boundary_column::Cell_type, matrix_z5_ra::Row_access_option>, Set_boundary_column::Cell_type, matrix_z5_ra::Row_access_option>, Unordered_set_boundary_column::Cell_type, matrix_z5_ra::Row_access_option>, - Vector_boundary_column::Cell_type, matrix_z5_ra::Row_access_option>, - Intrusive_list_column::Cell_type, matrix_z5_ra_i::Row_access_option>, - Intrusive_list_boundary_column::Cell_type, matrix_z5_ra_i::Row_access_option>, - Intrusive_list_column::Cell_type, matrix_z5_ra::Row_access_option>, - Intrusive_list_boundary_column::Cell_type, matrix_z5_ra::Row_access_option>, + Vector_boundary_column::Cell_type, matrix_z5_ra::Row_access_option>, */ + Intrusive_list_column >, + Intrusive_list_column >, + Intrusive_list_column >, + Intrusive_list_column >/* , Intrusive_set_column::Cell_type, matrix_z5_ra_i::Row_access_option>, Intrusive_set_boundary_column::Cell_type, matrix_z5_ra_i::Row_access_option>, Intrusive_set_column::Cell_type, matrix_z5_ra::Row_access_option>, - Intrusive_set_boundary_column::Cell_type, matrix_z5_ra::Row_access_option> + Intrusive_set_boundary_column::Cell_type, matrix_z5_ra::Row_access_option> */ > list_of_5_columns_with_row; -typedef boost::mpl::list::Cell_type, matrix_z2_ra_i::Row_access_option>, +typedef boost::mpl::list::Cell_type, matrix_z2_ra_i::Row_access_option>, Z2_vector_column::Cell_type, matrix_z2_ra_i::Row_access_option>, Z2_list_boundary_column::Cell_type, matrix_z2_ra_i::Row_access_option>, Z2_vector_boundary_column::Cell_type, matrix_z2_ra_i::Row_access_option>, @@ -411,80 +444,58 @@ typedef boost::mpl::list::Cell Z2_list_boundary_column::Cell_type, matrix_z2_ra::Row_access_option>, Z2_set_boundary_column::Cell_type, matrix_z2_ra::Row_access_option>, Z2_unordered_set_boundary_column::Cell_type, matrix_z2_ra::Row_access_option>, - Z2_vector_boundary_column::Cell_type, matrix_z2_ra::Row_access_option>, - Z2_intrusive_list_column::Cell_type, matrix_z2_ra_i::Row_access_option>, - Z2_intrusive_list_boundary_column::Cell_type, matrix_z2_ra_i::Row_access_option>, - Z2_intrusive_list_column::Cell_type, matrix_z2_ra::Row_access_option>, - Z2_intrusive_list_boundary_column::Cell_type, matrix_z2_ra::Row_access_option>, + Z2_vector_boundary_column::Cell_type, matrix_z2_ra::Row_access_option>, */ + Intrusive_list_column >, + Intrusive_list_column >, + Intrusive_list_column >, + Intrusive_list_column >/* , Z2_intrusive_set_column::Cell_type, matrix_z2_ra_i::Row_access_option>, Z2_intrusive_set_boundary_column::Cell_type, matrix_z2_ra_i::Row_access_option>, Z2_intrusive_set_column::Cell_type, matrix_z2_ra::Row_access_option>, - Z2_intrusive_set_boundary_column::Cell_type, matrix_z2_ra::Row_access_option> + Z2_intrusive_set_boundary_column::Cell_type, matrix_z2_ra::Row_access_option> */ > list_of_2_columns_with_row; -typedef boost::mpl::list::Cell_type, matrix_z5_ra_i::Row_access_option>, +typedef boost::mpl::list::Cell_type, matrix_z5_ra_i::Row_access_option>, Vector_chain_column::Cell_type, matrix_z5_ra_i::Row_access_option>, List_chain_column::Cell_type, matrix_z5_ra::Row_access_option>, Set_chain_column::Cell_type, matrix_z5_ra::Row_access_option>, Unordered_set_chain_column::Cell_type, matrix_z5_ra::Row_access_option>, - Vector_chain_column::Cell_type, matrix_z5_ra::Row_access_option>, - Intrusive_list_chain_column::Cell_type, matrix_z5_ra_i::Row_access_option>, - Intrusive_list_chain_column::Cell_type, matrix_z5_ra::Row_access_option>, + Vector_chain_column::Cell_type, matrix_z5_ra::Row_access_option>, */ + Intrusive_list_column >, + Intrusive_list_column >/* , Intrusive_set_chain_column::Cell_type, matrix_z5_ra_i::Row_access_option>, - Intrusive_set_chain_column::Cell_type, matrix_z5_ra::Row_access_option> + Intrusive_set_chain_column::Cell_type, matrix_z5_ra::Row_access_option> */ > list_of_5_chain_columns_with_row; -typedef boost::mpl::list::Cell_type, matrix_z2_ra_i::Row_access_option>, +typedef boost::mpl::list::Cell_type, matrix_z2_ra_i::Row_access_option>, Z2_vector_chain_column::Cell_type, matrix_z2_ra_i::Row_access_option>, Z2_list_chain_column::Cell_type, matrix_z2_ra::Row_access_option>, Z2_set_chain_column::Cell_type, matrix_z2_ra::Row_access_option>, Z2_unordered_set_chain_column::Cell_type, matrix_z2_ra::Row_access_option>, - Z2_vector_chain_column::Cell_type, matrix_z2_ra::Row_access_option>, - Z2_intrusive_list_chain_column::Cell_type, matrix_z2_ra_i::Row_access_option>, - Z2_intrusive_list_chain_column::Cell_type, matrix_z2_ra::Row_access_option>, + Z2_vector_chain_column::Cell_type, matrix_z2_ra::Row_access_option>, */ + Intrusive_list_column >, + Intrusive_list_column >/* , Z2_intrusive_set_chain_column::Cell_type, matrix_z2_ra_i::Row_access_option>, - Z2_intrusive_set_chain_column::Cell_type, matrix_z2_ra::Row_access_option> + Z2_intrusive_set_chain_column::Cell_type, matrix_z2_ra::Row_access_option> */ > list_of_2_chain_columns_with_row; ////////////// PAIRING ////////////// -typedef boost::mpl::list::Cell_type, matrix_z5::Row_access_option>, +typedef boost::mpl::list::Cell_type, matrix_z5::Row_access_option>, Set_chain_column::Cell_type, matrix_z5::Row_access_option>, Unordered_set_chain_column::Cell_type, matrix_z5::Row_access_option>, - Vector_chain_column::Cell_type, matrix_z5::Row_access_option>, - Intrusive_list_chain_column::Cell_type, matrix_z5::Row_access_option>, - Intrusive_set_chain_column::Cell_type, matrix_z5::Row_access_option>, - Z2_heap_chain_column, - Z2_list_chain_column::Cell_type, matrix_z2::Row_access_option>, - Z2_set_chain_column::Cell_type, matrix_z2::Row_access_option>, - Z2_unordered_set_chain_column::Cell_type, matrix_z2::Row_access_option>, - Z2_vector_chain_column::Cell_type, matrix_z2::Row_access_option>, - Z2_intrusive_list_chain_column::Cell_type, matrix_z2::Row_access_option>, - Z2_intrusive_set_chain_column::Cell_type, matrix_z2::Row_access_option> + Vector_chain_column::Cell_type, matrix_z5::Row_access_option>, */ + Intrusive_list_column >, + // Intrusive_set_chain_column::Cell_type, matrix_z5::Row_access_option>, + // Z2_heap_chain_column, + // Z2_list_chain_column::Cell_type, matrix_z2::Row_access_option>, + // Z2_set_chain_column::Cell_type, matrix_z2::Row_access_option>, + // Z2_unordered_set_chain_column::Cell_type, matrix_z2::Row_access_option>, + // Z2_vector_chain_column::Cell_type, matrix_z2::Row_access_option>, */ + Intrusive_list_column >/* , + Z2_intrusive_set_chain_column::Cell_type, matrix_z2::Row_access_option> */ > list_of_chain_pairing_columns; -typedef boost::mpl::list::Cell_type, matrix_z5_ra_i::Row_access_option>, - Vector_chain_column::Cell_type, matrix_z5_ra_i::Row_access_option>, - List_chain_column::Cell_type, matrix_z5_ra::Row_access_option>, - Set_chain_column::Cell_type, matrix_z5_ra::Row_access_option>, - Unordered_set_chain_column::Cell_type, matrix_z5_ra::Row_access_option>, - Vector_chain_column::Cell_type, matrix_z5_ra::Row_access_option>, - Intrusive_list_chain_column::Cell_type, matrix_z5_ra_i::Row_access_option>, - Intrusive_list_chain_column::Cell_type, matrix_z5_ra::Row_access_option>, - Intrusive_set_chain_column::Cell_type, matrix_z5_ra_i::Row_access_option>, - Intrusive_set_chain_column::Cell_type, matrix_z5_ra::Row_access_option>, - Z2_list_chain_column::Cell_type, matrix_z2_ra_i::Row_access_option>, - Z2_vector_chain_column::Cell_type, matrix_z2_ra_i::Row_access_option>, - Z2_list_chain_column::Cell_type, matrix_z2_ra::Row_access_option>, - Z2_set_chain_column::Cell_type, matrix_z2_ra::Row_access_option>, - Z2_unordered_set_chain_column::Cell_type, matrix_z2_ra::Row_access_option>, - Z2_vector_chain_column::Cell_type, matrix_z2_ra::Row_access_option>, - Z2_intrusive_list_chain_column::Cell_type, matrix_z2_ra_i::Row_access_option>, - Z2_intrusive_list_chain_column::Cell_type, matrix_z2_ra::Row_access_option>, - Z2_intrusive_set_chain_column::Cell_type, matrix_z2_ra_i::Row_access_option>, - Z2_intrusive_set_chain_column::Cell_type, matrix_z2_ra::Row_access_option> - > list_of_pairing_chain_columns_with_row; - template void common_5_test(std::vector &matrix) { @@ -551,9 +562,9 @@ void common_5_test(std::vector &matrix) ++it; BOOST_CHECK(it == rows.end()); - BOOST_CHECK_EQUAL(matrix[0].get_dimension(), 4); - BOOST_CHECK_EQUAL(matrix[1].get_dimension(), 4); - BOOST_CHECK_EQUAL(matrix[2].get_dimension(), 4); + // BOOST_CHECK_EQUAL(matrix[0].get_dimension(), 4); + // BOOST_CHECK_EQUAL(matrix[1].get_dimension(), 4); + // BOOST_CHECK_EQUAL(matrix[2].get_dimension(), 4); BOOST_CHECK(!matrix[0].is_empty()); BOOST_CHECK(!matrix[1].is_empty()); BOOST_CHECK(!matrix[2].is_empty()); @@ -585,8 +596,8 @@ void common_5_test(std::vector &matrix) BOOST_CHECK(matrix[1].begin() == matrix[1].end()); - BOOST_CHECK_EQUAL(matrix[0].get_dimension(), 4); - BOOST_CHECK_EQUAL(matrix[1].get_dimension(), 4); + // BOOST_CHECK_EQUAL(matrix[0].get_dimension(), 4); + // BOOST_CHECK_EQUAL(matrix[1].get_dimension(), 4); BOOST_CHECK(!matrix[0].is_empty()); BOOST_CHECK(matrix[1].is_empty()); BOOST_CHECK(matrix[0].is_non_zero(1)); @@ -594,7 +605,7 @@ void common_5_test(std::vector &matrix) matrix[0] *= 4; matrix[1] *= 2; - matrix[2] *= 0; + matrix[2] *= 1; rows.clear(); for (auto c : matrix[0]){ @@ -616,13 +627,13 @@ void common_5_test(std::vector &matrix) BOOST_CHECK(it == rows.end()); BOOST_CHECK(matrix[1].begin() == matrix[1].end()); - BOOST_CHECK(matrix[2].begin() == matrix[2].end()); + BOOST_CHECK(matrix[2].begin() != matrix[2].end()); - BOOST_CHECK_EQUAL(matrix[0].get_dimension(), 4); - BOOST_CHECK_EQUAL(matrix[1].get_dimension(), 4); + // BOOST_CHECK_EQUAL(matrix[0].get_dimension(), 4); + // BOOST_CHECK_EQUAL(matrix[1].get_dimension(), 4); BOOST_CHECK(!matrix[0].is_empty()); BOOST_CHECK(matrix[1].is_empty()); - BOOST_CHECK(matrix[2].is_empty()); + BOOST_CHECK(!matrix[2].is_empty()); BOOST_CHECK(matrix[0].is_non_zero(1)); BOOST_CHECK(!matrix[1].is_non_zero(2)); BOOST_CHECK(!matrix[2].is_non_zero(3)); @@ -688,13 +699,12 @@ BOOST_AUTO_TEST_CASE_TEMPLATE(Column_types_common_5, Column, list_of_5_columns) BOOST_AUTO_TEST_CASE_TEMPLATE(Chain_column_types_common_5, Column, list_of_5_chain_columns) { std::vector matrix; - dict_type dict(7); - matrix.push_back(Column(std::vector >{{0,Z5(1)},{1,Z5(2)},{3,Z5(3)},{5,Z5(4)}}, 4, dict)); - matrix.push_back(Column(std::vector >{{0,Z5(4)},{1,Z5(2)},{2,Z5(1)},{5,Z5(1)},{6,Z5(1)}}, 4, dict)); - matrix.push_back(Column(std::vector >{{0,Z5(1)},{1,Z5(3)},{2,Z5(4)},{5,Z5(4)},{6,Z5(4)}}, 4, dict)); - matrix.push_back(Column(dict)); - matrix.push_back(Column(std::vector >{{0,Z5(1)},{1,Z5(2)},{3,Z5(3)},{5,Z5(4)}}, 4, dict)); + matrix.push_back(Column(std::vector >{{0,Z5(1)},{1,Z5(2)},{3,Z5(3)},{5,Z5(4)}}, 4)); + matrix.push_back(Column(std::vector >{{0,Z5(4)},{1,Z5(2)},{2,Z5(1)},{5,Z5(1)},{6,Z5(1)}}, 4)); + matrix.push_back(Column(std::vector >{{0,Z5(1)},{1,Z5(3)},{2,Z5(4)},{5,Z5(4)},{6,Z5(4)}}, 4)); + matrix.push_back(Column()); + matrix.push_back(Column(std::vector >{{0,Z5(1)},{1,Z5(2)},{3,Z5(3)},{5,Z5(4)}}, 4)); matrix.push_back(Column(matrix[1])); Column add = matrix[4] + matrix[5]; @@ -730,9 +740,9 @@ void common_5_test_with_rows(Rows &rows){ matrix.push_back(Column(0, std::vector >{{0,Z5(1)},{1,Z5(2)},{3,Z5(3)},{5,Z5(4)}}, 4, rows)); matrix.push_back(Column(1, std::vector >{{0,Z5(4)},{1,Z5(2)},{2,Z5(1)},{5,Z5(1)},{6,Z5(1)}}, 4, rows)); matrix.push_back(Column(2, std::vector >{{0,Z5(1)},{1,Z5(3)},{2,Z5(4)},{5,Z5(4)},{6,Z5(4)}}, 4, rows)); - matrix.push_back(Column(3, rows)); + matrix.push_back(Column(3, std::vector >{}, rows)); matrix.push_back(Column(4, std::vector >{{0,Z5(1)},{1,Z5(2)},{3,Z5(3)},{5,Z5(4)}}, rows)); - matrix.push_back(Column(matrix[1], 5u)); + matrix.push_back(Column(matrix[1], 5, rows)); common_5_test(matrix); } @@ -745,14 +755,13 @@ BOOST_AUTO_TEST_CASE_TEMPLATE(Column_types_with_row_common_5_non_intr, Column, l template void common_5_chain_test_with_rows(Rows &rows){ std::vector matrix; - dict_type dict(7); - matrix.push_back(Column(0, std::vector >{{0,Z5(1)},{1,Z5(2)},{3,Z5(3)},{5,Z5(4)}}, 4, rows, dict)); - matrix.push_back(Column(1, std::vector >{{0,Z5(4)},{1,Z5(2)},{2,Z5(1)},{5,Z5(1)},{6,Z5(1)}}, 4, rows, dict)); - matrix.push_back(Column(2, std::vector >{{0,Z5(1)},{1,Z5(3)},{2,Z5(4)},{5,Z5(4)},{6,Z5(4)}}, 4, rows, dict)); - matrix.push_back(Column(3, rows, dict)); - matrix.push_back(Column(4, std::vector >{{0,Z5(1)},{1,Z5(2)},{3,Z5(3)},{5,Z5(4)}}, 4, rows, dict)); - matrix.push_back(Column(matrix[1], 5u)); + matrix.push_back(Column(0, std::vector >{{0,Z5(1)},{1,Z5(2)},{3,Z5(3)},{5,Z5(4)}}, 4, rows)); + matrix.push_back(Column(1, std::vector >{{0,Z5(4)},{1,Z5(2)},{2,Z5(1)},{5,Z5(1)},{6,Z5(1)}}, 4, rows)); + matrix.push_back(Column(2, std::vector >{{0,Z5(1)},{1,Z5(3)},{2,Z5(4)},{5,Z5(4)},{6,Z5(4)}}, 4, rows)); + matrix.push_back(Column(3, std::vector >{}, 4, rows)); + matrix.push_back(Column(4, std::vector >{{0,Z5(1)},{1,Z5(2)},{3,Z5(3)},{5,Z5(4)}}, 4, rows)); + matrix.push_back(Column(matrix[1], 5, rows)); common_5_test(matrix); } @@ -814,9 +823,9 @@ void common_2_test(std::vector &matrix) ++it; BOOST_CHECK(it == rows.end()); - BOOST_CHECK_EQUAL(matrix[0].get_dimension(), 4); - BOOST_CHECK_EQUAL(matrix[1].get_dimension(), 4); - BOOST_CHECK_EQUAL(matrix[2].get_dimension(), 4); + // BOOST_CHECK_EQUAL(matrix[0].get_dimension(), 4); + // BOOST_CHECK_EQUAL(matrix[1].get_dimension(), 4); + // BOOST_CHECK_EQUAL(matrix[2].get_dimension(), 4); BOOST_CHECK(!matrix[0].is_empty()); BOOST_CHECK(!matrix[1].is_empty()); BOOST_CHECK(!matrix[2].is_empty()); @@ -843,15 +852,15 @@ void common_2_test(std::vector &matrix) BOOST_CHECK(matrix[1].is_empty()); //forces heap column to prune. BOOST_CHECK(matrix[1].begin() == matrix[1].end()); - BOOST_CHECK_EQUAL(matrix[0].get_dimension(), 4); - BOOST_CHECK_EQUAL(matrix[1].get_dimension(), 4); + // BOOST_CHECK_EQUAL(matrix[0].get_dimension(), 4); + // BOOST_CHECK_EQUAL(matrix[1].get_dimension(), 4); BOOST_CHECK(!matrix[0].is_empty()); BOOST_CHECK(matrix[1].is_empty()); BOOST_CHECK(matrix[0].is_non_zero(2)); BOOST_CHECK(!matrix[1].is_non_zero(2)); matrix[0] *= 5; - matrix[2] *= 8; + // matrix[2] *= 8; rows.clear(); for (auto c : matrix[0]){ @@ -866,16 +875,16 @@ void common_2_test(std::vector &matrix) ++it; BOOST_CHECK(it == rows.end()); - BOOST_CHECK(matrix[2].begin() == matrix[2].end()); + // BOOST_CHECK(matrix[2].begin() == matrix[2].end()); - BOOST_CHECK_EQUAL(matrix[0].get_dimension(), 4); - BOOST_CHECK_EQUAL(matrix[1].get_dimension(), 4); + // BOOST_CHECK_EQUAL(matrix[0].get_dimension(), 4); + // BOOST_CHECK_EQUAL(matrix[1].get_dimension(), 4); BOOST_CHECK(!matrix[0].is_empty()); BOOST_CHECK(matrix[1].is_empty()); - BOOST_CHECK(matrix[2].is_empty()); + // BOOST_CHECK(matrix[2].is_empty()); BOOST_CHECK(matrix[0].is_non_zero(2)); BOOST_CHECK(!matrix[1].is_non_zero(2)); - BOOST_CHECK(!matrix[2].is_non_zero(3)); + // BOOST_CHECK(!matrix[2].is_non_zero(3)); swap(matrix[3], matrix[4]); auto res = matrix[4].get_content(7); @@ -940,13 +949,12 @@ BOOST_AUTO_TEST_CASE_TEMPLATE(Column_types_common_2, Column, list_of_2_columns) BOOST_AUTO_TEST_CASE_TEMPLATE(Chain_column_types_common_2, Column, list_of_2_chain_columns) { std::vector matrix; - dict_type dict(7); - matrix.push_back(Column(std::vector{0,1,3,5}, 4, dict)); - matrix.push_back(Column(std::vector{0,1,2,5,6}, 4, dict)); - matrix.push_back(Column(std::vector{0,1,2,5,6}, 4, dict)); - matrix.push_back(Column(dict)); - matrix.push_back(Column(std::vector{0,1,3,5}, 4, dict)); + matrix.push_back(Column(std::vector{0,1,3,5}, 4)); + matrix.push_back(Column(std::vector{0,1,2,5,6}, 4)); + matrix.push_back(Column(std::vector{0,1,2,5,6}, 4)); + matrix.push_back(Column()); + matrix.push_back(Column(std::vector{0,1,3,5}, 4)); matrix.push_back(Column(matrix[1])); Column add = matrix[4] + matrix[5]; @@ -969,11 +977,6 @@ BOOST_AUTO_TEST_CASE_TEMPLATE(Chain_column_types_common_2, Column, list_of_2_cha BOOST_CHECK_EQUAL(res[5], 1); BOOST_CHECK_EQUAL(res[6], 0); - Column mul2 = 4 * matrix[4]; - BOOST_CHECK(mul2.is_empty()); - for (auto f : mul2.get_content(7)) - BOOST_CHECK_EQUAL(f, 0); - common_2_test(matrix); } @@ -984,9 +987,9 @@ void common_2_test_with_rows(Rows &rows){ matrix.push_back(Column(0, std::vector{0,1,3,5}, 4, rows)); matrix.push_back(Column(1, std::vector{0,1,2,5,6}, 4, rows)); matrix.push_back(Column(2, std::vector{0,1,2,5,6}, 4, rows)); - matrix.push_back(Column(3, rows)); + matrix.push_back(Column(3, std::vector{}, 4, rows)); matrix.push_back(Column(4, std::vector{0,1,3,5}, rows)); - matrix.push_back(Column(matrix[1], 5u)); + matrix.push_back(Column(matrix[1], 5, rows)); common_2_test(matrix); } @@ -999,14 +1002,13 @@ BOOST_AUTO_TEST_CASE_TEMPLATE(Column_types_with_row_common_2_non_intr, Column, l template void common_2_chain_test_with_rows(Rows &rows){ std::vector matrix; - dict_type dict(7); - matrix.push_back(Column(0, std::vector{0,1,3,5}, 4, rows, dict)); - matrix.push_back(Column(1, std::vector{0,1,2,5,6}, 4, rows, dict)); - matrix.push_back(Column(2, std::vector{0,1,2,5,6}, 4, rows, dict)); - matrix.push_back(Column(3, rows, dict)); - matrix.push_back(Column(4, std::vector{0,1,3,5}, 4, rows, dict)); - matrix.push_back(Column(matrix[1], 5u)); + matrix.push_back(Column(0, std::vector{0,1,3,5}, 4, rows)); + matrix.push_back(Column(1, std::vector{0,1,2,5,6}, 4, rows)); + matrix.push_back(Column(2, std::vector{0,1,2,5,6}, 4, rows)); + matrix.push_back(Column(3, std::vector{}, 4, rows)); + matrix.push_back(Column(4, std::vector{0,1,3,5}, 4, rows)); + matrix.push_back(Column(matrix[1], 5, rows)); common_2_test(matrix); } @@ -1163,13 +1165,12 @@ BOOST_AUTO_TEST_CASE_TEMPLATE(Z2_boundary_column_types_methods, Column, list_of_ BOOST_AUTO_TEST_CASE_TEMPLATE(Chain_column_types_methods, Column, list_of_5_chain_columns) { std::vector matrix; - dict_type pivotToColumnIndex{0,1,2,3}; - matrix.push_back(Column(std::vector >{{0,Z5(1)}}, 4, pivotToColumnIndex)); - matrix.push_back(Column(std::vector >{{0,Z5(4)}, {1,Z5(3)}}, 4, pivotToColumnIndex)); - matrix.push_back(Column(std::vector >{{1,Z5(2)},{2,Z5(4)}}, 4, pivotToColumnIndex)); - matrix.push_back(Column(std::vector >{{0,Z5(2)},{2,Z5(3)},{3,Z5(1)}}, 4, pivotToColumnIndex)); - matrix.push_back(Column(pivotToColumnIndex)); + matrix.push_back(Column(std::vector >{{0,Z5(1)}}, 4)); + matrix.push_back(Column(std::vector >{{0,Z5(4)}, {1,Z5(3)}}, 4)); + matrix.push_back(Column(std::vector >{{1,Z5(2)},{2,Z5(4)}}, 4)); + matrix.push_back(Column(std::vector >{{0,Z5(2)},{2,Z5(3)},{3,Z5(1)}}, 4)); + matrix.push_back(Column()); matrix.push_back(matrix[0]); BOOST_CHECK_EQUAL(matrix[0].get_pivot(), 0); @@ -1185,28 +1186,22 @@ BOOST_AUTO_TEST_CASE_TEMPLATE(Chain_column_types_methods, Column, list_of_5_chai BOOST_CHECK_EQUAL(matrix[4].get_pivot_value(), 0u); BOOST_CHECK_EQUAL(matrix[5].get_pivot_value(), 1u); - BOOST_CHECK_EQUAL(pivotToColumnIndex[0], 0); - BOOST_CHECK_EQUAL(pivotToColumnIndex[1], 1); - matrix[0] += matrix[1]; BOOST_CHECK_EQUAL(matrix[0].get_pivot(), 1); BOOST_CHECK_EQUAL(matrix[1].get_pivot(), 0); BOOST_CHECK_EQUAL(matrix[0].get_pivot_value(), 3u); BOOST_CHECK_EQUAL(matrix[1].get_pivot_value(), 4u); - BOOST_CHECK_EQUAL(pivotToColumnIndex[0], 1); - BOOST_CHECK_EQUAL(pivotToColumnIndex[1], 0); } BOOST_AUTO_TEST_CASE_TEMPLATE(Z2_chain_column_types_methods, Column, list_of_2_chain_columns) { std::vector matrix; - dict_type pivotToColumnIndex{0,1,2,3}; - matrix.push_back(Column(std::vector{0}, 4, pivotToColumnIndex)); - matrix.push_back(Column(std::vector{0,1}, 4, pivotToColumnIndex)); - matrix.push_back(Column(std::vector{1,2}, 4, pivotToColumnIndex)); - matrix.push_back(Column(std::vector{0,2,3}, 4, pivotToColumnIndex)); - matrix.push_back(Column(pivotToColumnIndex)); + matrix.push_back(Column(std::vector{0}, 4)); + matrix.push_back(Column(std::vector{0,1}, 4)); + matrix.push_back(Column(std::vector{1,2}, 4)); + matrix.push_back(Column(std::vector{0,2,3}, 4)); + matrix.push_back(Column()); matrix.push_back(matrix[0]); BOOST_CHECK_EQUAL(matrix[0].get_pivot(), 0); @@ -1216,15 +1211,10 @@ BOOST_AUTO_TEST_CASE_TEMPLATE(Z2_chain_column_types_methods, Column, list_of_2_c BOOST_CHECK_EQUAL(matrix[4].get_pivot(), -1); BOOST_CHECK_EQUAL(matrix[5].get_pivot(), 0); - BOOST_CHECK_EQUAL(pivotToColumnIndex[0], 0); - BOOST_CHECK_EQUAL(pivotToColumnIndex[1], 1); - matrix[0] += matrix[1]; BOOST_CHECK_EQUAL(matrix[0].get_pivot(), 1); BOOST_CHECK_EQUAL(matrix[1].get_pivot(), 0); - BOOST_CHECK_EQUAL(pivotToColumnIndex[0], 1); - BOOST_CHECK_EQUAL(pivotToColumnIndex[1], 0); } template @@ -1291,8 +1281,8 @@ void row_methods_test(Row_container_type& row_cont) { for (unsigned int i = 0; i < 4; ++i){ matrix.push_back(Column(i, chains[i], 4, row_cont)); } - matrix.push_back(Column(4, row_cont)); - matrix.push_back(Column(matrix[0], 5u)); + matrix.push_back(Column(4, std::vector >{}, row_cont)); + matrix.push_back(Column(matrix[0], 5, row_cont)); row_methods_test_com(matrix, chains, row_cont); } @@ -1305,7 +1295,6 @@ BOOST_AUTO_TEST_CASE_TEMPLATE(Column_types_with_row_methods_non_intr, Column, li template void chain_row_methods_test(Row_container_type& row_cont) { std::vector matrix; - dict_type dict(7); std::vector > > chains; chains.push_back({{0,Z5(1)}}); @@ -1316,10 +1305,10 @@ void chain_row_methods_test(Row_container_type& row_cont) { chains.push_back({{0,Z5(1)}}); for (unsigned int i = 0; i < 4; ++i){ - matrix.push_back(Column(i, chains[i], 4, row_cont, dict)); + matrix.push_back(Column(i, chains[i], 4, row_cont)); } - matrix.push_back(Column(4, row_cont, dict)); - matrix.push_back(Column(matrix[0], 5u)); + matrix.push_back(Column(4, std::vector >{}, 4, row_cont)); + matrix.push_back(Column(matrix[0], 5, row_cont)); row_methods_test_com(matrix, chains, row_cont); } @@ -1391,8 +1380,8 @@ void z2_row_methods_test(Row_container_type& row_cont) { for (unsigned int i = 0; i < 4; ++i){ matrix.push_back(Column(i, chains[i], 4, row_cont)); } - matrix.push_back(Column(4, row_cont)); - matrix.push_back(Column(matrix[0], 5u)); + matrix.push_back(Column(4, std::vector{}, 4, row_cont)); + matrix.push_back(Column(matrix[0], 5, row_cont)); z2_row_methods_test_com(matrix, chains, row_cont); } @@ -1405,7 +1394,6 @@ BOOST_AUTO_TEST_CASE_TEMPLATE(Z2_column_types_with_row_methods_non_intr, Column, template void z2_chain_row_methods_test(Row_container_type& row_cont) { std::vector matrix; - dict_type dict(7); std::vector > chains; chains.push_back({0}); @@ -1416,10 +1404,10 @@ void z2_chain_row_methods_test(Row_container_type& row_cont) { chains.push_back({0}); for (unsigned int i = 0; i < 4; ++i){ - matrix.push_back(Column(i, chains[i], 4, row_cont, dict)); + matrix.push_back(Column(i, chains[i], 4, row_cont)); } - matrix.push_back(Column(4, row_cont, dict)); - matrix.push_back(Column(matrix[0], 5u)); + matrix.push_back(Column(4, std::vector{}, 4, row_cont)); + matrix.push_back(Column(matrix[0], 5, row_cont)); z2_row_methods_test_com(matrix, chains, row_cont); } @@ -1460,27 +1448,10 @@ void pairing_test(std::vector &matrix) BOOST_AUTO_TEST_CASE_TEMPLATE(Chain_column_types_pairing_option, Column, list_of_chain_pairing_columns) { std::vector matrix; - dict_type pivotToColumnIndex{0,0,0,0}; for (int i = 0; i < 4; ++i) - matrix.push_back(Column(pivotToColumnIndex)); + matrix.push_back(Column()); pairing_test(matrix); } -template -void pairing_test_with_chain_rows(Rows &rows){ - std::vector matrix; - dict_type dict(7); - - for (int i = 0; i < 4; ++i) - matrix.push_back(Column(i, rows, dict)); - - pairing_test(matrix); -} - -BOOST_AUTO_TEST_CASE_TEMPLATE(Chain_column_types_with_row_pairing_option_non_intr, Column, list_of_pairing_chain_columns_with_row) { - typename Column::Row_access::Row_container_type rows; - pairing_test_with_chain_rows(rows); -} - diff --git a/src/Persistence_matrix/test/Persistence_matrix_common_unit_test.cpp b/src/Persistence_matrix/test/Persistence_matrix_common_unit_test.cpp index 68c4abb276..6abd576055 100644 --- a/src/Persistence_matrix/test/Persistence_matrix_common_unit_test.cpp +++ b/src/Persistence_matrix/test/Persistence_matrix_common_unit_test.cpp @@ -385,64 +385,64 @@ typedef boost::mpl::list >, Matrix >*/ > list_of_matrix_types_chain_with_row_access_no_removals; -typedef boost::mpl::list >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix > +typedef boost::mpl::list >, + Matrix >, + Matrix >, + Matrix >, + Matrix >, + Matrix >, + Matrix >, + Matrix >, + Matrix >, + Matrix >, + Matrix >, + Matrix >, + Matrix >, + Matrix >, + Matrix >, + Matrix >, + Matrix >, + Matrix >, + Matrix >, + Matrix >, + Matrix >, + Matrix >, + Matrix >, + Matrix > > list_of_matrix_types; -typedef boost::mpl::list >, +typedef boost::mpl::list >, Matrix >, Matrix >, Matrix >, Matrix >, Matrix >, Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, + Matrix >, */ + Matrix >, + Matrix >, + Matrix >, + Matrix >, + Matrix >, + Matrix >, + Matrix >, + Matrix >, + Matrix >, + Matrix >, + // Matrix >, + // Matrix >, + Matrix >, + Matrix >, + // Matrix >, + // Matrix >, + Matrix >, + Matrix >, + // Matrix >, + // Matrix >, + Matrix >, + Matrix >/* , Matrix >, - Matrix > + Matrix > */ > list_of_matrix_types2; template @@ -512,30 +512,30 @@ BOOST_AUTO_TEST_CASE_TEMPLATE(Matrix_constructors_bis, Matrix, list_of_matrix_ty test_constructors(); } -typedef boost::mpl::list >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix > +typedef boost::mpl::list >, + Matrix >, + Matrix >, + Matrix >, + Matrix >, + Matrix >, + Matrix >, + Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + Matrix >, + Matrix >, + Matrix >, + Matrix >, + Matrix >, + Matrix >, + Matrix >, + Matrix > > list_of_base_matrix_types; BOOST_AUTO_TEST_CASE_TEMPLATE(Base_matrix_methods, Matrix, list_of_base_matrix_types) { @@ -598,14 +598,14 @@ BOOST_AUTO_TEST_CASE_TEMPLATE(Base_matrix_methods, Matrix, list_of_base_matrix_t BOOST_CHECK(m.is_zero_column(3)); } -typedef boost::mpl::list >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix > +typedef boost::mpl::list >, + Matrix >, + Matrix >, + Matrix >, + Matrix >, + Matrix >, + Matrix >, + Matrix > > list_of_boundary_matrix_types; BOOST_AUTO_TEST_CASE_TEMPLATE(Boundary_matrix_methods, Matrix, list_of_boundary_matrix_types) { @@ -671,30 +671,30 @@ BOOST_AUTO_TEST_CASE_TEMPLATE(Boundary_matrix_methods, Matrix, list_of_boundary_ BOOST_CHECK_EQUAL(m.get_column_with_pivot(5), 6); } -typedef boost::mpl::list >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, +typedef boost::mpl::list >, + Matrix >, + Matrix >, + Matrix >, + Matrix >, + Matrix >, + Matrix >, + Matrix >, + Matrix >, + Matrix >, + // Matrix >, + // Matrix >, + Matrix >, + Matrix >, + // Matrix >, + // Matrix >, + Matrix >, + Matrix >, + // Matrix >, + // Matrix >, + Matrix >, + Matrix >/* , Matrix >, - Matrix > + Matrix > */ > list_of_chain_matrix_types; BOOST_AUTO_TEST_CASE_TEMPLATE(Chain_matrix_methods, Matrix, list_of_chain_matrix_types) { @@ -766,31 +766,31 @@ BOOST_AUTO_TEST_CASE_TEMPLATE(Chain_matrix_methods, Matrix, list_of_chain_matrix BOOST_CHECK_EQUAL(m.get_column_with_pivot(6), 6); } -typedef boost::mpl::list >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix > +typedef boost::mpl::list >, + Matrix >, + Matrix >, + Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + Matrix >, + Matrix >, + Matrix >, + Matrix > > list_of_Z2_base_matrix_types; -typedef boost::mpl::list >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix > +typedef boost::mpl::list >, + Matrix >, + Matrix >, + Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + Matrix >, + Matrix >, + Matrix >, + Matrix > > list_of_Z5_base_matrix_types; BOOST_AUTO_TEST_CASE_TEMPLATE(Z2_base_matrix_methods, Matrix, list_of_Z2_base_matrix_types) { @@ -862,15 +862,15 @@ BOOST_AUTO_TEST_CASE_TEMPLATE(Z5_base_matrix_methods, Matrix, list_of_Z5_base_ma BOOST_CHECK_EQUAL(rowIndices.rbegin()->second, 4); } -typedef boost::mpl::list >, - Matrix >, - Matrix >, - Matrix > +typedef boost::mpl::list >, + Matrix >, + Matrix >, + Matrix > > list_of_Z2_boundary_matrix_types; -typedef boost::mpl::list >, - Matrix >, - Matrix >, - Matrix > +typedef boost::mpl::list >, + Matrix >, + Matrix >, + Matrix > > list_of_Z5_boundary_matrix_types; BOOST_AUTO_TEST_CASE_TEMPLATE(Z2_boundary_matrix_methods, Matrix, list_of_Z2_boundary_matrix_types) { @@ -948,31 +948,31 @@ BOOST_AUTO_TEST_CASE_TEMPLATE(Z5_boundary_matrix_methods, Matrix, list_of_Z5_bou BOOST_CHECK_EQUAL(rowIndices.rbegin()->second, 4); } -typedef boost::mpl::list >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix > +typedef boost::mpl::list >, + Matrix >, + Matrix >, + Matrix >, + Matrix >, + // Matrix >, + Matrix >, + // Matrix >, + Matrix >, + // Matrix >, + Matrix >/* , + Matrix > */ > list_of_Z2_chain_matrix_types; -typedef boost::mpl::list >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix > +typedef boost::mpl::list >, + Matrix >, + Matrix >, + Matrix >, + Matrix >, + // Matrix >, + Matrix >, + // Matrix >, + Matrix >, + // Matrix >, + Matrix >/* , + Matrix > */ > list_of_Z5_chain_matrix_types; BOOST_AUTO_TEST_CASE_TEMPLATE(Z2_chain_matrix_methods, Matrix, list_of_Z2_chain_matrix_types) { @@ -1061,14 +1061,14 @@ BOOST_AUTO_TEST_CASE_TEMPLATE(Z5_chain_matrix_methods, Matrix, list_of_Z5_chain_ BOOST_CHECK_EQUAL(rowIndices.rbegin()->second, 1); } -typedef boost::mpl::list >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix > +typedef boost::mpl::list >, + // Matrix >, + Matrix >, + // Matrix >, + Matrix >, + // Matrix >, + Matrix >/* , + Matrix > */ > list_of_z5_chain_options_with_row_access; BOOST_AUTO_TEST_CASE_TEMPLATE(Chain_row_access_columns_options, Matrix, list_of_z5_chain_options_with_row_access) { @@ -1108,14 +1108,14 @@ BOOST_AUTO_TEST_CASE_TEMPLATE(Chain_row_access_columns_options, Matrix, list_of_ } } -typedef boost::mpl::list >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix > +typedef boost::mpl::list >, + // Matrix >, + Matrix >, + // Matrix >, + Matrix >, + // Matrix >, + Matrix >/* , + Matrix > */ > list_of_z2_chain_options_with_row_access; BOOST_AUTO_TEST_CASE_TEMPLATE(Z2_chain_row_access_columns_options, Matrix, list_of_z2_chain_options_with_row_access) { @@ -1154,14 +1154,14 @@ BOOST_AUTO_TEST_CASE_TEMPLATE(Z2_chain_row_access_columns_options, Matrix, list_ } } -typedef boost::mpl::list >, +typedef boost::mpl::list >, Matrix >, Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix > + Matrix >, */ + Matrix >, + Matrix >, + Matrix >, + Matrix > > list_of_z5_base_options_with_row_access; BOOST_AUTO_TEST_CASE_TEMPLATE(Base_row_access_columns_options, Matrix, list_of_z5_base_options_with_row_access) { @@ -1200,14 +1200,14 @@ BOOST_AUTO_TEST_CASE_TEMPLATE(Base_row_access_columns_options, Matrix, list_of_z } } -typedef boost::mpl::list >, +typedef boost::mpl::list >, Matrix >, Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix > + Matrix >, */ + Matrix >, + Matrix >, + Matrix >, + Matrix > > list_of_z2_base_options_with_row_access; BOOST_AUTO_TEST_CASE_TEMPLATE(Z2_base_row_access_columns_options, Matrix, list_of_z2_base_options_with_row_access) { @@ -1245,34 +1245,34 @@ BOOST_AUTO_TEST_CASE_TEMPLATE(Z2_base_row_access_columns_options, Matrix, list_o } } -typedef boost::mpl::list >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix > +typedef boost::mpl::list >, + Matrix >, + // Matrix >, + // Matrix >, + Matrix >, + Matrix >, + // Matrix >, + // Matrix >, + Matrix >, + Matrix >, + Matrix >, + Matrix >, + Matrix >, + Matrix >, + Matrix >, + Matrix >, + Matrix >, + Matrix >, + Matrix >, + Matrix >, + // Matrix >, + // Matrix >, + Matrix >, + Matrix >, + // Matrix >, + // Matrix >, + Matrix >, + Matrix > > list_of_options_with_remove; BOOST_AUTO_TEST_CASE_TEMPLATE(Removable_columns_options, Matrix, list_of_options_with_remove) { diff --git a/src/Persistence_matrix/test/Persistence_matrix_specialized_unit_test.cpp b/src/Persistence_matrix/test/Persistence_matrix_specialized_unit_test.cpp index 7477afc58c..d8bfa0cc39 100644 --- a/src/Persistence_matrix/test/Persistence_matrix_specialized_unit_test.cpp +++ b/src/Persistence_matrix/test/Persistence_matrix_specialized_unit_test.cpp @@ -191,14 +191,14 @@ void build_boundary_matrix(std::vector >, +typedef boost::mpl::list >, Matrix >, Matrix >, Matrix >, Matrix >, Matrix >, Matrix >, - Matrix >, + Matrix >, */ Matrix >, Matrix >, Matrix >, @@ -206,7 +206,7 @@ typedef boost::mpl::list >, Matrix >, Matrix >, Matrix >, - Matrix >, + Matrix >/* , Matrix >, Matrix >, Matrix >, @@ -240,7 +240,7 @@ typedef boost::mpl::list >, Matrix >, Matrix >, Matrix >, - Matrix > + Matrix > */ > list_of_options_with_barcode_access; BOOST_AUTO_TEST_CASE_TEMPLATE(Barcode_options, Matrix, list_of_options_with_barcode_access) { @@ -311,14 +311,14 @@ BOOST_AUTO_TEST_CASE_TEMPLATE(Barcode_options, Matrix, list_of_options_with_barc } } -typedef boost::mpl::list >, +typedef boost::mpl::list >, Matrix >, Matrix >, Matrix >, Matrix >, Matrix >, Matrix >, - Matrix >, + Matrix >, */ Matrix >, Matrix >, Matrix >, @@ -326,7 +326,7 @@ typedef boost::mpl::list >, Matrix >, Matrix >, Matrix >, - Matrix >, + Matrix >/* , Matrix >, Matrix >, Matrix >, @@ -360,7 +360,7 @@ typedef boost::mpl::list >, Matrix >, Matrix >, Matrix >, - Matrix > + Matrix > */ > list_of_options_with_rep_cycles; BOOST_AUTO_TEST_CASE_TEMPLATE(Representative_cycle_options, Matrix, list_of_options_with_rep_cycles) { @@ -420,24 +420,24 @@ BOOST_AUTO_TEST_CASE_TEMPLATE(Representative_cycle_options, Matrix, list_of_opti } } -typedef boost::mpl::list >, - Matrix >, +typedef boost::mpl::list >, + Matrix >, */ Matrix >, Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, Matrix >, - Matrix >, + Matrix >/* , Matrix >, Matrix >, Matrix >, @@ -445,7 +445,7 @@ typedef boost::mpl::list >, Matrix >, Matrix >, Matrix >, - Matrix > + Matrix > */ > list_of_options_with_vine_and_position_index; BOOST_AUTO_TEST_CASE_TEMPLATE(Vine_option_with_position_indexing, Matrix, list_of_options_with_vine_and_position_index) { @@ -746,24 +746,24 @@ BOOST_AUTO_TEST_CASE_TEMPLATE(Vine_option_with_position_indexing, Matrix, list_o BOOST_CHECK(it == barcode.end()); } -typedef boost::mpl::list >, - Matrix >, +typedef boost::mpl::list >, + Matrix >, */ Matrix >, Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, Matrix >, - Matrix >, + Matrix >/* , Matrix >, Matrix >, Matrix >, @@ -771,7 +771,7 @@ typedef boost::mpl::list >, Matrix >, Matrix >, Matrix >, - Matrix > + Matrix > */ > list_of_options_with_vine_and_id_index; BOOST_AUTO_TEST_CASE_TEMPLATE(Vine_option_with_id_indexing, Matrix, list_of_options_with_vine_and_id_index) { @@ -1387,14 +1387,14 @@ struct opt_ra_ni_vine_r_ii : Default_options >, +typedef boost::mpl::list >, Matrix >, Matrix >, - Matrix >, + Matrix >, */ Matrix >, Matrix >, Matrix >, - Matrix >, + Matrix >/* , Matrix >, Matrix >, Matrix >, @@ -1410,7 +1410,7 @@ typedef boost::mpl::list >, Matrix >, Matrix >, Matrix >, - Matrix > + Matrix > */ > list_of_options_with_barcode_and_row_access; BOOST_AUTO_TEST_CASE_TEMPLATE(Barcode_and_row_access_options, Matrix, list_of_options_with_barcode_and_row_access) { @@ -1469,14 +1469,14 @@ BOOST_AUTO_TEST_CASE_TEMPLATE(Barcode_and_row_access_options, Matrix, list_of_op } } -typedef boost::mpl::list >, +typedef boost::mpl::list >, Matrix >, Matrix >, - Matrix >, + Matrix >, */ Matrix >, Matrix >, Matrix >, - Matrix >, + Matrix >/* , Matrix >, Matrix >, Matrix >, @@ -1492,7 +1492,7 @@ typedef boost::mpl::list >, Matrix >, Matrix >, Matrix >, - Matrix > + Matrix > */ > list_of_z2_options_with_barcode_and_row_access; BOOST_AUTO_TEST_CASE_TEMPLATE(Barcode_and_row_access_z2_options, Matrix, list_of_z2_options_with_barcode_and_row_access) { @@ -1550,14 +1550,14 @@ BOOST_AUTO_TEST_CASE_TEMPLATE(Barcode_and_row_access_z2_options, Matrix, list_of } } -typedef boost::mpl::list >, +typedef boost::mpl::list >, Matrix >, Matrix >, - Matrix >, + Matrix >, */ Matrix >, Matrix >, Matrix >, - Matrix >, + Matrix >/* , Matrix >, Matrix >, Matrix >, @@ -1573,7 +1573,7 @@ typedef boost::mpl::list >, Matrix >, Matrix >, Matrix >, - Matrix > + Matrix > */ > list_of_options_with_rep_cycles_and_row_access; BOOST_AUTO_TEST_CASE_TEMPLATE(Representative_cycle_and_row_access_options, Matrix, list_of_options_with_rep_cycles_and_row_access) { @@ -1633,14 +1633,14 @@ BOOST_AUTO_TEST_CASE_TEMPLATE(Representative_cycle_and_row_access_options, Matri } } -typedef boost::mpl::list >, +typedef boost::mpl::list >, Matrix >, Matrix >, - Matrix >, + Matrix >, */ Matrix >, Matrix >, Matrix >, - Matrix >, + Matrix >/* , Matrix >, Matrix >, Matrix >, @@ -1656,7 +1656,7 @@ typedef boost::mpl::list >, Matrix >, Matrix >, Matrix >, - Matrix > + Matrix > */ > list_of_z2_options_with_rep_cycles_and_row_access; BOOST_AUTO_TEST_CASE_TEMPLATE(Representative_cycle_and_row_access_z2_options, Matrix, list_of_z2_options_with_rep_cycles_and_row_access) { @@ -1715,22 +1715,22 @@ BOOST_AUTO_TEST_CASE_TEMPLATE(Representative_cycle_and_row_access_z2_options, Ma } } -typedef boost::mpl::list >, - Matrix >, +typedef boost::mpl::list >, + Matrix >, */ Matrix >, Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, Matrix >, - Matrix >, + Matrix >/* , Matrix >, Matrix >, Matrix >, @@ -1738,7 +1738,7 @@ typedef boost::mpl::list >, Matrix >, Matrix >, Matrix >, - Matrix > + Matrix > */ > list_of_options_with_vine_and_position_index_and_row_access; BOOST_AUTO_TEST_CASE_TEMPLATE(Vine_option_with_position_indexing_and_row_access, Matrix, list_of_options_with_vine_and_position_index_and_row_access) { @@ -1813,22 +1813,22 @@ BOOST_AUTO_TEST_CASE_TEMPLATE(Vine_option_with_position_indexing_and_row_access, } } -typedef boost::mpl::list >, - Matrix >, +typedef boost::mpl::list >, + Matrix >, */ Matrix >, Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, - Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, + // Matrix >, Matrix >, - Matrix >, + Matrix >/* , Matrix >, Matrix >, Matrix >, @@ -1836,7 +1836,7 @@ typedef boost::mpl::list Matrix >, Matrix >, Matrix >, - Matrix > + Matrix > */ > list_of_options_with_vine_and_id_index_and_row_access; BOOST_AUTO_TEST_CASE_TEMPLATE(Vine_option_with_id_indexing_and_row_access, Matrix, list_of_options_with_vine_and_id_index_and_row_access) {