From 241149ba6e85ad225d9c355e2033f0595e29a10d Mon Sep 17 00:00:00 2001 From: Nao Pross Date: Thu, 10 Oct 2019 01:18:49 +0200 Subject: Update access model for operator| The operator | can access the matrix either by directly changing the values (by reference, mm::mutate(M)) or by creating first a a copy of the matrix (mm::clone(M)). ToDo: The order of destruction of the mutate object is not yet well defined, and therefore it is not yet deterministic enough to work with expressions like (pseudocode) matrix m = a * b * (a | mm::alg::invert) because operator| (defaults to mutate), should not but could, change the value of a before the product a * b gets evaluated. --- include/mm/view.hpp | 134 +++++++++++++++++++++++++++++++++++----------------- 1 file changed, 90 insertions(+), 44 deletions(-) (limited to 'include/mm/view.hpp') diff --git a/include/mm/view.hpp b/include/mm/view.hpp index 910c16a..e701a78 100644 --- a/include/mm/view.hpp +++ b/include/mm/view.hpp @@ -1,66 +1,112 @@ #pragma once -#include +#include "mm/mmmatrix.hpp" +#include +#include +#include +#include -namespace mm::alg { - template < - template typename Matrix, - typename T, std::size_t Rows, std::size_t Cols - > - struct visitor - { - using type = T; +namespace mm { + + namespace algorithm { + // does nothing + struct visit + { + visit() = default; - // copy constructible - visitor(const visitor& other) = default; + template + void operator()(Matrix& m) {} + }; - T& operator()(const Matrix& m, index row, index col) { - return m.at(row, col); - } + struct transpose : public visit + { + /// does not work with non-square matrices + template + void operator()(Matrix& m) { + static_assert(Matrix::rows == Matrix::cols); + // naiive impl + for (index r = 0; r < m.rows / 2; r++) + for (index c = 0; c < m.cols; c++) + if (c != r) + std::swap(m.at(r, c), m.at(c, r)); + } + }; - const T& operator()(const Matrix& m, index row, index col) { - return operator()(m, row, col); - } - }; + /// algorithm aliases + using tr = transpose; + } + + /// namespace alias + namespace alg = algorithm; - template < - template typename Matrix, - typename T, std::size_t Rows, std::size_t Cols - > - struct transpose : public visitor + template + struct clone { - T& operator()(const Matrix m, index row, index col) { - // assert(col < Rows) - // assert(row < Cols) - return m.at(col, row); + Matrix matrix; + + explicit clone(Matrix&& m) : matrix(m) {} + explicit clone(const Matrix& m) : matrix(m) {} + + operator Matrix() { + return std::move(matrix); } }; -} -namespace mm { - template < - template typename Matrix, - typename T, std::size_t Rows, std::size_t Cols - > - struct view + template + struct mutate { - Matrix& m; - // std::stack> visitors; - std::unique_ptr visitor; + Matrix& matrix; + std::tuple visitors; + + explicit mutate(Matrix& m) : matrix(m) {} + + template + explicit mutate(Matrix& m, std::tuple&& t, Alg&& v) + : matrix(m) + { + /// append the new operator + visitors = std::tuple_cat(t, std::make_tuple(v)); + } + + ~mutate() { + visit(); + } - T& at(index row, index col) { - return visitor(m, row, col); + void visit() { + std::apply([this](auto&&... v) { + (v(matrix),...); + }, visitors); } - view& operator|=(const alg::visitor& other) { - // visitors.push(std::move(std::make_unique(other))); - visitor = std::make_unique(other); + operator Matrix() { + return std::move(matrix); } }; - view operator|(const view& left, const alg::visitor& right) { - return left |= right; + template + clone operator|(clone&& cl, Alg&& v) { + static_assert(std::is_convertible::value); + /// apply alg operator + v(cl.matrix); + /// forward to next alg + return clone(std::move(cl)); + } + + template + mutate operator|(mutate&& mut, Alg&& v) { + static_assert(std::is_convertible::value); + /// append alg to the visitors tuple + return mutate( + mut.matrix, + std::move(mut.visitors), + v + ); + } + + template + mutate operator|(Matrix& m, Alg&& v) { + return mutate(m) | std::move(v); } } -- cgit v1.2.1